diff --git a/package-lock.json b/package-lock.json index f4e5fd72..941dd0ce 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1853,7 +1853,6 @@ "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "@babel/code-frame": "^7.29.0", "@babel/generator": "^7.29.0", @@ -2666,7 +2665,6 @@ "resolved": "https://registry.npmjs.org/@elastic/elasticsearch/-/elasticsearch-8.19.1.tgz", "integrity": "sha512-+1j9NnQVOX+lbWB8LhCM7IkUmjU05Y4+BmSLfusq0msCsQb1Va+OUKFCoOXjCJqQrcgdRdQCjYYyolQ/npQALQ==", "license": "Apache-2.0", - "peer": true, "dependencies": { "@elastic/transport": "^8.9.6", "apache-arrow": "18.x - 21.x", @@ -2888,7 +2886,6 @@ "resolved": "https://registry.npmjs.org/@grpc/grpc-js/-/grpc-js-1.14.3.tgz", "integrity": "sha512-Iq8QQQ/7X3Sac15oB6p0FmUg/klxQvXLeileoqrTRGJYLV+/9tubbr9ipz0GKHjmXVsgFPo/+W+2cA8eNcR+XA==", "license": "Apache-2.0", - "peer": true, "dependencies": { "@grpc/proto-loader": "^0.8.0", "@js-sdsl/ordered-map": "^4.4.2" @@ -2902,7 +2899,6 @@ "resolved": "https://registry.npmjs.org/@grpc/proto-loader/-/proto-loader-0.8.0.tgz", "integrity": "sha512-rc1hOQtjIWGxcxpb9aHAfLpIctjEnsDehj0DAiVfBlmT84uvR0uUtN2hEi/ecvWVjXUGf5qPF4qEgiLOx1YIMQ==", "license": "Apache-2.0", - "peer": true, "dependencies": { "lodash.camelcase": "^4.3.0", "long": "^5.0.0", @@ -4473,7 +4469,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/axios/-/axios-4.0.1.tgz", "integrity": "sha512-68pFJgu+/AZbWkGu65Z3r55bTsCPlgyKaV4BSG8yUAD72q1PPuyVRgUwFv6BxdnibTUHlyxm06FmYWNC+bjN7A==", "license": "MIT", - "peer": true, "peerDependencies": { "@nestjs/common": "^10.0.0 || ^11.0.0", "axios": "^1.3.1", @@ -4720,7 +4715,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/common/-/common-10.4.22.tgz", "integrity": "sha512-fxJ4v85nDHaqT1PmfNCQ37b/jcv2OojtXTaK1P2uAXhzLf9qq6WNUOFvxBrV4fhQek1EQoT1o9oj5xAZmv3NRw==", "license": "MIT", - "peer": true, "dependencies": { "file-type": "20.4.1", "iterare": "1.2.1", @@ -4767,7 +4761,6 @@ "integrity": "sha512-6IX9+VwjiKtCjx+mXVPncpkQ5ZjKfmssOZPFexmT+6T9H9wZ3svpYACAo7+9e7Nr9DZSoRZw3pffkJP7Z0UjaA==", "hasInstallScript": true, "license": "MIT", - "peer": true, "dependencies": { "@nuxtjs/opencollective": "0.3.2", "fast-safe-stringify": "2.1.1", @@ -4829,7 +4822,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/graphql/-/graphql-12.2.2.tgz", "integrity": "sha512-lUDy/1uqbRA1kBKpXcmY0aHhcPbfeG52Wg5+9Jzd1d57dwSjCAmuO+mWy5jz9ugopVCZeK0S/kdAMvA+r9fNdA==", "license": "MIT", - "peer": true, "dependencies": { "@graphql-tools/merge": "9.0.11", "@graphql-tools/schema": "10.0.10", @@ -5018,7 +5010,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/platform-express/-/platform-express-10.4.22.tgz", "integrity": "sha512-ySSq7Py/DFozzZdNDH67m/vHoeVdphDniWBnl6q5QVoXldDdrZIHLXLRMPayTDh5A95nt7jjJzmD4qpTbNQ6tA==", "license": "MIT", - "peer": true, "dependencies": { "body-parser": "1.20.4", "cors": "2.8.5", @@ -5591,7 +5582,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/typeorm/-/typeorm-11.0.1.tgz", "integrity": "sha512-8rw/nKT0S+L+MkzgE9F2/mox7mAgsPlwfzmW9gsESN1lmQtIrVEfiiBwC2O8+guS1jBfQehJIdcdUj2OAp4VUQ==", "license": "MIT", - "peer": true, "peerDependencies": { "@nestjs/common": "^10.0.0 || ^11.0.0", "@nestjs/core": "^10.0.0 || ^11.0.0", @@ -5605,7 +5595,6 @@ "resolved": "https://registry.npmjs.org/@nestjs/websockets/-/websockets-10.4.22.tgz", "integrity": "sha512-OLd4i0Faq7vgdtB5vVUrJ54hWEtcXy9poJ6n7kbbh/5ms+KffUl+wwGsbe7uSXLrkoyI8xXU6fZPkFArI+XiRg==", "license": "MIT", - "peer": true, "dependencies": { "iterare": "1.2.1", "object-hash": "3.0.0", @@ -5738,7 +5727,6 @@ "resolved": "https://registry.npmjs.org/@opentelemetry/api/-/api-1.9.1.tgz", "integrity": "sha512-gLyJlPHPZYdAk1JENA9LeHejZe1Ti77/pTeFm/nMXmQH/HFZlcS/O2XJB+L8fkbrNSqhdtlvjBVjxwUYanNH5Q==", "license": "Apache-2.0", - "peer": true, "engines": { "node": ">=8.0.0" } @@ -6659,6 +6647,7 @@ "resolved": "https://registry.npmjs.org/@redis/bloom/-/bloom-5.12.1.tgz", "integrity": "sha512-PUUfv+ms7jgPSBVoo/DN4AkPHj4D5TZSd6SbJX7egzBplkYUcKmHRE8RKia7UtZ8bSQbLguLvxVO+asKtQfZWA==", "license": "MIT", + "peer": true, "engines": { "node": ">= 18.19.0" }, @@ -6696,6 +6685,7 @@ "resolved": "https://registry.npmjs.org/@redis/json/-/json-5.12.1.tgz", "integrity": "sha512-eOze75esLve4vfqDel7aMX08CNaiLLQS2fV8mpRN9NxPe1rVR4vQyYiW/OgtGUysF6QOr9ANhfxABKNOJfXdKg==", "license": "MIT", + "peer": true, "engines": { "node": ">= 18.19.0" }, @@ -6708,6 +6698,7 @@ "resolved": "https://registry.npmjs.org/@redis/search/-/search-5.12.1.tgz", "integrity": "sha512-ItlxbxC9cKI6IU1TLWoczwJCRb6TdmkEpWv05UrPawqaAnWGRu3rcIqsc5vN483T2fSociuyV1UkWIL5I4//2w==", "license": "MIT", + "peer": true, "engines": { "node": ">= 18.19.0" }, @@ -6720,6 +6711,7 @@ "resolved": "https://registry.npmjs.org/@redis/time-series/-/time-series-5.12.1.tgz", "integrity": "sha512-c6JL6E3EcZJuNqKFz+KM+l9l5mpcQiKvTwgA3blt5glWJ8hjDk0yeHN3beE/MpqYIQ8UEX44ItQzgkE/gCBELQ==", "license": "MIT", + "peer": true, "engines": { "node": ">= 18.19.0" }, @@ -7702,7 +7694,6 @@ "integrity": "sha512-FXx2pKgId/WyYo2jXw63kk7/+TY7u7AziEJxJAnSFzHlqTAS3Ync6SvgYAN/k4/PQpnnVuzoMuVnByKK2qp0ag==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "@types/estree": "*", "@types/json-schema": "*" @@ -7913,7 +7904,6 @@ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.39.tgz", "integrity": "sha512-orrrD74MBUyK8jOAD/r0+lfa1I2MO6I+vAkmAWzMYbCcgrN4lCrmK52gRFQq/JRxfYPfonkr4b0jcY7Olqdqbw==", "license": "MIT", - "peer": true, "dependencies": { "undici-types": "~6.21.0" } @@ -8144,7 +8134,6 @@ "integrity": "sha512-TI1XGwKbDpo9tRW8UDIXCOeLk55qe9ZFGs8MTKU6/M08HWTw52DD/IYhfQtOEhEdPhLMT26Ka/x7p70nd3dzDg==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "@typescript-eslint/scope-manager": "8.59.0", "@typescript-eslint/types": "8.59.0", @@ -8604,7 +8593,6 @@ "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", "license": "MIT", - "peer": true, "bin": { "acorn": "bin/acorn" }, @@ -8627,6 +8615,7 @@ "integrity": "sha512-wKmbr/DDiIXzEOiWrTTUcDm24kQ2vGfZQvM2fwg2vXqR5uW6aapr7ObPtj1th32b9u90/Pf4AItvdTh42fBmVQ==", "dev": true, "license": "MIT", + "peer": true, "engines": { "node": ">=10.13.0" }, @@ -8663,7 +8652,6 @@ "integrity": "sha512-PlXPeEWMXMZ7sPYOHqmDyCJzcfNrUr3fGNKtezX14ykXOEIvyK81d+qydx89KY5O71FKMPaQ2vBfBFI5NHR63A==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", @@ -8951,7 +8939,6 @@ "resolved": "https://registry.npmjs.org/axios/-/axios-1.15.2.tgz", "integrity": "sha512-wLrXxPtcrPTsNlJmKjkPnNPK2Ihe0hn0wGSaTEiHRPxwjvJwT3hKmXF4dpqxmPO9SoNb2FsYXj/xEo0gHN+D5A==", "license": "MIT", - "peer": true, "dependencies": { "follow-redirects": "^1.15.11", "form-data": "^4.0.5", @@ -9476,7 +9463,6 @@ } ], "license": "MIT", - "peer": true, "dependencies": { "baseline-browser-mapping": "^2.10.12", "caniuse-lite": "^1.0.30001782", @@ -9555,7 +9541,6 @@ "resolved": "https://registry.npmjs.org/bull/-/bull-4.16.5.tgz", "integrity": "sha512-lDsx2BzkKe7gkCYiT5Acj02DpTwDznl/VNN7Psn7M3USPG7Vs/BaClZJJTAG+ufAR9++N1/NiUTdaFBWDIl5TQ==", "license": "MIT", - "peer": true, "dependencies": { "cron-parser": "^4.9.0", "get-port": "^5.1.1", @@ -9603,7 +9588,6 @@ "resolved": "https://registry.npmjs.org/cache-manager/-/cache-manager-7.2.8.tgz", "integrity": "sha512-0HDaDLBBY/maa/LmUVAr70XUOwsiQD+jyzCBjmUErYZUKdMS9dT59PqW59PpVqfGM7ve6H0J6307JTpkCYefHQ==", "license": "MIT", - "peer": true, "dependencies": { "@cacheable/utils": "^2.3.3", "keyv": "^5.5.5" @@ -9650,7 +9634,6 @@ "resolved": "https://registry.npmjs.org/@redis/client/-/client-1.6.1.tgz", "integrity": "sha512-/KCsg3xSlR+nCK8/8ZYSknYxvXHwubJrU82F3Lm1Fp6789VQ0/3RJKfsmRXjqfaTA++23CvC3hqmqe/2GEt6Kw==", "license": "MIT", - "peer": true, "dependencies": { "cluster-key-slot": "1.1.2", "generic-pool": "3.9.0", @@ -9961,15 +9944,13 @@ "version": "0.5.1", "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.5.1.tgz", "integrity": "sha512-SQa1Ws6hUbfC98vKGxZH3KFY0Y1lm5Zm0SY8XX9zbK7FJCyVEac3ATW0RIpwzW+oOfmHE5PMPufDG9hCfoEOMw==", - "license": "MIT", - "peer": true + "license": "MIT" }, "node_modules/class-validator": { "version": "0.14.4", "resolved": "https://registry.npmjs.org/class-validator/-/class-validator-0.14.4.tgz", "integrity": "sha512-AwNusCCam51q703dW82x95tOqQp6oC9HNUl724KxJJOfnKscI8dOloXFgyez7LbTTKWuRBA37FScqVbJEoq8Yw==", "license": "MIT", - "peer": true, "dependencies": { "@types/validator": "^13.15.3", "libphonenumber-js": "^1.11.1", @@ -10475,7 +10456,6 @@ "integrity": "sha512-hr4ihw+DBqcvrsEDioRO31Z17x71pUYoNe/4h6Z0wB72p7MU7/9gH8Q3s12NFhHPfYBBOV3qyfUxmr/Yn3shnQ==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "env-paths": "^2.2.1", "import-fresh": "^3.3.0", @@ -11257,7 +11237,8 @@ "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-2.0.0.tgz", "integrity": "sha512-5POEcUuZybH7IdmGsD8wlf0AI55wMecM9rVBTI/qEAy2c1kTOm3DjFYjrBdI2K3BaJjJYfYFeRtM0t9ssnRuxw==", "dev": true, - "license": "MIT" + "license": "MIT", + "peer": true }, "node_modules/es-object-atoms": { "version": "1.1.1", @@ -11321,7 +11302,6 @@ "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", "@eslint-community/regexpp": "^4.6.1", @@ -11378,7 +11358,6 @@ "integrity": "sha512-iI1f+D2ViGn+uvv5HuHVUamg8ll4tN+JRHGc6IJi4TP9Kl976C57fzPXgseXNs8v0iA8aSJpHsTWjDb9QJamGQ==", "dev": true, "license": "MIT", - "peer": true, "bin": { "eslint-config-prettier": "bin/cli.js" }, @@ -11850,7 +11829,6 @@ "resolved": "https://registry.npmjs.org/express-session/-/express-session-1.19.0.tgz", "integrity": "sha512-0csaMkGq+vaiZTmSMMGkfdCOabYv192VbytFypcvI0MANrp+4i/7yEkJ0sbAEhycQjntaKGzYfjfXQyVb7BHMA==", "license": "MIT", - "peer": true, "dependencies": { "cookie": "~0.7.2", "cookie-signature": "~1.0.7", @@ -12929,7 +12907,6 @@ "resolved": "https://registry.npmjs.org/graphql/-/graphql-16.13.2.tgz", "integrity": "sha512-5bJ+nf/UCpAjHM8i06fl7eLyVC9iuNAjm9qzkiu2ZGhM0VscSvS6WDPfAwkdkBuoXGM9FJSbKl6wylMwP9Ktig==", "license": "MIT", - "peer": true, "engines": { "node": "^12.22.0 || ^14.16.0 || ^16.0.0 || >=17.0.0" } @@ -13398,7 +13375,6 @@ "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.10.1.tgz", "integrity": "sha512-HuEDBTI70aYdx1v6U97SbNx9F1+svQKBDo30o0b9fw055LMepzpOOd0Ccg9Q6tbqmBSJaMuY0fB7yw9/vjBYCA==", "license": "MIT", - "peer": true, "dependencies": { "@ioredis/commands": "1.5.1", "cluster-key-slot": "^1.1.0", @@ -13746,7 +13722,6 @@ "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "@jest/core": "^29.7.0", "@jest/types": "^29.6.3", @@ -15112,7 +15087,6 @@ "resolved": "https://registry.npmjs.org/keyv/-/keyv-5.6.0.tgz", "integrity": "sha512-CYDD3SOtsHtyXeEORYRx2qBtpDJFjRTGXUtmNEMGyzYOKj1TE3tycdlho7kA1Ufx9OYWZzg52QFBGALTirzDSw==", "license": "MIT", - "peer": true, "dependencies": { "@keyv/serialize": "^1.1.1" } @@ -16787,7 +16761,6 @@ "resolved": "https://registry.npmjs.org/passport/-/passport-0.7.0.tgz", "integrity": "sha512-cPLl+qZpSc+ireUvt+IzqbED1cHHkDoVYMo30jbJIdOOjQ1MQYZBPiNvmi8UM6lJuOpTPXJGZQk0DtC4y61MYQ==", "license": "MIT", - "peer": true, "dependencies": { "passport-strategy": "1.x.x", "pause": "0.0.1", @@ -16933,7 +16906,6 @@ "resolved": "https://registry.npmjs.org/pg/-/pg-8.20.0.tgz", "integrity": "sha512-ldhMxz2r8fl/6QkXnBD3CR9/xg694oT6DZQ2s6c/RI28OjtSOpxnPrUCGOBJ46RCUxcWdx3p6kw/xnDHjKvaRA==", "license": "MIT", - "peer": true, "dependencies": { "pg-connection-string": "^2.12.0", "pg-pool": "^3.13.0", @@ -17275,7 +17247,6 @@ "integrity": "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw==", "dev": true, "license": "MIT", - "peer": true, "bin": { "prettier": "bin/prettier.cjs" }, @@ -17958,7 +17929,6 @@ "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==", "dev": true, "license": "MIT", - "peer": true, "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", @@ -19503,7 +19473,6 @@ "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", "devOptional": true, "license": "MIT", - "peer": true, "dependencies": { "@cspotcode/source-map-support": "^0.8.0", "@tsconfig/node10": "^1.0.7", @@ -19737,7 +19706,6 @@ "resolved": "https://registry.npmjs.org/typeorm/-/typeorm-0.3.28.tgz", "integrity": "sha512-6GH7wXhtfq2D33ZuRXYwIsl/qM5685WZcODZb7noOOcRMteM9KF2x2ap3H0EBjnSV0VO4gNAfJT5Ukp0PkOlvg==", "license": "MIT", - "peer": true, "dependencies": { "@sqltools/formatter": "^1.2.5", "ansis": "^4.2.0", @@ -19941,7 +19909,6 @@ "integrity": "sha512-y2TvuxSZPDyQakkFRPZHKFm+KKVqIisdg9/CZwm9ftvKXLP8NRWj38/ODjNbr43SsoXqNuAisEf1GdCxqWcdBw==", "devOptional": true, "license": "Apache-2.0", - "peer": true, "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" @@ -20288,6 +20255,7 @@ "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", "dev": true, "license": "BSD-2-Clause", + "peer": true, "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^4.1.1" @@ -20302,6 +20270,7 @@ "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", "dev": true, "license": "BSD-2-Clause", + "peer": true, "engines": { "node": ">=4.0" } @@ -20312,6 +20281,7 @@ "integrity": "sha512-eflK8wEtyOE6+hsaRVPxvUKYCpRgzLqDTb8krvAsRIwOGlHoSgYLgBXoubGgLd2fT41/OUYdb48v4k4WWHQurA==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@types/json-schema": "^7.0.9", "ajv": "^8.9.0", diff --git a/scripts/replace-any-with-unknown.js b/scripts/replace-any-with-unknown.js new file mode 100644 index 00000000..69c290ff --- /dev/null +++ b/scripts/replace-any-with-unknown.js @@ -0,0 +1,58 @@ +const ts = require('typescript'); +const fs = require('fs'); +const path = require('path'); + +const SRC_DIR = path.resolve(__dirname, '../src'); + +function replaceAny(node, context) { + if (node.kind === ts.SyntaxKind.AnyKeyword) { + return ts.factory.createKeywordTypeNode(ts.SyntaxKind.UnknownKeyword); + } + + if (ts.isAsExpression(node) && node.type.kind === ts.SyntaxKind.AnyKeyword) { + return ts.factory.createAsExpression(node.expression, ts.factory.createKeywordTypeNode(ts.SyntaxKind.UnknownKeyword)); + } + + return ts.visitEachChild(node, (child) => replaceAny(child, context), context); +} + +function transformerFactory(context) { + return (rootNode) => ts.visitNode(rootNode, (node) => replaceAny(node, context)); +} + +function transformSourceText(text, filePath) { + const sourceFile = ts.createSourceFile(filePath, text, ts.ScriptTarget.Latest, true, ts.ScriptKind.TS); + const result = ts.transform(sourceFile, [transformerFactory]); + const transformed = result.transformed[0]; + const printer = ts.createPrinter({ newLine: ts.NewLineKind.LineFeed }); + const output = printer.printFile(transformed); + result.dispose(); + return output; +} + +function walkDir(dir, callback) { + for (const entry of fs.readdirSync(dir, { withFileTypes: true })) { + const fullPath = path.join(dir, entry.name); + if (entry.isDirectory()) { + walkDir(fullPath, callback); + } else if (entry.isFile() && fullPath.endsWith('.ts')) { + callback(fullPath); + } + } +} + +const changed = []; + +walkDir(SRC_DIR, (filePath) => { + const content = fs.readFileSync(filePath, 'utf8'); + const transformed = transformSourceText(content, filePath); + if (transformed !== content) { + fs.writeFileSync(filePath, transformed, 'utf8'); + changed.push(filePath); + } +}); + +console.log(`Processed ${changed.length} files`); +if (changed.length > 0) { + console.log(changed.join('\n')); +} diff --git a/src/ab-testing/ab-testing.controller.ts b/src/ab-testing/ab-testing.controller.ts index 2d69291f..4e82aac7 100644 --- a/src/ab-testing/ab-testing.controller.ts +++ b/src/ab-testing/ab-testing.controller.ts @@ -1,17 +1,4 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Param, - Body, - Query, - Logger, - HttpCode, - HttpStatus, - UseGuards, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Param, Body, Query, Logger, HttpCode, HttpStatus, UseGuards, } from '@nestjs/common'; import { ABTestingService, CreateExperimentDto } from './ab-testing.service'; import { ExperimentService } from './experiments/experiment.service'; import { StatisticalAnalysisService } from './analysis/statistical-analysis.service'; @@ -21,207 +8,225 @@ import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RolesGuard } from '../auth/guards/roles.guard'; import { Roles } from '../auth/decorators/roles.decorator'; import { UserRole } from '../users/entities/user.entity'; - @Controller('ab-testing') @UseGuards(JwtAuthGuard, RolesGuard) export class ABTestingController { - private readonly logger = new Logger(ABTestingController.name); - - constructor( - private abTestingService: ABTestingService, - private experimentService: ExperimentService, - private statisticalAnalysisService: StatisticalAnalysisService, - private automatedDecisionService: AutomatedDecisionService, - private reportsService: ABTestingReportsService, - ) {} - - @Get('experiments') - @Roles(UserRole.ADMIN, UserRole.TEACHER) - async getAllExperiments(): Promise { - this.logger.log('Fetching all experiments'); - return await this.abTestingService.getAllExperiments(); - } - - @Get('experiments/:id') - async getExperimentById(@Param('id') id: string): Promise { - this.logger.log(`Fetching experiment: ${id}`); - return await this.abTestingService.getExperimentById(id); - } - - @Post('experiments') - @HttpCode(HttpStatus.CREATED) - @Roles(UserRole.ADMIN) - async createExperiment(@Body() createExperimentDto: CreateExperimentDto): Promise { - this.logger.log(`Creating new experiment: ${createExperimentDto.name}`); - return await this.abTestingService.createExperiment(createExperimentDto); - } - - @Post('experiments/:id/start') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async startExperiment(@Param('id') id: string): Promise { - this.logger.log(`Starting experiment: ${id}`); - return await this.abTestingService.startExperiment(id); - } - - @Post('experiments/:id/stop') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async stopExperiment(@Param('id') id: string): Promise { - this.logger.log(`Stopping experiment: ${id}`); - return await this.abTestingService.stopExperiment(id); - } - - @Put('experiments/:id') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async updateExperiment(@Param('id') id: string, @Body() updateData: any): Promise { - this.logger.log(`Updating experiment: ${id}`); - return await this.experimentService.updateExperiment(id, updateData); - } - - @Delete('experiments/:id') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async deleteExperiment(@Param('id') id: string): Promise { - this.logger.log(`Deleting experiment: ${id}`); - // Implementation would go here - return { message: 'Experiment deleted successfully' }; - } - - @Get('experiments/:id/results') - async getExperimentResults(@Param('id') id: string): Promise { - this.logger.log(`Fetching results for experiment: ${id}`); - return await this.experimentService.getExperimentResults(id); - } - - @Post('experiments/:id/variants') - @HttpCode(HttpStatus.CREATED) - @Roles(UserRole.ADMIN) - async addVariant(@Param('id') experimentId: string, @Body() variantData: any): Promise { - this.logger.log(`Adding variant to experiment: ${experimentId}`); - return await this.experimentService.addVariant(experimentId, variantData); - } - - @Delete('variants/:id') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async removeVariant(@Param('id') variantId: string): Promise { - this.logger.log(`Removing variant: ${variantId}`); - await this.experimentService.removeVariant(variantId); - return { message: 'Variant removed successfully' }; - } - - @Put('experiments/:id/traffic-allocation') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async updateTrafficAllocation( - @Param('id') experimentId: string, - @Body() allocations: Record, - ): Promise { - this.logger.log(`Updating traffic allocation for experiment: ${experimentId}`); - await this.experimentService.updateTrafficAllocation(experimentId, allocations); - return { message: 'Traffic allocation updated successfully' }; - } - - @Get('experiments/:id/statistical-analysis') - async getStatisticalAnalysis(@Param('id') id: string): Promise { - this.logger.log(`Performing statistical analysis for experiment: ${id}`); - return await this.statisticalAnalysisService.calculateStatisticalSignificance(id); - } - - @Get('experiments/:id/effect-size') - async getEffectSize(@Param('id') id: string): Promise { - this.logger.log(`Calculating effect size for experiment: ${id}`); - return await this.statisticalAnalysisService.calculateEffectSize(id); - } - - @Post('experiments/:id/auto-select-winner') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async autoSelectWinner(@Param('id') id: string, @Body() criteria?: any): Promise { - this.logger.log(`Auto-selecting winner for experiment: ${id}`); - return await this.automatedDecisionService.autoSelectWinner(id, criteria); - } - - @Get('experiments/:id/decision-recommendations') - async getDecisionRecommendations(@Param('id') id: string): Promise { - this.logger.log(`Getting decision recommendations for experiment: ${id}`); - return await this.automatedDecisionService.getDecisionRecommendations(id); - } - - @Post('experiments/:id/auto-allocate-traffic') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async autoAllocateTraffic(@Param('id') id: string): Promise { - this.logger.log(`Auto-allocating traffic for experiment: ${id}`); - await this.automatedDecisionService.autoAllocateTraffic(id); - return { message: 'Traffic auto-allocated successfully' }; - } - - @Get('reports/dashboard') - @Roles(UserRole.ADMIN, UserRole.TEACHER) - async getDashboardSummary(@Query() filters?: any): Promise { - this.logger.log('Generating dashboard summary'); - return await this.reportsService.getDashboardSummary(filters); - } - - @Get('reports/experiment/:id') - async generateExperimentReport(@Param('id') id: string): Promise { - this.logger.log(`Generating report for experiment: ${id}`); - return await this.reportsService.generateExperimentReport(id); - } - - @Get('reports/performance-comparison') - async getPerformanceComparisonReport(): Promise { - this.logger.log('Generating performance comparison report'); - return await this.reportsService.generatePerformanceComparisonReport(); - } - - @Get('reports/timeline') - async getExperimentTimeline(): Promise { - this.logger.log('Generating experiment timeline'); - return await this.reportsService.getExperimentTimeline(); - } - - @Get('reports/experiment/:id/export') - async exportExperimentData(@Param('id') id: string): Promise { - this.logger.log(`Exporting data for experiment: ${id}`); - const csvData = await this.reportsService.exportExperimentData(id); - return { - filename: `experiment-${id}-data.csv`, - data: csvData, - }; - } - - @Post('experiments/:id/pause') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async pauseExperiment(@Param('id') id: string): Promise { - this.logger.log(`Pausing experiment: ${id}`); - return await this.experimentService.pauseExperiment(id); - } - - @Post('experiments/:id/resume') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async resumeExperiment(@Param('id') id: string): Promise { - this.logger.log(`Resuming experiment: ${id}`); - return await this.experimentService.resumeExperiment(id); - } - - @Post('experiments/:id/archive') - @HttpCode(HttpStatus.OK) - @Roles(UserRole.ADMIN) - async archiveExperiment(@Param('id') id: string): Promise { - this.logger.log(`Archiving experiment: ${id}`); - return await this.experimentService.archiveExperiment(id); - } - - @Get('experiments/:id/assign-user/:userId') - @Roles(UserRole.ADMIN) - async assignUserToVariant(@Param('id') experimentId: string, @Param('userId') userId: string): Promise { - this.logger.log(`Assigning user ${userId} to variant for experiment: ${experimentId}`); - return await this.abTestingService.assignUserToVariant(experimentId, userId); - } + private readonly logger = new Logger(ABTestingController.name); + constructor(private abTestingService: ABTestingService, private experimentService: ExperimentService, private statisticalAnalysisService: StatisticalAnalysisService, private automatedDecisionService: AutomatedDecisionService, private reportsService: ABTestingReportsService) { } + @Get('experiments') + @Roles(UserRole.ADMIN, UserRole.TEACHER) + async getAllExperiments(): Promise { + this.logger.log('Fetching all experiments'); + return await this.abTestingService.getAllExperiments(); + } + @Get('experiments/:id') + async getExperimentById( + @Param('id') + id: string): Promise { + this.logger.log(`Fetching experiment: ${id}`); + return await this.abTestingService.getExperimentById(id); + } + @Post('experiments') + @HttpCode(HttpStatus.CREATED) + @Roles(UserRole.ADMIN) + async createExperiment( + @Body() + createExperimentDto: CreateExperimentDto): Promise { + this.logger.log(`Creating new experiment: ${createExperimentDto.name}`); + return await this.abTestingService.createExperiment(createExperimentDto); + } + @Post('experiments/:id/start') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async startExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Starting experiment: ${id}`); + return await this.abTestingService.startExperiment(id); + } + @Post('experiments/:id/stop') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async stopExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Stopping experiment: ${id}`); + return await this.abTestingService.stopExperiment(id); + } + @Put('experiments/:id') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async updateExperiment( + @Param('id') + id: string, + @Body() + updateData: unknown): Promise { + this.logger.log(`Updating experiment: ${id}`); + return await this.experimentService.updateExperiment(id, updateData); + } + @Delete('experiments/:id') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async deleteExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Deleting experiment: ${id}`); + // Implementation would go here + return { message: 'Experiment deleted successfully' }; + } + @Get('experiments/:id/results') + async getExperimentResults( + @Param('id') + id: string): Promise { + this.logger.log(`Fetching results for experiment: ${id}`); + return await this.experimentService.getExperimentResults(id); + } + @Post('experiments/:id/variants') + @HttpCode(HttpStatus.CREATED) + @Roles(UserRole.ADMIN) + async addVariant( + @Param('id') + experimentId: string, + @Body() + variantData: unknown): Promise { + this.logger.log(`Adding variant to experiment: ${experimentId}`); + return await this.experimentService.addVariant(experimentId, variantData); + } + @Delete('variants/:id') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async removeVariant( + @Param('id') + variantId: string): Promise { + this.logger.log(`Removing variant: ${variantId}`); + await this.experimentService.removeVariant(variantId); + return { message: 'Variant removed successfully' }; + } + @Put('experiments/:id/traffic-allocation') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async updateTrafficAllocation( + @Param('id') + experimentId: string, + @Body() + allocations: Record): Promise { + this.logger.log(`Updating traffic allocation for experiment: ${experimentId}`); + await this.experimentService.updateTrafficAllocation(experimentId, allocations); + return { message: 'Traffic allocation updated successfully' }; + } + @Get('experiments/:id/statistical-analysis') + async getStatisticalAnalysis( + @Param('id') + id: string): Promise { + this.logger.log(`Performing statistical analysis for experiment: ${id}`); + return await this.statisticalAnalysisService.calculateStatisticalSignificance(id); + } + @Get('experiments/:id/effect-size') + async getEffectSize( + @Param('id') + id: string): Promise { + this.logger.log(`Calculating effect size for experiment: ${id}`); + return await this.statisticalAnalysisService.calculateEffectSize(id); + } + @Post('experiments/:id/auto-select-winner') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async autoSelectWinner( + @Param('id') + id: string, + @Body() + criteria?: unknown): Promise { + this.logger.log(`Auto-selecting winner for experiment: ${id}`); + return await this.automatedDecisionService.autoSelectWinner(id, criteria); + } + @Get('experiments/:id/decision-recommendations') + async getDecisionRecommendations( + @Param('id') + id: string): Promise { + this.logger.log(`Getting decision recommendations for experiment: ${id}`); + return await this.automatedDecisionService.getDecisionRecommendations(id); + } + @Post('experiments/:id/auto-allocate-traffic') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async autoAllocateTraffic( + @Param('id') + id: string): Promise { + this.logger.log(`Auto-allocating traffic for experiment: ${id}`); + await this.automatedDecisionService.autoAllocateTraffic(id); + return { message: 'Traffic auto-allocated successfully' }; + } + @Get('reports/dashboard') + @Roles(UserRole.ADMIN, UserRole.TEACHER) + async getDashboardSummary( + @Query() + filters?: unknown): Promise { + this.logger.log('Generating dashboard summary'); + return await this.reportsService.getDashboardSummary(filters); + } + @Get('reports/experiment/:id') + async generateExperimentReport( + @Param('id') + id: string): Promise { + this.logger.log(`Generating report for experiment: ${id}`); + return await this.reportsService.generateExperimentReport(id); + } + @Get('reports/performance-comparison') + async getPerformanceComparisonReport(): Promise { + this.logger.log('Generating performance comparison report'); + return await this.reportsService.generatePerformanceComparisonReport(); + } + @Get('reports/timeline') + async getExperimentTimeline(): Promise { + this.logger.log('Generating experiment timeline'); + return await this.reportsService.getExperimentTimeline(); + } + @Get('reports/experiment/:id/export') + async exportExperimentData( + @Param('id') + id: string): Promise { + this.logger.log(`Exporting data for experiment: ${id}`); + const csvData = await this.reportsService.exportExperimentData(id); + return { + filename: `experiment-${id}-data.csv`, + data: csvData, + }; + } + @Post('experiments/:id/pause') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async pauseExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Pausing experiment: ${id}`); + return await this.experimentService.pauseExperiment(id); + } + @Post('experiments/:id/resume') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async resumeExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Resuming experiment: ${id}`); + return await this.experimentService.resumeExperiment(id); + } + @Post('experiments/:id/archive') + @HttpCode(HttpStatus.OK) + @Roles(UserRole.ADMIN) + async archiveExperiment( + @Param('id') + id: string): Promise { + this.logger.log(`Archiving experiment: ${id}`); + return await this.experimentService.archiveExperiment(id); + } + @Get('experiments/:id/assign-user/:userId') + @Roles(UserRole.ADMIN) + async assignUserToVariant( + @Param('id') + experimentId: string, + @Param('userId') + userId: string): Promise { + this.logger.log(`Assigning user ${userId} to variant for experiment: ${experimentId}`); + return await this.abTestingService.assignUserToVariant(experimentId, userId); + } } diff --git a/src/ab-testing/ab-testing.module.ts b/src/ab-testing/ab-testing.module.ts index 50ee541a..f94c7bc5 100644 --- a/src/ab-testing/ab-testing.module.ts +++ b/src/ab-testing/ab-testing.module.ts @@ -10,25 +10,25 @@ import { StatisticalAnalysisService } from './analysis/statistical-analysis.serv import { AutomatedDecisionService } from './automation/automated-decision.service'; import { ABTestingReportsService } from './reporting/ab-testing-reports.service'; import { ABTestingController } from './ab-testing.controller'; - @Module({ - imports: [ - TypeOrmModule.forFeature([Experiment, ExperimentVariant, ExperimentMetric, VariantMetric]), - ], - controllers: [ABTestingController], - providers: [ - ABTestingService, - ExperimentService, - StatisticalAnalysisService, - AutomatedDecisionService, - ABTestingReportsService, - ], - exports: [ - ABTestingService, - ExperimentService, - StatisticalAnalysisService, - AutomatedDecisionService, - ABTestingReportsService, - ], + imports: [ + TypeOrmModule.forFeature([Experiment, ExperimentVariant, ExperimentMetric, VariantMetric]), + ], + controllers: [ABTestingController], + providers: [ + ABTestingService, + ExperimentService, + StatisticalAnalysisService, + AutomatedDecisionService, + ABTestingReportsService, + ], + exports: [ + ABTestingService, + ExperimentService, + StatisticalAnalysisService, + AutomatedDecisionService, + ABTestingReportsService, + ], }) -export class ABTestingModule {} +export class ABTestingModule { +} diff --git a/src/ab-testing/ab-testing.service.ts b/src/ab-testing/ab-testing.service.ts index 2b84e3ea..02912d74 100644 --- a/src/ab-testing/ab-testing.service.ts +++ b/src/ab-testing/ab-testing.service.ts @@ -3,200 +3,169 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { Experiment, ExperimentStatus, ExperimentType } from './entities/experiment.entity'; import { ExperimentVariant } from './entities/experiment-variant.entity'; - export interface CreateExperimentDto { - name: string; - description: string; - type: ExperimentType; - startDate: Date; - endDate?: Date; - trafficAllocation: number; - autoAllocateTraffic: boolean; - confidenceLevel: number; - minimumSampleSize: number; - hypothesis: string; - targetingCriteria?: any; - exclusionCriteria?: any; - variants: CreateVariantDto[]; - metrics: CreateMetricDto[]; + name: string; + description: string; + type: ExperimentType; + startDate: Date; + endDate?: Date; + trafficAllocation: number; + autoAllocateTraffic: boolean; + confidenceLevel: number; + minimumSampleSize: number; + hypothesis: string; + targetingCriteria?: unknown; + exclusionCriteria?: unknown; + variants: CreateVariantDto[]; + metrics: CreateMetricDto[]; } - export interface CreateVariantDto { - name: string; - description: string; - configuration: any; - isControl: boolean; + name: string; + description: string; + configuration: unknown; + isControl: boolean; } - export interface CreateMetricDto { - name: string; - description: string; - type: string; - isPrimary: boolean; - configuration?: any; + name: string; + description: string; + type: string; + isPrimary: boolean; + configuration?: unknown; } - @Injectable() export class ABTestingService { - private readonly logger = new Logger(ABTestingService.name); - - constructor( + private readonly logger = new Logger(ABTestingService.name); + constructor( @InjectRepository(Experiment) - private experimentRepository: Repository, + private experimentRepository: Repository, @InjectRepository(ExperimentVariant) - private variantRepository: Repository, - ) {} - - /** - * Creates a new experiment - */ - async createExperiment(createExperimentDto: CreateExperimentDto): Promise { - this.logger.log(`Creating new experiment: ${createExperimentDto.name}`); - - const experiment = new Experiment(); - experiment.name = createExperimentDto.name; - experiment.description = createExperimentDto.description; - experiment.type = createExperimentDto.type; - experiment.startDate = createExperimentDto.startDate; - experiment.endDate = createExperimentDto.endDate; - experiment.trafficAllocation = createExperimentDto.trafficAllocation; - experiment.autoAllocateTraffic = createExperimentDto.autoAllocateTraffic; - experiment.confidenceLevel = createExperimentDto.confidenceLevel; - experiment.minimumSampleSize = createExperimentDto.minimumSampleSize; - experiment.hypothesis = createExperimentDto.hypothesis; - experiment.targetingCriteria = createExperimentDto.targetingCriteria; - experiment.exclusionCriteria = createExperimentDto.exclusionCriteria; - experiment.status = ExperimentStatus.DRAFT; - - // Save the experiment first - const savedExperiment = await this.experimentRepository.save(experiment); - - // Create variants - const variants = createExperimentDto.variants.map((variantDto) => { - const variant = new ExperimentVariant(); - variant.name = variantDto.name; - variant.description = variantDto.description; - variant.configuration = variantDto.configuration; - variant.isControl = variantDto.isControl; - variant.experiment = savedExperiment; - return variant; - }); - - await this.variantRepository.save(variants); - - this.logger.log(`Experiment created successfully: ${savedExperiment.name}`); - return savedExperiment; - } - - /** - * Gets all experiments - */ - async getAllExperiments(): Promise { - return await this.experimentRepository.find({ - relations: ['variants', 'metrics'], - order: { createdAt: 'DESC' }, - }); - } - - /** - * Gets experiment by ID - */ - async getExperimentById(id: string): Promise { - const experiment = await this.experimentRepository.findOne({ - where: { id }, - relations: ['variants', 'metrics', 'variants.metrics'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${id} not found`); + private variantRepository: Repository) { } + /** + * Creates a new experiment + */ + async createExperiment(createExperimentDto: CreateExperimentDto): Promise { + this.logger.log(`Creating new experiment: ${createExperimentDto.name}`); + const experiment = new Experiment(); + experiment.name = createExperimentDto.name; + experiment.description = createExperimentDto.description; + experiment.type = createExperimentDto.type; + experiment.startDate = createExperimentDto.startDate; + experiment.endDate = createExperimentDto.endDate; + experiment.trafficAllocation = createExperimentDto.trafficAllocation; + experiment.autoAllocateTraffic = createExperimentDto.autoAllocateTraffic; + experiment.confidenceLevel = createExperimentDto.confidenceLevel; + experiment.minimumSampleSize = createExperimentDto.minimumSampleSize; + experiment.hypothesis = createExperimentDto.hypothesis; + experiment.targetingCriteria = createExperimentDto.targetingCriteria; + experiment.exclusionCriteria = createExperimentDto.exclusionCriteria; + experiment.status = ExperimentStatus.DRAFT; + // Save the experiment first + const savedExperiment = await this.experimentRepository.save(experiment); + // Create variants + const variants = createExperimentDto.variants.map((variantDto) => { + const variant = new ExperimentVariant(); + variant.name = variantDto.name; + variant.description = variantDto.description; + variant.configuration = variantDto.configuration; + variant.isControl = variantDto.isControl; + variant.experiment = savedExperiment; + return variant; + }); + await this.variantRepository.save(variants); + this.logger.log(`Experiment created successfully: ${savedExperiment.name}`); + return savedExperiment; } - - return experiment; - } - - /** - * Starts an experiment - */ - async startExperiment(id: string): Promise { - this.logger.log(`Starting experiment: ${id}`); - - const experiment = await this.getExperimentById(id); - - if (experiment.status !== ExperimentStatus.DRAFT) { - throw new Error('Only draft experiments can be started'); + /** + * Gets all experiments + */ + async getAllExperiments(): Promise { + return await this.experimentRepository.find({ + relations: ['variants', 'metrics'], + order: { createdAt: 'DESC' }, + }); } - - if (!experiment.variants || experiment.variants.length < 2) { - throw new Error('Experiment must have at least 2 variants'); + /** + * Gets experiment by ID + */ + async getExperimentById(id: string): Promise { + const experiment = await this.experimentRepository.findOne({ + where: { id }, + relations: ['variants', 'metrics', 'variants.metrics'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${id} not found`); + } + return experiment; } - - // Validate that there's exactly one control variant - const controlVariants = experiment.variants.filter((v) => v.isControl); - if (controlVariants.length !== 1) { - throw new Error('Experiment must have exactly one control variant'); + /** + * Starts an experiment + */ + async startExperiment(id: string): Promise { + this.logger.log(`Starting experiment: ${id}`); + const experiment = await this.getExperimentById(id); + if (experiment.status !== ExperimentStatus.DRAFT) { + throw new Error('Only draft experiments can be started'); + } + if (!experiment.variants || experiment.variants.length < 2) { + throw new Error('Experiment must have at least 2 variants'); + } + // Validate that there's exactly one control variant + const controlVariants = experiment.variants.filter((v) => v.isControl); + if (controlVariants.length !== 1) { + throw new Error('Experiment must have exactly one control variant'); + } + experiment.status = ExperimentStatus.RUNNING; + experiment.startDate = new Date(); + const updatedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment started: ${updatedExperiment.name}`); + return updatedExperiment; } - - experiment.status = ExperimentStatus.RUNNING; - experiment.startDate = new Date(); - - const updatedExperiment = await this.experimentRepository.save(experiment); - this.logger.log(`Experiment started: ${updatedExperiment.name}`); - return updatedExperiment; - } - - /** - * Stops an experiment - */ - async stopExperiment(id: string): Promise { - this.logger.log(`Stopping experiment: ${id}`); - - const experiment = await this.getExperimentById(id); - experiment.status = ExperimentStatus.COMPLETED; - experiment.endDate = new Date(); - - const updatedExperiment = await this.experimentRepository.save(experiment); - this.logger.log(`Experiment stopped: ${updatedExperiment.name}`); - return updatedExperiment; - } - - /** - * Gets active experiments for a user - */ - async getActiveExperimentsForUser(_userId: string): Promise { - return await this.experimentRepository.find({ - where: { - status: ExperimentStatus.RUNNING, - startDate: new Date(), - }, - relations: ['variants'], - }); - } - - /** - * Assigns a user to a variant - */ - async assignUserToVariant(experimentId: string, userId: string): Promise { - const experiment = await this.getExperimentById(experimentId); - - if (experiment.status !== ExperimentStatus.RUNNING) { - throw new Error('Experiment is not running'); + /** + * Stops an experiment + */ + async stopExperiment(id: string): Promise { + this.logger.log(`Stopping experiment: ${id}`); + const experiment = await this.getExperimentById(id); + experiment.status = ExperimentStatus.COMPLETED; + experiment.endDate = new Date(); + const updatedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment stopped: ${updatedExperiment.name}`); + return updatedExperiment; } - - // Simple hash-based assignment for consistent user-to-variant mapping - const variantIndex = this.hashUserIdToVariant(userId, experiment.variants.length); - return experiment.variants[variantIndex]; - } - - /** - * Hashes user ID to determine variant assignment - */ - private hashUserIdToVariant(userId: string, variantCount: number): number { - let hash = 0; - for (let i = 0; i < userId.length; i++) { - const char = userId.charCodeAt(i); - hash = (hash << 5) - hash + char; - hash = hash & hash; // Convert to 32-bit integer + /** + * Gets active experiments for a user + */ + async getActiveExperimentsForUser(_userId: string): Promise { + return await this.experimentRepository.find({ + where: { + status: ExperimentStatus.RUNNING, + startDate: new Date(), + }, + relations: ['variants'], + }); + } + /** + * Assigns a user to a variant + */ + async assignUserToVariant(experimentId: string, userId: string): Promise { + const experiment = await this.getExperimentById(experimentId); + if (experiment.status !== ExperimentStatus.RUNNING) { + throw new Error('Experiment is not running'); + } + // Simple hash-based assignment for consistent user-to-variant mapping + const variantIndex = this.hashUserIdToVariant(userId, experiment.variants.length); + return experiment.variants[variantIndex]; + } + /** + * Hashes user ID to determine variant assignment + */ + private hashUserIdToVariant(userId: string, variantCount: number): number { + let hash = 0; + for (let i = 0; i < userId.length; i++) { + const char = userId.charCodeAt(i); + hash = (hash << 5) - hash + char; + hash = hash & hash; // Convert to 32-bit integer + } + return Math.abs(hash) % variantCount; } - return Math.abs(hash) % variantCount; - } } diff --git a/src/ab-testing/analysis/statistical-analysis.service.ts b/src/ab-testing/analysis/statistical-analysis.service.ts index e9b86b64..2083f168 100644 --- a/src/ab-testing/analysis/statistical-analysis.service.ts +++ b/src/ab-testing/analysis/statistical-analysis.service.ts @@ -4,310 +4,234 @@ import { Repository } from 'typeorm'; import { Experiment } from '../entities/experiment.entity'; import { ExperimentVariant } from '../entities/experiment-variant.entity'; import { VariantMetric } from '../entities/variant-metric.entity'; - @Injectable() export class StatisticalAnalysisService { - private readonly logger = new Logger(StatisticalAnalysisService.name); - - constructor( + private readonly logger = new Logger(StatisticalAnalysisService.name); + constructor( @InjectRepository(Experiment) - private experimentRepository: Repository, + private experimentRepository: Repository, @InjectRepository(ExperimentVariant) - private variantRepository: Repository, + private variantRepository: Repository, @InjectRepository(VariantMetric) - private variantMetricRepository: Repository, - ) {} - - /** - * Calculates statistical significance for experiment results - */ - async calculateStatisticalSignificance(experimentId: string): Promise { - this.logger.log(`Calculating statistical significance for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants', 'variants.metrics'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + private variantMetricRepository: Repository) { } + /** + * Calculates statistical significance for experiment results + */ + async calculateStatisticalSignificance(experimentId: string): Promise { + this.logger.log(`Calculating statistical significance for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants', 'variants.metrics'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + const results = { + experimentId: experiment.id, + confidenceLevel: experiment.confidenceLevel, + variants: [] as unknown[], + statisticallySignificant: false, + }; + // Calculate statistics for each variant + for (const variant of experiment.variants) { + const variantAnalysis = await this.analyzeVariant(variant, experiment.confidenceLevel); + results.variants.push(variantAnalysis); + } + // Check if any variant is statistically significant compared to control + const controlVariant = experiment.variants.find((v) => v.isControl); + if (controlVariant) { + const controlMetrics = await this.getVariantMetrics(controlVariant.id); + results.statisticallySignificant = await this.checkSignificanceAgainstControl(experiment.variants, controlMetrics, experiment.confidenceLevel); + } + return results; } - - const results = { - experimentId: experiment.id, - confidenceLevel: experiment.confidenceLevel, - variants: [] as any[], - statisticallySignificant: false, - }; - - // Calculate statistics for each variant - for (const variant of experiment.variants) { - const variantAnalysis = await this.analyzeVariant(variant, experiment.confidenceLevel); - results.variants.push(variantAnalysis); + /** + * Analyzes a single variant's metrics + */ + private async analyzeVariant(variant: ExperimentVariant, confidenceLevel: number): Promise { + const metrics = await this.getVariantMetrics(variant.id); + const analysis = { + variantId: variant.id, + variantName: variant.name, + isControl: variant.isControl, + metrics: [] as unknown[], + overallPerformance: 0, + }; + for (const metric of metrics) { + const statisticalData = await this.calculateMetricStatistics(metric, confidenceLevel); + analysis.metrics.push(statisticalData); + // For overall performance, we'll use conversion rate or value depending on metric type + if (statisticalData.conversionRate) { + analysis.overallPerformance += statisticalData.conversionRate; + } + else { + analysis.overallPerformance += statisticalData.value; + } + } + return analysis; } - - // Check if any variant is statistically significant compared to control - const controlVariant = experiment.variants.find((v) => v.isControl); - if (controlVariant) { - const controlMetrics = await this.getVariantMetrics(controlVariant.id); - results.statisticallySignificant = await this.checkSignificanceAgainstControl( - experiment.variants, - controlMetrics, - experiment.confidenceLevel, - ); + /** + * Calculates statistics for a specific metric + */ + private async calculateMetricStatistics(metric: VariantMetric, confidenceLevel: number): Promise { + // Calculate standard error + const standardError = metric.standardDeviation && metric.sampleSize > 0 + ? metric.standardDeviation / Math.sqrt(metric.sampleSize) + : 0; + // Calculate confidence interval + const zScore = this.getZScore(confidenceLevel); + const marginOfError = zScore * standardError; + const confidenceIntervalLower = metric.value - marginOfError; + const confidenceIntervalUpper = metric.value + marginOfError; + // Simple p-value calculation (would be more complex in a real implementation) + const pValue = this.calculatePValue(metric.value, standardError); + // Determine if statistically significant + const isStatisticallySignificant = pValue < 1 - confidenceLevel / 100; + return { + metricId: metric.id, + value: metric.value, + sampleSize: metric.sampleSize, + conversionRate: metric.conversionRate, + standardDeviation: metric.standardDeviation, + confidenceInterval: [confidenceIntervalLower, confidenceIntervalUpper], + pValue, + isStatisticallySignificant, + }; } - - return results; - } - - /** - * Analyzes a single variant's metrics - */ - private async analyzeVariant(variant: ExperimentVariant, confidenceLevel: number): Promise { - const metrics = await this.getVariantMetrics(variant.id); - - const analysis = { - variantId: variant.id, - variantName: variant.name, - isControl: variant.isControl, - metrics: [] as any[], - overallPerformance: 0, - }; - - for (const metric of metrics) { - const statisticalData = await this.calculateMetricStatistics(metric, confidenceLevel); - analysis.metrics.push(statisticalData); - - // For overall performance, we'll use conversion rate or value depending on metric type - if (statisticalData.conversionRate) { - analysis.overallPerformance += statisticalData.conversionRate; - } else { - analysis.overallPerformance += statisticalData.value; - } + /** + * Gets metrics for a variant + */ + private async getVariantMetrics(variantId: string): Promise { + return await this.variantMetricRepository.find({ + where: { variant: { id: variantId } }, + }); } - - return analysis; - } - - /** - * Calculates statistics for a specific metric - */ - private async calculateMetricStatistics( - metric: VariantMetric, - confidenceLevel: number, - ): Promise { - // Calculate standard error - const standardError = - metric.standardDeviation && metric.sampleSize > 0 - ? metric.standardDeviation / Math.sqrt(metric.sampleSize) - : 0; - - // Calculate confidence interval - const zScore = this.getZScore(confidenceLevel); - const marginOfError = zScore * standardError; - - const confidenceIntervalLower = metric.value - marginOfError; - const confidenceIntervalUpper = metric.value + marginOfError; - - // Simple p-value calculation (would be more complex in a real implementation) - const pValue = this.calculatePValue(metric.value, standardError); - - // Determine if statistically significant - const isStatisticallySignificant = pValue < 1 - confidenceLevel / 100; - - return { - metricId: metric.id, - value: metric.value, - sampleSize: metric.sampleSize, - conversionRate: metric.conversionRate, - standardDeviation: metric.standardDeviation, - confidenceInterval: [confidenceIntervalLower, confidenceIntervalUpper], - pValue, - isStatisticallySignificant, - }; - } - - /** - * Gets metrics for a variant - */ - private async getVariantMetrics(variantId: string): Promise { - return await this.variantMetricRepository.find({ - where: { variant: { id: variantId } }, - }); - } - - /** - * Checks if any variant is significantly different from control - */ - private async checkSignificanceAgainstControl( - variants: ExperimentVariant[], - controlMetrics: VariantMetric[], - confidenceLevel: number, - ): Promise { - const controlVariant = variants.find((v) => v.isControl); - if (!controlVariant) return false; - - for (const variant of variants) { - if (variant.id === controlVariant.id) continue; - - const variantMetrics = await this.getVariantMetrics(variant.id); - - // Compare each metric - for (let i = 0; i < controlMetrics.length && i < variantMetrics.length; i++) { - const controlMetric = controlMetrics[i]; - const variantMetric = variantMetrics[i]; - - const isSignificant = await this.compareMetrics( - controlMetric, - variantMetric, - confidenceLevel, - ); - - if (isSignificant) { - return true; + /** + * Checks if any variant is significantly different from control + */ + private async checkSignificanceAgainstControl(variants: ExperimentVariant[], controlMetrics: VariantMetric[], confidenceLevel: number): Promise { + const controlVariant = variants.find((v) => v.isControl); + if (!controlVariant) + return false; + for (const variant of variants) { + if (variant.id === controlVariant.id) + continue; + const variantMetrics = await this.getVariantMetrics(variant.id); + // Compare each metric + for (let i = 0; i < controlMetrics.length && i < variantMetrics.length; i++) { + const controlMetric = controlMetrics[i]; + const variantMetric = variantMetrics[i]; + const isSignificant = await this.compareMetrics(controlMetric, variantMetric, confidenceLevel); + if (isSignificant) { + return true; + } + } } - } + return false; + } + /** + * Compares two metrics for statistical significance + */ + private async compareMetrics(metric1: VariantMetric, metric2: VariantMetric, confidenceLevel: number): Promise { + // Calculate pooled standard error for comparison + const pooledSE = Math.sqrt(Math.pow(metric1.standardDeviation || 0, 2) / (metric1.sampleSize || 1) + + Math.pow(metric2.standardDeviation || 0, 2) / (metric2.sampleSize || 1)); + // Calculate z-score for the difference + const difference = metric2.value - metric1.value; + const zScore = pooledSE > 0 ? Math.abs(difference / pooledSE) : 0; + // Get critical z-value for the confidence level + const criticalZ = this.getZScore(confidenceLevel); + return zScore > criticalZ; } - - return false; - } - - /** - * Compares two metrics for statistical significance - */ - private async compareMetrics( - metric1: VariantMetric, - metric2: VariantMetric, - confidenceLevel: number, - ): Promise { - // Calculate pooled standard error for comparison - const pooledSE = Math.sqrt( - Math.pow(metric1.standardDeviation || 0, 2) / (metric1.sampleSize || 1) + - Math.pow(metric2.standardDeviation || 0, 2) / (metric2.sampleSize || 1), - ); - - // Calculate z-score for the difference - const difference = metric2.value - metric1.value; - const zScore = pooledSE > 0 ? Math.abs(difference / pooledSE) : 0; - - // Get critical z-value for the confidence level - const criticalZ = this.getZScore(confidenceLevel); - - return zScore > criticalZ; - } - - /** - * Gets z-score for a given confidence level - */ - private getZScore(confidenceLevel: number): number { - const confidence = confidenceLevel / 100; - const _alpha = 1 - confidence; - - // Z-scores for common confidence levels - const zScores: Record = { - 90: 1.645, - 95: 1.96, - 99: 2.576, - }; - - return zScores[confidenceLevel] || 1.96; // Default to 95% confidence - } - - /** - * Calculates p-value (simplified implementation) - */ - private calculatePValue(value: number, standardError: number): number { - if (standardError === 0) return 1; - - // Simplified p-value calculation - const zScore = Math.abs(value / standardError); - // This is a very simplified approximation - return Math.max(0, Math.min(1, 1 / (1 + Math.exp(-zScore + 2)))); - } - - /** - * Calculates effect size - */ - async calculateEffectSize(experimentId: string): Promise { - this.logger.log(`Calculating effect size for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + /** + * Gets z-score for a given confidence level + */ + private getZScore(confidenceLevel: number): number { + const confidence = confidenceLevel / 100; + const _alpha = 1 - confidence; + // Z-scores for common confidence levels + const zScores: Record = { + 90: 1.645, + 95: 1.96, + 99: 2.576, + }; + return zScores[confidenceLevel] || 1.96; // Default to 95% confidence + } + /** + * Calculates p-value (simplified implementation) + */ + private calculatePValue(value: number, standardError: number): number { + if (standardError === 0) + return 1; + // Simplified p-value calculation + const zScore = Math.abs(value / standardError); + // This is a very simplified approximation + return Math.max(0, Math.min(1, 1 / (1 + Math.exp(-zScore + 2)))); + } + /** + * Calculates effect size + */ + async calculateEffectSize(experimentId: string): Promise { + this.logger.log(`Calculating effect size for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + const controlVariant = experiment.variants.find((v) => v.isControl); + if (!controlVariant) { + throw new Error('No control variant found'); + } + const effectSizes = []; + for (const variant of experiment.variants) { + if (variant.id === controlVariant.id) + continue; + const controlMetrics = await this.getVariantMetrics(controlVariant.id); + const variantMetrics = await this.getVariantMetrics(variant.id); + const effectSize = await this.calculateCohensD(controlMetrics, variantMetrics); + effectSizes.push({ + variantId: variant.id, + variantName: variant.name, + effectSize, + interpretation: this.interpretEffectSize(effectSize), + }); + } + return { + experimentId: experiment.id, + controlVariantId: controlVariant.id, + effectSizes, + }; } - - const controlVariant = experiment.variants.find((v) => v.isControl); - if (!controlVariant) { - throw new Error('No control variant found'); + /** + * Calculates Cohen's d effect size + */ + private async calculateCohensD(controlMetrics: VariantMetric[], variantMetrics: VariantMetric[]): Promise { + if (controlMetrics.length === 0 || variantMetrics.length === 0) + return 0; + // Simplified Cohen's d calculation + const controlMean = controlMetrics.reduce((sum, m) => sum + m.value, 0) / controlMetrics.length; + const variantMean = variantMetrics.reduce((sum, m) => sum + m.value, 0) / variantMetrics.length; + const controlStdDev = Math.sqrt(controlMetrics.reduce((sum, m) => sum + Math.pow(m.value - controlMean, 2), 0) / + (controlMetrics.length - 1)); + const variantStdDev = Math.sqrt(variantMetrics.reduce((sum, m) => sum + Math.pow(m.value - variantMean, 2), 0) / + (variantMetrics.length - 1)); + const pooledStdDev = Math.sqrt(((controlMetrics.length - 1) * Math.pow(controlStdDev, 2) + + (variantMetrics.length - 1) * Math.pow(variantStdDev, 2)) / + (controlMetrics.length + variantMetrics.length - 2)); + return pooledStdDev > 0 ? Math.abs(variantMean - controlMean) / pooledStdDev : 0; } - - const effectSizes = []; - - for (const variant of experiment.variants) { - if (variant.id === controlVariant.id) continue; - - const controlMetrics = await this.getVariantMetrics(controlVariant.id); - const variantMetrics = await this.getVariantMetrics(variant.id); - - const effectSize = await this.calculateCohensD(controlMetrics, variantMetrics); - effectSizes.push({ - variantId: variant.id, - variantName: variant.name, - effectSize, - interpretation: this.interpretEffectSize(effectSize), - }); + /** + * Interprets effect size magnitude + */ + private interpretEffectSize(effectSize: number): string { + if (effectSize < 0.2) + return 'negligible'; + if (effectSize < 0.5) + return 'small'; + if (effectSize < 0.8) + return 'medium'; + return 'large'; } - - return { - experimentId: experiment.id, - controlVariantId: controlVariant.id, - effectSizes, - }; - } - - /** - * Calculates Cohen's d effect size - */ - private async calculateCohensD( - controlMetrics: VariantMetric[], - variantMetrics: VariantMetric[], - ): Promise { - if (controlMetrics.length === 0 || variantMetrics.length === 0) return 0; - - // Simplified Cohen's d calculation - const controlMean = controlMetrics.reduce((sum, m) => sum + m.value, 0) / controlMetrics.length; - const variantMean = variantMetrics.reduce((sum, m) => sum + m.value, 0) / variantMetrics.length; - - const controlStdDev = Math.sqrt( - controlMetrics.reduce((sum, m) => sum + Math.pow(m.value - controlMean, 2), 0) / - (controlMetrics.length - 1), - ); - - const variantStdDev = Math.sqrt( - variantMetrics.reduce((sum, m) => sum + Math.pow(m.value - variantMean, 2), 0) / - (variantMetrics.length - 1), - ); - - const pooledStdDev = Math.sqrt( - ((controlMetrics.length - 1) * Math.pow(controlStdDev, 2) + - (variantMetrics.length - 1) * Math.pow(variantStdDev, 2)) / - (controlMetrics.length + variantMetrics.length - 2), - ); - - return pooledStdDev > 0 ? Math.abs(variantMean - controlMean) / pooledStdDev : 0; - } - - /** - * Interprets effect size magnitude - */ - private interpretEffectSize(effectSize: number): string { - if (effectSize < 0.2) return 'negligible'; - if (effectSize < 0.5) return 'small'; - if (effectSize < 0.8) return 'medium'; - return 'large'; - } } diff --git a/src/ab-testing/automation/automated-decision.service.ts b/src/ab-testing/automation/automated-decision.service.ts index b5948848..7d9be893 100644 --- a/src/ab-testing/automation/automated-decision.service.ts +++ b/src/ab-testing/automation/automated-decision.service.ts @@ -4,343 +4,266 @@ import { Repository } from 'typeorm'; import { Experiment, ExperimentStatus } from '../entities/experiment.entity'; import { ExperimentVariant } from '../entities/experiment-variant.entity'; import { StatisticalAnalysisService } from '../analysis/statistical-analysis.service'; - export interface WinnerSelectionCriteria { - confidenceLevel: number; - minimumSampleSize: number; - effectSizeThreshold: number; - durationThreshold: number; // in days + confidenceLevel: number; + minimumSampleSize: number; + effectSizeThreshold: number; + durationThreshold: number; // in days } - @Injectable() export class AutomatedDecisionService { - private readonly logger = new Logger(AutomatedDecisionService.name); - - constructor( + private readonly logger = new Logger(AutomatedDecisionService.name); + constructor( @InjectRepository(Experiment) - private experimentRepository: Repository, + private experimentRepository: Repository, @InjectRepository(ExperimentVariant) - private variantRepository: Repository, - private statisticalAnalysisService: StatisticalAnalysisService, - ) {} - - /** - * Automatically selects winner for an experiment - */ - async autoSelectWinner(experimentId: string, criteria?: WinnerSelectionCriteria): Promise { - this.logger.log(`Auto-selecting winner for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); - } - - if (experiment.status !== ExperimentStatus.RUNNING) { - throw new Error('Only running experiments can have winners selected'); - } - - const defaultCriteria: WinnerSelectionCriteria = { - confidenceLevel: experiment.confidenceLevel || 95, - minimumSampleSize: experiment.minimumSampleSize || 100, - effectSizeThreshold: 0.1, - durationThreshold: 7, - }; - - const selectionCriteria = { ...defaultCriteria, ...criteria }; - - // Check if experiment meets duration threshold - const experimentDuration = this.calculateExperimentDuration(experiment); - if (experimentDuration < selectionCriteria.durationThreshold) { - return { - experimentId: experiment.id, - decision: 'no_winner', - reason: `Experiment duration (${experimentDuration} days) below threshold (${selectionCriteria.durationThreshold} days)`, - }; - } - - // Perform statistical analysis - const statisticalResults = - await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); - - // Check if results are statistically significant - if (!statisticalResults.statisticallySignificant) { - return { - experimentId: experiment.id, - decision: 'no_winner', - reason: 'No statistically significant results found', - }; - } - - // Find the winning variant - const winner = await this.determineWinner(experiment, statisticalResults, selectionCriteria); - - if (winner) { - // Mark winner - winner.isWinner = true; - await this.variantRepository.save(winner); - - // Update experiment status - experiment.status = ExperimentStatus.COMPLETED; - experiment.endDate = new Date(); - await this.experimentRepository.save(experiment); - - return { - experimentId: experiment.id, - decision: 'winner_selected', - winnerId: winner.id, - winnerName: winner.name, - confidenceLevel: statisticalResults.confidenceLevel, - effectSize: await this.calculateEffectSizeForWinner(experiment.id, winner.id), - }; - } else { - return { - experimentId: experiment.id, - decision: 'no_winner', - reason: 'No clear winner could be determined', - }; - } - } - - /** - * Determines the winning variant based on analysis results - */ - private async determineWinner( - experiment: Experiment, - statisticalResults: any, - criteria: WinnerSelectionCriteria, - ): Promise { - const controlVariant = experiment.variants.find((v) => v.isControl); - if (!controlVariant) return null; - - let bestVariant: ExperimentVariant | null = null; - let bestPerformance = -Infinity; - - // Find the variant with the best performance that meets criteria - for (const variantAnalysis of statisticalResults.variants) { - const variant = experiment.variants.find((v) => v.id === variantAnalysis.variantId); - if (!variant || variant.isControl) continue; - - // Check minimum sample size - const hasSufficientSample = variantAnalysis.metrics.every( - (metric: any) => metric.sampleSize >= criteria.minimumSampleSize, - ); - - if (!hasSufficientSample) continue; - - // Check if statistically significant - const isSignificant = variantAnalysis.metrics.some( - (metric: any) => metric.isStatisticallySignificant, - ); - - if (!isSignificant) continue; - - // Check effect size - const effectSize = await this.calculateEffectSizeForVariant( - experiment.id, - variant.id, - controlVariant.id, - ); - if (effectSize < criteria.effectSizeThreshold) continue; - - // Compare performance (simplified - would be more complex in real implementation) - const performance = variantAnalysis.overallPerformance; - if (performance > bestPerformance) { - bestPerformance = performance; - bestVariant = variant; - } + private variantRepository: Repository, private statisticalAnalysisService: StatisticalAnalysisService) { } + /** + * Automatically selects winner for an experiment + */ + async autoSelectWinner(experimentId: string, criteria?: WinnerSelectionCriteria): Promise { + this.logger.log(`Auto-selecting winner for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + if (experiment.status !== ExperimentStatus.RUNNING) { + throw new Error('Only running experiments can have winners selected'); + } + const defaultCriteria: WinnerSelectionCriteria = { + confidenceLevel: experiment.confidenceLevel || 95, + minimumSampleSize: experiment.minimumSampleSize || 100, + effectSizeThreshold: 0.1, + durationThreshold: 7, + }; + const selectionCriteria = { ...defaultCriteria, ...criteria }; + // Check if experiment meets duration threshold + const experimentDuration = this.calculateExperimentDuration(experiment); + if (experimentDuration < selectionCriteria.durationThreshold) { + return { + experimentId: experiment.id, + decision: 'no_winner', + reason: `Experiment duration (${experimentDuration} days) below threshold (${selectionCriteria.durationThreshold} days)`, + }; + } + // Perform statistical analysis + const statisticalResults = await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); + // Check if results are statistically significant + if (!statisticalResults.statisticallySignificant) { + return { + experimentId: experiment.id, + decision: 'no_winner', + reason: 'No statistically significant results found', + }; + } + // Find the winning variant + const winner = await this.determineWinner(experiment, statisticalResults, selectionCriteria); + if (winner) { + // Mark winner + winner.isWinner = true; + await this.variantRepository.save(winner); + // Update experiment status + experiment.status = ExperimentStatus.COMPLETED; + experiment.endDate = new Date(); + await this.experimentRepository.save(experiment); + return { + experimentId: experiment.id, + decision: 'winner_selected', + winnerId: winner.id, + winnerName: winner.name, + confidenceLevel: statisticalResults.confidenceLevel, + effectSize: await this.calculateEffectSizeForWinner(experiment.id, winner.id), + }; + } + else { + return { + experimentId: experiment.id, + decision: 'no_winner', + reason: 'No clear winner could be determined', + }; + } } - - return bestVariant; - } - - /** - * Calculates effect size for a specific variant compared to control - */ - private async calculateEffectSizeForVariant( - _experimentId: string, - _variantId: string, - _controlId: string, - ): Promise { - // This would use the statistical analysis service to calculate effect size - // For now, returning a placeholder value - return 0.25; - } - - /** - * Calculates effect size for the winning variant - */ - private async calculateEffectSizeForWinner( - experimentId: string, - winnerId: string, - ): Promise { - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - const controlVariant = experiment?.variants.find((v) => v.isControl); - if (!controlVariant) return 0; - - return await this.calculateEffectSizeForVariant(experimentId, winnerId, controlVariant.id); - } - - /** - * Calculates experiment duration in days - */ - private calculateExperimentDuration(experiment: Experiment): number { - const startDate = new Date(experiment.startDate); - const endDate = experiment.endDate ? new Date(experiment.endDate) : new Date(); - const diffTime = Math.abs(endDate.getTime() - startDate.getTime()); - return Math.ceil(diffTime / (1000 * 60 * 60 * 24)); - } - - /** - * Checks if an experiment is ready for winner selection - */ - async isReadyForWinnerSelection(experimentId: string): Promise { - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment || experiment.status !== ExperimentStatus.RUNNING) { - return false; + /** + * Determines the winning variant based on analysis results + */ + private async determineWinner(experiment: Experiment, statisticalResults: unknown, criteria: WinnerSelectionCriteria): Promise { + const controlVariant = experiment.variants.find((v) => v.isControl); + if (!controlVariant) + return null; + let bestVariant: ExperimentVariant | null = null; + let bestPerformance = -Infinity; + // Find the variant with the best performance that meets criteria + for (const variantAnalysis of statisticalResults.variants) { + const variant = experiment.variants.find((v) => v.id === variantAnalysis.variantId); + if (!variant || variant.isControl) + continue; + // Check minimum sample size + const hasSufficientSample = variantAnalysis.metrics.every((metric: unknown) => metric.sampleSize >= criteria.minimumSampleSize); + if (!hasSufficientSample) + continue; + // Check if statistically significant + const isSignificant = variantAnalysis.metrics.some((metric: unknown) => metric.isStatisticallySignificant); + if (!isSignificant) + continue; + // Check effect size + const effectSize = await this.calculateEffectSizeForVariant(experiment.id, variant.id, controlVariant.id); + if (effectSize < criteria.effectSizeThreshold) + continue; + // Compare performance (simplified - would be more complex in real implementation) + const performance = variantAnalysis.overallPerformance; + if (performance > bestPerformance) { + bestPerformance = performance; + bestVariant = variant; + } + } + return bestVariant; } - - // Check if experiment has run for minimum duration - const duration = this.calculateExperimentDuration(experiment); - const minimumDuration = 7; // 7 days minimum - - if (duration < minimumDuration) { - return false; + /** + * Calculates effect size for a specific variant compared to control + */ + private async calculateEffectSizeForVariant(_experimentId: string, _variantId: string, _controlId: string): Promise { + // This would use the statistical analysis service to calculate effect size + // For now, returning a placeholder value + return 0.25; } - - // Check if all variants have sufficient sample size - const _minimumSampleSize = experiment.minimumSampleSize || 100; - - for (const _variant of experiment.variants) { - // This would check actual sample sizes from metrics - // For now, we'll assume variants are ready + /** + * Calculates effect size for the winning variant + */ + private async calculateEffectSizeForWinner(experimentId: string, winnerId: string): Promise { + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + const controlVariant = experiment?.variants.find((v) => v.isControl); + if (!controlVariant) + return 0; + return await this.calculateEffectSizeForVariant(experimentId, winnerId, controlVariant.id); } - - return true; - } - - /** - * Gets automated decision recommendations - */ - async getDecisionRecommendations(experimentId: string): Promise { - this.logger.log(`Getting decision recommendations for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + /** + * Calculates experiment duration in days + */ + private calculateExperimentDuration(experiment: Experiment): number { + const startDate = new Date(experiment.startDate); + const endDate = experiment.endDate ? new Date(experiment.endDate) : new Date(); + const diffTime = Math.abs(endDate.getTime() - startDate.getTime()); + return Math.ceil(diffTime / (1000 * 60 * 60 * 24)); } - - const recommendations = { - experimentId: experiment.id, - status: experiment.status, - readyForDecision: false, - recommendations: [] as string[], - winnerCandidate: null as string | null, - }; - - if (experiment.status !== ExperimentStatus.RUNNING) { - recommendations.recommendations.push('Experiment is not running'); - return recommendations; + /** + * Checks if an experiment is ready for winner selection + */ + async isReadyForWinnerSelection(experimentId: string): Promise { + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + if (!experiment || experiment.status !== ExperimentStatus.RUNNING) { + return false; + } + // Check if experiment has run for minimum duration + const duration = this.calculateExperimentDuration(experiment); + const minimumDuration = 7; // 7 days minimum + if (duration < minimumDuration) { + return false; + } + // Check if all variants have sufficient sample size + const _minimumSampleSize = experiment.minimumSampleSize || 100; + for (const _variant of experiment.variants) { + // This would check actual sample sizes from metrics + // For now, we'll assume variants are ready + } + return true; } - - const duration = this.calculateExperimentDuration(experiment); - recommendations.recommendations.push(`Experiment has been running for ${duration} days`); - - const ready = await this.isReadyForWinnerSelection(experimentId); - recommendations.readyForDecision = ready; - - if (ready) { - recommendations.recommendations.push('Experiment is ready for winner selection'); - - // Get potential winner - const statisticalResults = - await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); - if (statisticalResults.statisticallySignificant) { - const winner = await this.determineWinner(experiment, statisticalResults, { - confidenceLevel: experiment.confidenceLevel || 95, - minimumSampleSize: experiment.minimumSampleSize || 100, - effectSizeThreshold: 0.1, - durationThreshold: 7, + /** + * Gets automated decision recommendations + */ + async getDecisionRecommendations(experimentId: string): Promise { + this.logger.log(`Getting decision recommendations for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], }); - - if (winner) { - recommendations.winnerCandidate = winner.id; - recommendations.recommendations.push( - `Variant "${winner.name}" is the recommended winner`, - ); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); } - } - } else { - const remainingDays = Math.max(0, 7 - duration); - recommendations.recommendations.push( - `Wait ${remainingDays} more days before making decision`, - ); + const recommendations = { + experimentId: experiment.id, + status: experiment.status, + readyForDecision: false, + recommendations: [] as string[], + winnerCandidate: null as string | null, + }; + if (experiment.status !== ExperimentStatus.RUNNING) { + recommendations.recommendations.push('Experiment is not running'); + return recommendations; + } + const duration = this.calculateExperimentDuration(experiment); + recommendations.recommendations.push(`Experiment has been running for ${duration} days`); + const ready = await this.isReadyForWinnerSelection(experimentId); + recommendations.readyForDecision = ready; + if (ready) { + recommendations.recommendations.push('Experiment is ready for winner selection'); + // Get potential winner + const statisticalResults = await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); + if (statisticalResults.statisticallySignificant) { + const winner = await this.determineWinner(experiment, statisticalResults, { + confidenceLevel: experiment.confidenceLevel || 95, + minimumSampleSize: experiment.minimumSampleSize || 100, + effectSizeThreshold: 0.1, + durationThreshold: 7, + }); + if (winner) { + recommendations.winnerCandidate = winner.id; + recommendations.recommendations.push(`Variant "${winner.name}" is the recommended winner`); + } + } + } + else { + const remainingDays = Math.max(0, 7 - duration); + recommendations.recommendations.push(`Wait ${remainingDays} more days before making decision`); + } + return recommendations; } - - return recommendations; - } - - /** - * Auto-allocates traffic based on performance - */ - async autoAllocateTraffic(experimentId: string): Promise { - this.logger.log(`Auto-allocating traffic for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment || !experiment.autoAllocateTraffic) { - return; + /** + * Auto-allocates traffic based on performance + */ + async autoAllocateTraffic(experimentId: string): Promise { + this.logger.log(`Auto-allocating traffic for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + if (!experiment || !experiment.autoAllocateTraffic) { + return; + } + // This would implement multi-armed bandit algorithm or similar + // For now, we'll implement a simple performance-based allocation + const variants = experiment.variants; + if (variants.length < 2) + return; + // Calculate performance scores for each variant + const performanceScores = await this.calculateVariantPerformanceScores(variants); + // Allocate traffic proportionally to performance scores + const totalScore = performanceScores.reduce((sum, score) => sum + score.score, 0); + for (let i = 0; i < variants.length; i++) { + const variant = variants[i]; + const score = performanceScores[i]; + variant.trafficAllocation = totalScore > 0 ? score.score / totalScore : 1 / variants.length; + await this.variantRepository.save(variant); + } + this.logger.log(`Traffic auto-allocated for experiment: ${experiment.name}`); } - - // This would implement multi-armed bandit algorithm or similar - // For now, we'll implement a simple performance-based allocation - - const variants = experiment.variants; - if (variants.length < 2) return; - - // Calculate performance scores for each variant - const performanceScores = await this.calculateVariantPerformanceScores(variants); - - // Allocate traffic proportionally to performance scores - const totalScore = performanceScores.reduce((sum, score) => sum + score.score, 0); - - for (let i = 0; i < variants.length; i++) { - const variant = variants[i]; - const score = performanceScores[i]; - variant.trafficAllocation = totalScore > 0 ? score.score / totalScore : 1 / variants.length; - await this.variantRepository.save(variant); + /** + * Calculates performance scores for variants + */ + private async calculateVariantPerformanceScores(variants: ExperimentVariant[]): Promise { + // This would fetch actual performance data + // For now, returning equal scores + return variants.map((variant) => ({ + variantId: variant.id, + score: 1.0, // Placeholder score + })); } - - this.logger.log(`Traffic auto-allocated for experiment: ${experiment.name}`); - } - - /** - * Calculates performance scores for variants - */ - private async calculateVariantPerformanceScores(variants: ExperimentVariant[]): Promise { - // This would fetch actual performance data - // For now, returning equal scores - return variants.map((variant) => ({ - variantId: variant.id, - score: 1.0, // Placeholder score - })); - } } diff --git a/src/ab-testing/entities/experiment-metric.entity.ts b/src/ab-testing/entities/experiment-metric.entity.ts index 4424c76d..e57cf52a 100644 --- a/src/ab-testing/entities/experiment-metric.entity.ts +++ b/src/ab-testing/entities/experiment-metric.entity.ts @@ -1,51 +1,34 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, -} from 'typeorm'; +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, ManyToOne, } from 'typeorm'; import { Experiment } from './experiment.entity'; - export enum MetricType { - CONVERSION = 'conversion', - REVENUE = 'revenue', - ENGAGEMENT = 'engagement', - RETENTION = 'retention', - CUSTOM = 'custom', + CONVERSION = 'conversion', + REVENUE = 'revenue', + ENGAGEMENT = 'engagement', + RETENTION = 'retention', + CUSTOM = 'custom' } - @Entity({ name: 'experiment_metrics' }) export class ExperimentMetric { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - name: string; - - @Column({ type: 'text' }) - description: string; - - @Column({ - type: 'enum', - enum: MetricType, - default: MetricType.CONVERSION, - }) - type: MetricType; - - @Column({ type: 'json', nullable: true }) - configuration: any; - - @Column({ type: 'boolean', default: true }) - isPrimary: boolean; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @ManyToOne(() => Experiment, (experiment) => experiment.metrics) - experiment: Experiment; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + name: string; + @Column({ type: 'text' }) + description: string; + @Column({ + type: 'enum', + enum: MetricType, + default: MetricType.CONVERSION, + }) + type: MetricType; + @Column({ type: 'json', nullable: true }) + configuration: unknown; + @Column({ type: 'boolean', default: true }) + isPrimary: boolean; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @ManyToOne(() => Experiment, (experiment) => experiment.metrics) + experiment: Experiment; } diff --git a/src/ab-testing/entities/experiment-variant.entity.ts b/src/ab-testing/entities/experiment-variant.entity.ts index a092681c..1c0e5eb2 100644 --- a/src/ab-testing/entities/experiment-variant.entity.ts +++ b/src/ab-testing/entities/experiment-variant.entity.ts @@ -1,51 +1,30 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - OneToMany, -} from 'typeorm'; +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, OneToMany, } from 'typeorm'; import { Experiment } from './experiment.entity'; import { VariantMetric } from './variant-metric.entity'; - @Entity({ name: 'experiment_variants' }) export class ExperimentVariant { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - name: string; - - @Column({ type: 'text' }) - description: string; - - @Column({ type: 'json' }) - configuration: any; - - @Column({ type: 'decimal', precision: 5, scale: 4, default: 0 }) - trafficAllocation: number; - - @Column({ type: 'boolean', default: false }) - isControl: boolean; - - @Column({ type: 'boolean', default: false }) - isWinner: boolean; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; - - @ManyToOne(() => Experiment, (experiment) => experiment.variants) - experiment: Experiment; - - @OneToMany(() => VariantMetric, (metric) => metric.variant) - metrics: VariantMetric[]; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + name: string; + @Column({ type: 'text' }) + description: string; + @Column({ type: 'json' }) + configuration: unknown; + @Column({ type: 'decimal', precision: 5, scale: 4, default: 0 }) + trafficAllocation: number; + @Column({ type: 'boolean', default: false }) + isControl: boolean; + @Column({ type: 'boolean', default: false }) + isWinner: boolean; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; + @ManyToOne(() => Experiment, (experiment) => experiment.variants) + experiment: Experiment; + @OneToMany(() => VariantMetric, (metric) => metric.variant) + metrics: VariantMetric[]; } diff --git a/src/ab-testing/entities/experiment.entity.ts b/src/ab-testing/entities/experiment.entity.ts index 9b1c1191..36a65f06 100644 --- a/src/ab-testing/entities/experiment.entity.ts +++ b/src/ab-testing/entities/experiment.entity.ts @@ -1,89 +1,62 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, - OneToMany, -} from 'typeorm'; +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, OneToMany, } from 'typeorm'; import { ExperimentVariant } from './experiment-variant.entity'; import { ExperimentMetric } from './experiment-metric.entity'; - export enum ExperimentStatus { - DRAFT = 'draft', - RUNNING = 'running', - PAUSED = 'paused', - COMPLETED = 'completed', - ARCHIVED = 'archived', + DRAFT = 'draft', + RUNNING = 'running', + PAUSED = 'paused', + COMPLETED = 'completed', + ARCHIVED = 'archived' } - export enum ExperimentType { - A_B_TEST = 'a_b_test', - MULTIVARIATE = 'multivariate', - MULTI_ARMED_BANDIT = 'multi_armed_bandit', + A_B_TEST = 'a_b_test', + MULTIVARIATE = 'multivariate', + MULTI_ARMED_BANDIT = 'multi_armed_bandit' } - @Entity({ name: 'experiments' }) export class Experiment { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ unique: true }) - name: string; - - @Column({ type: 'text' }) - description: string; - - @Column({ - type: 'enum', - enum: ExperimentType, - default: ExperimentType.A_B_TEST, - }) - type: ExperimentType; - - @Column({ - type: 'enum', - enum: ExperimentStatus, - default: ExperimentStatus.DRAFT, - }) - status: ExperimentStatus; - - @Column({ type: 'timestamp' }) - startDate: Date; - - @Column({ type: 'timestamp', nullable: true }) - endDate?: Date; - - @Column({ type: 'decimal', precision: 5, scale: 4, default: 1.0 }) - trafficAllocation: number; - - @Column({ type: 'boolean', default: false }) - autoAllocateTraffic: boolean; - - @Column({ type: 'integer', default: 95 }) - confidenceLevel: number; - - @Column({ type: 'integer', default: 80 }) - minimumSampleSize: number; - - @Column({ type: 'text', nullable: true }) - hypothesis: string; - - @Column({ type: 'json', nullable: true }) - targetingCriteria: any; - - @Column({ type: 'json', nullable: true }) - exclusionCriteria: any; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @OneToMany(() => ExperimentVariant, (variant) => variant.experiment) - variants: ExperimentVariant[]; - - @OneToMany(() => ExperimentMetric, (metric) => metric.experiment) - metrics: ExperimentMetric[]; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ unique: true }) + name: string; + @Column({ type: 'text' }) + description: string; + @Column({ + type: 'enum', + enum: ExperimentType, + default: ExperimentType.A_B_TEST, + }) + type: ExperimentType; + @Column({ + type: 'enum', + enum: ExperimentStatus, + default: ExperimentStatus.DRAFT, + }) + status: ExperimentStatus; + @Column({ type: 'timestamp' }) + startDate: Date; + @Column({ type: 'timestamp', nullable: true }) + endDate?: Date; + @Column({ type: 'decimal', precision: 5, scale: 4, default: 1.0 }) + trafficAllocation: number; + @Column({ type: 'boolean', default: false }) + autoAllocateTraffic: boolean; + @Column({ type: 'integer', default: 95 }) + confidenceLevel: number; + @Column({ type: 'integer', default: 80 }) + minimumSampleSize: number; + @Column({ type: 'text', nullable: true }) + hypothesis: string; + @Column({ type: 'json', nullable: true }) + targetingCriteria: unknown; + @Column({ type: 'json', nullable: true }) + exclusionCriteria: unknown; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @OneToMany(() => ExperimentVariant, (variant) => variant.experiment) + variants: ExperimentVariant[]; + @OneToMany(() => ExperimentMetric, (metric) => metric.experiment) + metrics: ExperimentMetric[]; } diff --git a/src/ab-testing/entities/variant-metric.entity.ts b/src/ab-testing/entities/variant-metric.entity.ts index 8875b8d1..458631a3 100644 --- a/src/ab-testing/entities/variant-metric.entity.ts +++ b/src/ab-testing/entities/variant-metric.entity.ts @@ -1,48 +1,29 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, -} from 'typeorm'; +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, ManyToOne, } from 'typeorm'; import { ExperimentVariant } from './experiment-variant.entity'; - @Entity({ name: 'variant_metrics' }) export class VariantMetric { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'decimal', precision: 15, scale: 4, default: 0 }) - value: number; - - @Column({ type: 'integer', default: 0 }) - sampleSize: number; - - @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) - conversionRate?: number; - - @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) - standardDeviation?: number; - - @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) - confidenceIntervalLower?: number; - - @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) - confidenceIntervalUpper?: number; - - @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) - pValue?: number; - - @Column({ type: 'boolean', default: false }) - isStatisticallySignificant: boolean; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @ManyToOne(() => ExperimentVariant, (variant) => variant.metrics) - variant: ExperimentVariant; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'decimal', precision: 15, scale: 4, default: 0 }) + value: number; + @Column({ type: 'integer', default: 0 }) + sampleSize: number; + @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) + conversionRate?: number; + @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) + standardDeviation?: number; + @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) + confidenceIntervalLower?: number; + @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) + confidenceIntervalUpper?: number; + @Column({ type: 'decimal', precision: 10, scale: 4, nullable: true }) + pValue?: number; + @Column({ type: 'boolean', default: false }) + isStatisticallySignificant: boolean; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @ManyToOne(() => ExperimentVariant, (variant) => variant.metrics) + variant: ExperimentVariant; } diff --git a/src/ab-testing/experiments/experiment.service.ts b/src/ab-testing/experiments/experiment.service.ts index dac99762..787a1ad9 100644 --- a/src/ab-testing/experiments/experiment.service.ts +++ b/src/ab-testing/experiments/experiment.service.ts @@ -5,225 +5,177 @@ import { Experiment, ExperimentStatus } from '../entities/experiment.entity'; import { ExperimentVariant } from '../entities/experiment-variant.entity'; import { ExperimentMetric } from '../entities/experiment-metric.entity'; import { VariantMetric } from '../entities/variant-metric.entity'; - @Injectable() export class ExperimentService { - private readonly logger = new Logger(ExperimentService.name); - - constructor( + private readonly logger = new Logger(ExperimentService.name); + constructor( @InjectRepository(Experiment) - private experimentRepository: Repository, + private experimentRepository: Repository, @InjectRepository(ExperimentVariant) - private variantRepository: Repository, + private variantRepository: Repository, @InjectRepository(ExperimentMetric) - private experimentMetricRepository: Repository, + private experimentMetricRepository: Repository, @InjectRepository(VariantMetric) - private variantMetricRepository: Repository, - ) {} - - /** - * Updates experiment configuration - */ - async updateExperiment(id: string, updateData: Partial): Promise { - this.logger.log(`Updating experiment: ${id}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id }, - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${id} not found`); + private variantMetricRepository: Repository) { } + /** + * Updates experiment configuration + */ + async updateExperiment(id: string, updateData: Partial): Promise { + this.logger.log(`Updating experiment: ${id}`); + const experiment = await this.experimentRepository.findOne({ + where: { id }, + }); + if (!experiment) { + throw new Error(`Experiment with ID ${id} not found`); + } + Object.assign(experiment, updateData); + const updatedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment updated: ${updatedExperiment.name}`); + return updatedExperiment; } - - Object.assign(experiment, updateData); - const updatedExperiment = await this.experimentRepository.save(experiment); - - this.logger.log(`Experiment updated: ${updatedExperiment.name}`); - return updatedExperiment; - } - - /** - * Adds a variant to an experiment - */ - async addVariant( - experimentId: string, - variantData: Partial, - ): Promise { - this.logger.log(`Adding variant to experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + /** + * Adds a variant to an experiment + */ + async addVariant(experimentId: string, variantData: Partial): Promise { + this.logger.log(`Adding variant to experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + const variant = new ExperimentVariant(); + Object.assign(variant, variantData); + variant.experiment = experiment; + const savedVariant = await this.variantRepository.save(variant); + this.logger.log(`Variant added: ${savedVariant.name}`); + return savedVariant; } - - const variant = new ExperimentVariant(); - Object.assign(variant, variantData); - variant.experiment = experiment; - - const savedVariant = await this.variantRepository.save(variant); - this.logger.log(`Variant added: ${savedVariant.name}`); - return savedVariant; - } - - /** - * Removes a variant from an experiment - */ - async removeVariant(variantId: string): Promise { - this.logger.log(`Removing variant: ${variantId}`); - await this.variantRepository.softDelete(variantId); - this.logger.log(`Variant removed: ${variantId}`); - } - - /** - * Updates traffic allocation for variants - */ - async updateTrafficAllocation( - experimentId: string, - allocations: Record, - ): Promise { - this.logger.log(`Updating traffic allocation for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + /** + * Removes a variant from an experiment + */ + async removeVariant(variantId: string): Promise { + this.logger.log(`Removing variant: ${variantId}`); + await this.variantRepository.softDelete(variantId); + this.logger.log(`Variant removed: ${variantId}`); } - - // Validate that allocations sum to 100% - const totalAllocation = Object.values(allocations).reduce((sum, alloc) => sum + alloc, 0); - if (Math.abs(totalAllocation - 1) > 0.01) { - throw new Error('Traffic allocations must sum to 100%'); + /** + * Updates traffic allocation for variants + */ + async updateTrafficAllocation(experimentId: string, allocations: Record): Promise { + this.logger.log(`Updating traffic allocation for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + // Validate that allocations sum to 100% + const totalAllocation = Object.values(allocations).reduce((sum, alloc) => sum + alloc, 0); + if (Math.abs(totalAllocation - 1) > 0.01) { + throw new Error('Traffic allocations must sum to 100%'); + } + // Update each variant's allocation + for (const variant of experiment.variants) { + if (allocations[variant.id] !== undefined) { + variant.trafficAllocation = allocations[variant.id]; + await this.variantRepository.save(variant); + } + } + this.logger.log(`Traffic allocation updated for experiment: ${experiment.name}`); } - - // Update each variant's allocation - for (const variant of experiment.variants) { - if (allocations[variant.id] !== undefined) { - variant.trafficAllocation = allocations[variant.id]; - await this.variantRepository.save(variant); - } + /** + * Gets experiment results + */ + async getExperimentResults(experimentId: string): Promise { + this.logger.log(`Getting results for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants', 'metrics', 'variants.metrics'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + // Calculate results for each variant + const results = { + experiment: { + id: experiment.id, + name: experiment.name, + status: experiment.status, + type: experiment.type, + }, + variants: experiment.variants.map((variant) => ({ + id: variant.id, + name: variant.name, + isControl: variant.isControl, + isWinner: variant.isWinner, + trafficAllocation: variant.trafficAllocation, + metrics: variant.metrics.map((metric) => ({ + id: metric.id, + value: metric.value, + sampleSize: metric.sampleSize, + conversionRate: metric.conversionRate, + confidenceInterval: [metric.confidenceIntervalLower, metric.confidenceIntervalUpper], + pValue: metric.pValue, + isStatisticallySignificant: metric.isStatisticallySignificant, + })), + })), + }; + return results; } - - this.logger.log(`Traffic allocation updated for experiment: ${experiment.name}`); - } - - /** - * Gets experiment results - */ - async getExperimentResults(experimentId: string): Promise { - this.logger.log(`Getting results for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants', 'metrics', 'variants.metrics'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); + /** + * Archives an experiment + */ + async archiveExperiment(id: string): Promise { + this.logger.log(`Archiving experiment: ${id}`); + const experiment = await this.experimentRepository.findOne({ + where: { id }, + }); + if (!experiment) { + throw new Error(`Experiment with ID ${id} not found`); + } + experiment.status = ExperimentStatus.ARCHIVED; + const archivedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment archived: ${archivedExperiment.name}`); + return archivedExperiment; } - - // Calculate results for each variant - const results = { - experiment: { - id: experiment.id, - name: experiment.name, - status: experiment.status, - type: experiment.type, - }, - variants: experiment.variants.map((variant) => ({ - id: variant.id, - name: variant.name, - isControl: variant.isControl, - isWinner: variant.isWinner, - trafficAllocation: variant.trafficAllocation, - metrics: variant.metrics.map((metric) => ({ - id: metric.id, - value: metric.value, - sampleSize: metric.sampleSize, - conversionRate: metric.conversionRate, - confidenceInterval: [metric.confidenceIntervalLower, metric.confidenceIntervalUpper], - pValue: metric.pValue, - isStatisticallySignificant: metric.isStatisticallySignificant, - })), - })), - }; - - return results; - } - - /** - * Archives an experiment - */ - async archiveExperiment(id: string): Promise { - this.logger.log(`Archiving experiment: ${id}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id }, - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${id} not found`); + /** + * Pauses an experiment + */ + async pauseExperiment(id: string): Promise { + this.logger.log(`Pausing experiment: ${id}`); + const experiment = await this.experimentRepository.findOne({ + where: { id }, + }); + if (!experiment) { + throw new Error(`Experiment with ID ${id} not found`); + } + if (experiment.status !== ExperimentStatus.RUNNING) { + throw new Error('Only running experiments can be paused'); + } + experiment.status = ExperimentStatus.PAUSED; + const pausedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment paused: ${pausedExperiment.name}`); + return pausedExperiment; } - - experiment.status = ExperimentStatus.ARCHIVED; - const archivedExperiment = await this.experimentRepository.save(experiment); - - this.logger.log(`Experiment archived: ${archivedExperiment.name}`); - return archivedExperiment; - } - - /** - * Pauses an experiment - */ - async pauseExperiment(id: string): Promise { - this.logger.log(`Pausing experiment: ${id}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id }, - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${id} not found`); + /** + * Resumes a paused experiment + */ + async resumeExperiment(id: string): Promise { + this.logger.log(`Resuming experiment: ${id}`); + const experiment = await this.experimentRepository.findOne({ + where: { id }, + }); + if (!experiment) { + throw new Error(`Experiment with ID ${id} not found`); + } + if (experiment.status !== ExperimentStatus.PAUSED) { + throw new Error('Only paused experiments can be resumed'); + } + experiment.status = ExperimentStatus.RUNNING; + const resumedExperiment = await this.experimentRepository.save(experiment); + this.logger.log(`Experiment resumed: ${resumedExperiment.name}`); + return resumedExperiment; } - - if (experiment.status !== ExperimentStatus.RUNNING) { - throw new Error('Only running experiments can be paused'); - } - - experiment.status = ExperimentStatus.PAUSED; - const pausedExperiment = await this.experimentRepository.save(experiment); - - this.logger.log(`Experiment paused: ${pausedExperiment.name}`); - return pausedExperiment; - } - - /** - * Resumes a paused experiment - */ - async resumeExperiment(id: string): Promise { - this.logger.log(`Resuming experiment: ${id}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id }, - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${id} not found`); - } - - if (experiment.status !== ExperimentStatus.PAUSED) { - throw new Error('Only paused experiments can be resumed'); - } - - experiment.status = ExperimentStatus.RUNNING; - const resumedExperiment = await this.experimentRepository.save(experiment); - - this.logger.log(`Experiment resumed: ${resumedExperiment.name}`); - return resumedExperiment; - } } diff --git a/src/ab-testing/reporting/ab-testing-reports.service.ts b/src/ab-testing/reporting/ab-testing-reports.service.ts index ed6b1f1c..6259b06b 100644 --- a/src/ab-testing/reporting/ab-testing-reports.service.ts +++ b/src/ab-testing/reporting/ab-testing-reports.service.ts @@ -5,322 +5,258 @@ import { Experiment, ExperimentStatus, ExperimentType } from '../entities/experi import { ExperimentVariant } from '../entities/experiment-variant.entity'; import { StatisticalAnalysisService } from '../analysis/statistical-analysis.service'; import { AutomatedDecisionService } from '../automation/automated-decision.service'; - export interface ReportFilters { - status?: ExperimentStatus; - type?: ExperimentType; - startDate?: Date; - endDate?: Date; - includeArchived?: boolean; + status?: ExperimentStatus; + type?: ExperimentType; + startDate?: Date; + endDate?: Date; + includeArchived?: boolean; } - @Injectable() export class ABTestingReportsService { - private readonly logger = new Logger(ABTestingReportsService.name); - - constructor( + private readonly logger = new Logger(ABTestingReportsService.name); + constructor( @InjectRepository(Experiment) - private experimentRepository: Repository, + private experimentRepository: Repository, @InjectRepository(ExperimentVariant) - private variantRepository: Repository, - private statisticalAnalysisService: StatisticalAnalysisService, - private automatedDecisionService: AutomatedDecisionService, - ) {} - - /** - * Generates comprehensive experiment report - */ - async generateExperimentReport(experimentId: string): Promise { - this.logger.log(`Generating report for experiment: ${experimentId}`); - - const experiment = await this.experimentRepository.findOne({ - where: { id: experimentId }, - relations: ['variants', 'metrics', 'variants.metrics'], - }); - - if (!experiment) { - throw new Error(`Experiment with ID ${experimentId} not found`); - } - - const statisticalAnalysis = - await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); - const decisionRecommendations = - await this.automatedDecisionService.getDecisionRecommendations(experimentId); - - const report = { - experiment: { - id: experiment.id, - name: experiment.name, - description: experiment.description, - type: experiment.type, - status: experiment.status, - startDate: experiment.startDate, - endDate: experiment.endDate, - duration: this.calculateExperimentDuration(experiment), - hypothesis: experiment.hypothesis, - confidenceLevel: experiment.confidenceLevel, - minimumSampleSize: experiment.minimumSampleSize, - trafficAllocation: experiment.trafficAllocation, - }, - variants: experiment.variants.map((variant) => ({ - id: variant.id, - name: variant.name, - description: variant.description, - isControl: variant.isControl, - isWinner: variant.isWinner, - trafficAllocation: variant.trafficAllocation, - configuration: variant.configuration, - metrics: variant.metrics.map((metric) => ({ - id: metric.id, - value: metric.value, - sampleSize: metric.sampleSize, - conversionRate: metric.conversionRate, - standardDeviation: metric.standardDeviation, - confidenceInterval: [metric.confidenceIntervalLower, metric.confidenceIntervalUpper], - pValue: metric.pValue, - isStatisticallySignificant: metric.isStatisticallySignificant, - })), - })), - statisticalAnalysis, - decisionRecommendations, - summary: this.generateSummary(experiment, statisticalAnalysis), - }; - - return report; - } - - /** - * Generates summary statistics for the experiment - */ - private generateSummary(experiment: Experiment, statisticalAnalysis: any): any { - const controlVariant = experiment.variants.find((v) => v.isControl); - const winnerVariant = experiment.variants.find((v) => v.isWinner); - - return { - totalVariants: experiment.variants.length, - controlVariant: controlVariant ? controlVariant.name : null, - winner: winnerVariant ? winnerVariant.name : null, - isStatisticallySignificant: statisticalAnalysis.statisticallySignificant, - duration: this.calculateExperimentDuration(experiment), - status: experiment.status, - recommendations: statisticalAnalysis.statisticallySignificant - ? 'Statistically significant results found' - : 'Continue running experiment for more data', - }; - } - - /** - * Gets dashboard summary of all experiments - */ - async getDashboardSummary(filters?: ReportFilters): Promise { - this.logger.log('Generating dashboard summary'); - - const experiments = await this.getFilteredExperiments(filters); - - const summary = { - totalExperiments: experiments.length, - experimentsByStatus: this.groupExperimentsByStatus(experiments), - experimentsByType: this.groupExperimentsByType(experiments), - runningExperiments: experiments.filter((e) => e.status === ExperimentStatus.RUNNING).length, - completedExperiments: experiments.filter((e) => e.status === ExperimentStatus.COMPLETED) - .length, - recentExperiments: experiments.slice(0, 5), // Last 5 experiments - upcomingExperiments: experiments - .filter((e) => e.status === ExperimentStatus.DRAFT && e.startDate > new Date()) - .slice(0, 5), - }; - - return summary; - } - - /** - * Gets filtered experiments based on criteria - */ - private async getFilteredExperiments(filters?: ReportFilters): Promise { - const queryBuilder = this.experimentRepository.createQueryBuilder('experiment'); - - if (filters?.status) { - queryBuilder.andWhere('experiment.status = :status', { status: filters.status }); + private variantRepository: Repository, private statisticalAnalysisService: StatisticalAnalysisService, private automatedDecisionService: AutomatedDecisionService) { } + /** + * Generates comprehensive experiment report + */ + async generateExperimentReport(experimentId: string): Promise { + this.logger.log(`Generating report for experiment: ${experimentId}`); + const experiment = await this.experimentRepository.findOne({ + where: { id: experimentId }, + relations: ['variants', 'metrics', 'variants.metrics'], + }); + if (!experiment) { + throw new Error(`Experiment with ID ${experimentId} not found`); + } + const statisticalAnalysis = await this.statisticalAnalysisService.calculateStatisticalSignificance(experimentId); + const decisionRecommendations = await this.automatedDecisionService.getDecisionRecommendations(experimentId); + const report = { + experiment: { + id: experiment.id, + name: experiment.name, + description: experiment.description, + type: experiment.type, + status: experiment.status, + startDate: experiment.startDate, + endDate: experiment.endDate, + duration: this.calculateExperimentDuration(experiment), + hypothesis: experiment.hypothesis, + confidenceLevel: experiment.confidenceLevel, + minimumSampleSize: experiment.minimumSampleSize, + trafficAllocation: experiment.trafficAllocation, + }, + variants: experiment.variants.map((variant) => ({ + id: variant.id, + name: variant.name, + description: variant.description, + isControl: variant.isControl, + isWinner: variant.isWinner, + trafficAllocation: variant.trafficAllocation, + configuration: variant.configuration, + metrics: variant.metrics.map((metric) => ({ + id: metric.id, + value: metric.value, + sampleSize: metric.sampleSize, + conversionRate: metric.conversionRate, + standardDeviation: metric.standardDeviation, + confidenceInterval: [metric.confidenceIntervalLower, metric.confidenceIntervalUpper], + pValue: metric.pValue, + isStatisticallySignificant: metric.isStatisticallySignificant, + })), + })), + statisticalAnalysis, + decisionRecommendations, + summary: this.generateSummary(experiment, statisticalAnalysis), + }; + return report; } - - if (filters?.type) { - queryBuilder.andWhere('experiment.type = :type', { type: filters.type }); + /** + * Generates summary statistics for the experiment + */ + private generateSummary(experiment: Experiment, statisticalAnalysis: unknown): unknown { + const controlVariant = experiment.variants.find((v) => v.isControl); + const winnerVariant = experiment.variants.find((v) => v.isWinner); + return { + totalVariants: experiment.variants.length, + controlVariant: controlVariant ? controlVariant.name : null, + winner: winnerVariant ? winnerVariant.name : null, + isStatisticallySignificant: statisticalAnalysis.statisticallySignificant, + duration: this.calculateExperimentDuration(experiment), + status: experiment.status, + recommendations: statisticalAnalysis.statisticallySignificant + ? 'Statistically significant results found' + : 'Continue running experiment for more data', + }; } - - if (filters?.startDate) { - queryBuilder.andWhere('experiment.startDate >= :startDate', { startDate: filters.startDate }); + /** + * Gets dashboard summary of all experiments + */ + async getDashboardSummary(filters?: ReportFilters): Promise { + this.logger.log('Generating dashboard summary'); + const experiments = await this.getFilteredExperiments(filters); + const summary = { + totalExperiments: experiments.length, + experimentsByStatus: this.groupExperimentsByStatus(experiments), + experimentsByType: this.groupExperimentsByType(experiments), + runningExperiments: experiments.filter((e) => e.status === ExperimentStatus.RUNNING).length, + completedExperiments: experiments.filter((e) => e.status === ExperimentStatus.COMPLETED) + .length, + recentExperiments: experiments.slice(0, 5), // Last 5 experiments + upcomingExperiments: experiments + .filter((e) => e.status === ExperimentStatus.DRAFT && e.startDate > new Date()) + .slice(0, 5), + }; + return summary; } - - if (filters?.endDate) { - queryBuilder.andWhere('experiment.startDate <= :endDate', { endDate: filters.endDate }); + /** + * Gets filtered experiments based on criteria + */ + private async getFilteredExperiments(filters?: ReportFilters): Promise { + const queryBuilder = this.experimentRepository.createQueryBuilder('experiment'); + if (filters?.status) { + queryBuilder.andWhere('experiment.status = :status', { status: filters.status }); + } + if (filters?.type) { + queryBuilder.andWhere('experiment.type = :type', { type: filters.type }); + } + if (filters?.startDate) { + queryBuilder.andWhere('experiment.startDate >= :startDate', { startDate: filters.startDate }); + } + if (filters?.endDate) { + queryBuilder.andWhere('experiment.startDate <= :endDate', { endDate: filters.endDate }); + } + if (!filters?.includeArchived) { + queryBuilder.andWhere('experiment.status != :archived', { + archived: ExperimentStatus.ARCHIVED, + }); + } + queryBuilder.orderBy('experiment.createdAt', 'DESC'); + return await queryBuilder.getMany(); } - - if (!filters?.includeArchived) { - queryBuilder.andWhere('experiment.status != :archived', { - archived: ExperimentStatus.ARCHIVED, - }); + /** + * Groups experiments by status + */ + private groupExperimentsByStatus(experiments: Experiment[]): Record { + const statusGroups: Record = {}; + for (const experiment of experiments) { + const status = experiment.status; + statusGroups[status] = (statusGroups[status] || 0) + 1; + } + return statusGroups; } - - queryBuilder.orderBy('experiment.createdAt', 'DESC'); - - return await queryBuilder.getMany(); - } - - /** - * Groups experiments by status - */ - private groupExperimentsByStatus(experiments: Experiment[]): Record { - const statusGroups: Record = {}; - - for (const experiment of experiments) { - const status = experiment.status; - statusGroups[status] = (statusGroups[status] || 0) + 1; + /** + * Groups experiments by type + */ + private groupExperimentsByType(experiments: Experiment[]): Record { + const typeGroups: Record = {}; + for (const experiment of experiments) { + const type = experiment.type; + typeGroups[type] = (typeGroups[type] || 0) + 1; + } + return typeGroups; } - - return statusGroups; - } - - /** - * Groups experiments by type - */ - private groupExperimentsByType(experiments: Experiment[]): Record { - const typeGroups: Record = {}; - - for (const experiment of experiments) { - const type = experiment.type; - typeGroups[type] = (typeGroups[type] || 0) + 1; + /** + * Calculates experiment duration in days + */ + private calculateExperimentDuration(experiment: Experiment): number { + const startDate = new Date(experiment.startDate); + const endDate = experiment.endDate ? new Date(experiment.endDate) : new Date(); + const diffTime = Math.abs(endDate.getTime() - startDate.getTime()); + return Math.ceil(diffTime / (1000 * 60 * 60 * 24)); } - - return typeGroups; - } - - /** - * Calculates experiment duration in days - */ - private calculateExperimentDuration(experiment: Experiment): number { - const startDate = new Date(experiment.startDate); - const endDate = experiment.endDate ? new Date(experiment.endDate) : new Date(); - const diffTime = Math.abs(endDate.getTime() - startDate.getTime()); - return Math.ceil(diffTime / (1000 * 60 * 60 * 24)); - } - - /** - * Generates performance comparison report - */ - async generatePerformanceComparisonReport(): Promise { - this.logger.log('Generating performance comparison report'); - - const experiments = await this.experimentRepository.find({ - where: { status: ExperimentStatus.COMPLETED }, - relations: ['variants'], - order: { createdAt: 'DESC' }, - take: 20, - }); - - const performanceData = []; - - for (const experiment of experiments) { - const winner = experiment.variants.find((v) => v.isWinner); - const control = experiment.variants.find((v) => v.isControl); - - if (winner && control && winner.id !== control.id) { - const improvement = await this.calculateImprovementPercentage( - experiment.id, - winner.id, - control.id, - ); - - performanceData.push({ - experimentId: experiment.id, - experimentName: experiment.name, - experimentType: experiment.type, - winnerVariant: winner.name, - controlVariant: control.name, - improvementPercentage: improvement, - duration: this.calculateExperimentDuration(experiment), - confidenceLevel: experiment.confidenceLevel, + /** + * Generates performance comparison report + */ + async generatePerformanceComparisonReport(): Promise { + this.logger.log('Generating performance comparison report'); + const experiments = await this.experimentRepository.find({ + where: { status: ExperimentStatus.COMPLETED }, + relations: ['variants'], + order: { createdAt: 'DESC' }, + take: 20, }); - } + const performanceData = []; + for (const experiment of experiments) { + const winner = experiment.variants.find((v) => v.isWinner); + const control = experiment.variants.find((v) => v.isControl); + if (winner && control && winner.id !== control.id) { + const improvement = await this.calculateImprovementPercentage(experiment.id, winner.id, control.id); + performanceData.push({ + experimentId: experiment.id, + experimentName: experiment.name, + experimentType: experiment.type, + winnerVariant: winner.name, + controlVariant: control.name, + improvementPercentage: improvement, + duration: this.calculateExperimentDuration(experiment), + confidenceLevel: experiment.confidenceLevel, + }); + } + } + return { + reportTitle: 'Performance Comparison Report', + generatedAt: new Date(), + totalComparisons: performanceData.length, + averageImprovement: performanceData.length > 0 + ? performanceData.reduce((sum, data) => sum + data.improvementPercentage, 0) / + performanceData.length + : 0, + bestPerforming: performanceData.length > 0 + ? [...performanceData].sort((a, b) => b.improvementPercentage - a.improvementPercentage)[0] + : null, + performanceData, + }; + } + /** + * Calculates improvement percentage between winner and control + */ + private async calculateImprovementPercentage(_experimentId: string, _winnerId: string, _controlId: string): Promise { + // This would fetch actual metric data and calculate improvement + // For now, returning a placeholder value + return 15.5; // 15.5% improvement } - - return { - reportTitle: 'Performance Comparison Report', - generatedAt: new Date(), - totalComparisons: performanceData.length, - averageImprovement: - performanceData.length > 0 - ? performanceData.reduce((sum, data) => sum + data.improvementPercentage, 0) / - performanceData.length - : 0, - bestPerforming: - performanceData.length > 0 - ? [...performanceData].sort( - (a, b) => b.improvementPercentage - a.improvementPercentage, - )[0] - : null, - performanceData, - }; - } - - /** - * Calculates improvement percentage between winner and control - */ - private async calculateImprovementPercentage( - _experimentId: string, - _winnerId: string, - _controlId: string, - ): Promise { - // This would fetch actual metric data and calculate improvement - // For now, returning a placeholder value - return 15.5; // 15.5% improvement - } - - /** - * Exports experiment data in CSV format - */ - async exportExperimentData(experimentId: string): Promise { - this.logger.log(`Exporting data for experiment: ${experimentId}`); - - const report = await this.generateExperimentReport(experimentId); - - // Convert report to CSV format - let csv = - 'Metric,Variant,Value,Sample Size,Conversion Rate,Confidence Interval,P-Value,Statistically Significant\n'; - - for (const variant of report.variants) { - for (const metric of variant.metrics) { - csv += `${metric.id},${variant.name},${metric.value},${metric.sampleSize},${metric.conversionRate || ''},`; - csv += `"${metric.confidenceInterval ? `[${metric.confidenceInterval[0]}, ${metric.confidenceInterval[1]}]` : ''}",`; - csv += `${metric.pValue || ''},${metric.isStatisticallySignificant}\n`; - } + /** + * Exports experiment data in CSV format + */ + async exportExperimentData(experimentId: string): Promise { + this.logger.log(`Exporting data for experiment: ${experimentId}`); + const report = await this.generateExperimentReport(experimentId); + // Convert report to CSV format + let csv = 'Metric,Variant,Value,Sample Size,Conversion Rate,Confidence Interval,P-Value,Statistically Significant\n'; + for (const variant of report.variants) { + for (const metric of variant.metrics) { + csv += `${metric.id},${variant.name},${metric.value},${metric.sampleSize},${metric.conversionRate || ''},`; + csv += `"${metric.confidenceInterval ? `[${metric.confidenceInterval[0]}, ${metric.confidenceInterval[1]}]` : ''}",`; + csv += `${metric.pValue || ''},${metric.isStatisticallySignificant}\n`; + } + } + return csv; + } + /** + * Gets experiment timeline data + */ + async getExperimentTimeline(): Promise { + const experiments = await this.experimentRepository.find({ + order: { startDate: 'ASC' }, + }); + const timeline = experiments.map((experiment) => ({ + id: experiment.id, + name: experiment.name, + startDate: experiment.startDate, + endDate: experiment.endDate || new Date(), + status: experiment.status, + type: experiment.type, + duration: this.calculateExperimentDuration(experiment), + })); + return { + timeline, + totalExperiments: timeline.length, + startDate: timeline.length > 0 ? timeline[0].startDate : null, + endDate: timeline.length > 0 ? timeline[timeline.length - 1].endDate : null, + }; } - - return csv; - } - - /** - * Gets experiment timeline data - */ - async getExperimentTimeline(): Promise { - const experiments = await this.experimentRepository.find({ - order: { startDate: 'ASC' }, - }); - - const timeline = experiments.map((experiment) => ({ - id: experiment.id, - name: experiment.name, - startDate: experiment.startDate, - endDate: experiment.endDate || new Date(), - status: experiment.status, - type: experiment.type, - duration: this.calculateExperimentDuration(experiment), - })); - - return { - timeline, - totalExperiments: timeline.length, - startDate: timeline.length > 0 ? timeline[0].startDate : null, - endDate: timeline.length > 0 ? timeline[timeline.length - 1].endDate : null, - }; - } } diff --git a/src/app.controller.spec.ts b/src/app.controller.spec.ts index d22f3890..2908bafc 100644 --- a/src/app.controller.spec.ts +++ b/src/app.controller.spec.ts @@ -1,22 +1,18 @@ import { Test, TestingModule } from '@nestjs/testing'; import { AppController } from './app.controller'; import { AppService } from './app.service'; - describe('AppController', () => { - let appController: AppController; - - beforeEach(async () => { - const app: TestingModule = await Test.createTestingModule({ - controllers: [AppController], - providers: [AppService], - }).compile(); - - appController = app.get(AppController); - }); - - describe('root', () => { - it('should return "Hello World!"', () => { - expect(appController.getHello()).toBe('Hello World!'); + let appController: AppController; + beforeEach(async () => { + const app: TestingModule = await Test.createTestingModule({ + controllers: [AppController], + providers: [AppService], + }).compile(); + appController = app.get(AppController); + }); + describe('root', () => { + it('should return "Hello World!"', () => { + expect(appController.getHello()).toBe('Hello World!'); + }); }); - }); }); diff --git a/src/app.controller.ts b/src/app.controller.ts index fe159320..dec450e1 100644 --- a/src/app.controller.ts +++ b/src/app.controller.ts @@ -1,15 +1,13 @@ import { Controller, Get, HttpStatus } from '@nestjs/common'; import { ApiResponse, ApiTags } from '@nestjs/swagger'; import { AppService } from './app.service'; - @ApiTags('app') @Controller() export class AppController { - constructor(private readonly appService: AppService) {} - - @Get() - @ApiResponse({ status: HttpStatus.OK, description: 'Root endpoint response' }) - getHello(): string { - return this.appService.getHello(); - } + constructor(private readonly appService: AppService) { } + @Get() + @ApiResponse({ status: HttpStatus.OK, description: 'Root endpoint response' }) + getHello(): string { + return this.appService.getHello(); + } } diff --git a/src/app.module.ts b/src/app.module.ts index 2d02a2e8..d0fc753f 100644 --- a/src/app.module.ts +++ b/src/app.module.ts @@ -23,7 +23,6 @@ import { loadFeatureFlags } from './config/feature-flags.config'; import { StartupLogger } from './common/lazy-loading/startup-logger.service'; import { TimeoutInterceptor } from './common/interceptors/timeout.interceptor'; import { ApiVersioningModule } from './common/modules/api-versioning.module'; - // Feature modules - conditionally loaded based on feature flags import { SyncModule } from './sync/sync.module'; import { MediaModule } from './media/media.module'; @@ -55,350 +54,338 @@ import { LocalizationModule } from './localization/localization.module'; import { CsrfModule } from './common/csrf/csrf.module'; import { TimeoutModule } from './common/timeout/timeout.module'; import { ShutdownStateService } from './common/services/shutdown-state.service'; - @Global() @Module({}) export class AppModule { - static async forRoot(): Promise { - const flags = loadFeatureFlags(); - const startupLogger = new StartupLogger(); - - // Core modules - always loaded - const coreModules = [ - ConfigModule.forRoot({ - isGlobal: true, - validationSchema: envValidationSchema, - }), - TypeOrmModule.forRootAsync({ - imports: [MonitoringModule], - inject: [MetricsCollectionService], - useFactory: (metricsService: MetricsCollectionService) => { - // Tune postgres pool to avoid connection exhaustion in high-traffic workloads. - // Values can be overridden with DATABASE_POOL_* environment variables. - const poolMax = parseInt(process.env.DATABASE_POOL_MAX || '30', 10); - const poolMin = parseInt(process.env.DATABASE_POOL_MIN || '5', 10); - const poolAcquireTimeoutMs = parseInt( - process.env.DATABASE_POOL_ACQUIRE_TIMEOUT_MS || '10000', - 10, - ); - const poolIdleTimeoutMs = parseInt( - process.env.DATABASE_POOL_IDLE_TIMEOUT_MS || '30000', - 10, - ); - - return { - type: 'postgres', - host: process.env.DATABASE_HOST || 'localhost', - port: parseInt(process.env.DATABASE_PORT || '5432'), - username: process.env.DATABASE_USER || 'postgres', - password: process.env.DATABASE_PASSWORD || 'postgres', - database: process.env.DATABASE_NAME || 'teachlink', - autoLoadEntities: true, - synchronize: process.env.NODE_ENV !== 'production', - logging: true, - logger: new TypeOrmMonitoringLogger(metricsService), - maxQueryExecutionTime: 1000, - extra: { - // pg Pool options used by TypeORM postgres driver - max: poolMax, - min: poolMin, - connectionTimeoutMillis: poolAcquireTimeoutMs, - idleTimeoutMillis: poolIdleTimeoutMs, - }, - }; - }, - }), - MonitoringModule, - EventEmitterModule.forRoot(), - BullModule.forRoot({ - redis: { - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379', 10), - }, - createClient: createBullRedisClient, - }), - CacheModule.register({ - isGlobal: true, - store: redisStore, - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379'), - }), - SessionModule, - ThrottlerModule.forRoot([ - { - ttl: parseInt(process.env.THROTTLE_TTL || '60'), - limit: parseInt(process.env.THROTTLE_LIMIT || '60'), - }, - ]), - ApiVersioningModule, - HealthModule, - DatabaseModule, - CsrfModule, - TimeoutModule, - ]; - - // Feature modules - conditionally loaded based on feature flags - const featureModules: Array> = []; - - // Auth Module - if (flags.ENABLE_AUTH) { - const startTime = Date.now(); - featureModules.push(AuthModule); - startupLogger.recordModuleLoaded('AuthModule', startTime); - } else { - startupLogger.recordModuleSkipped('AuthModule', 'ENABLE_AUTH=false'); + static async forRoot(): Promise { + const flags = loadFeatureFlags(); + const startupLogger = new StartupLogger(); + // Core modules - always loaded + const coreModules = [ + ConfigModule.forRoot({ + isGlobal: true, + validationSchema: envValidationSchema, + }), + TypeOrmModule.forRootAsync({ + imports: [MonitoringModule], + inject: [MetricsCollectionService], + useFactory: (metricsService: MetricsCollectionService) => { + // Tune postgres pool to avoid connection exhaustion in high-traffic workloads. + // Values can be overridden with DATABASE_POOL_* environment variables. + const poolMax = parseInt(process.env.DATABASE_POOL_MAX || '30', 10); + const poolMin = parseInt(process.env.DATABASE_POOL_MIN || '5', 10); + const poolAcquireTimeoutMs = parseInt(process.env.DATABASE_POOL_ACQUIRE_TIMEOUT_MS || '10000', 10); + const poolIdleTimeoutMs = parseInt(process.env.DATABASE_POOL_IDLE_TIMEOUT_MS || '30000', 10); + return { + type: 'postgres', + host: process.env.DATABASE_HOST || 'localhost', + port: parseInt(process.env.DATABASE_PORT || '5432'), + username: process.env.DATABASE_USER || 'postgres', + password: process.env.DATABASE_PASSWORD || 'postgres', + database: process.env.DATABASE_NAME || 'teachlink', + autoLoadEntities: true, + synchronize: process.env.NODE_ENV !== 'production', + logging: true, + logger: new TypeOrmMonitoringLogger(metricsService), + maxQueryExecutionTime: 1000, + extra: { + // pg Pool options used by TypeORM postgres driver + max: poolMax, + min: poolMin, + connectionTimeoutMillis: poolAcquireTimeoutMs, + idleTimeoutMillis: poolIdleTimeoutMs, + }, + }; + }, + }), + MonitoringModule, + EventEmitterModule.forRoot(), + BullModule.forRoot({ + redis: { + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379', 10), + }, + createClient: createBullRedisClient, + }), + CacheModule.register({ + isGlobal: true, + store: redisStore, + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379'), + }), + SessionModule, + ThrottlerModule.forRoot([ + { + ttl: parseInt(process.env.THROTTLE_TTL || '60'), + limit: parseInt(process.env.THROTTLE_LIMIT || '60'), + }, + ]), + ApiVersioningModule, + HealthModule, + DatabaseModule, + CsrfModule, + TimeoutModule, + ]; + // Feature modules - conditionally loaded based on feature flags + const featureModules: Array> = []; + // Auth Module + if (flags.ENABLE_AUTH) { + const startTime = Date.now(); + featureModules.push(AuthModule); + startupLogger.recordModuleLoaded('AuthModule', startTime); + } + else { + startupLogger.recordModuleSkipped('AuthModule', 'ENABLE_AUTH=false'); + } + // Payments Module + if (flags.ENABLE_PAYMENTS) { + const startTime = Date.now(); + featureModules.push(PaymentsModule); + startupLogger.recordModuleLoaded('PaymentsModule', startTime); + } + else { + startupLogger.recordModuleSkipped('PaymentsModule', 'ENABLE_PAYMENTS=false'); + } + // AB Testing Module + if (flags.ENABLE_AB_TESTING) { + const startTime = Date.now(); + featureModules.push(ABTestingModule); + startupLogger.recordModuleLoaded('ABTestingModule', startTime); + } + else { + startupLogger.recordModuleSkipped('ABTestingModule', 'ENABLE_AB_TESTING=false'); + } + // Data Warehouse Module + if (flags.ENABLE_DATA_WAREHOUSE) { + const startTime = Date.now(); + featureModules.push(DataWarehouseModule); + startupLogger.recordModuleLoaded('DataWarehouseModule', startTime); + } + else { + startupLogger.recordModuleSkipped('DataWarehouseModule', 'ENABLE_DATA_WAREHOUSE=false'); + } + // Collaboration Module + if (flags.ENABLE_COLLABORATION) { + const startTime = Date.now(); + featureModules.push(CollaborationModule); + startupLogger.recordModuleLoaded('CollaborationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('CollaborationModule', 'ENABLE_COLLABORATION=false'); + } + // Media Module + if (flags.ENABLE_MEDIA_PROCESSING) { + const startTime = Date.now(); + featureModules.push(MediaModule); + startupLogger.recordModuleLoaded('MediaModule', startTime); + } + else { + startupLogger.recordModuleSkipped('MediaModule', 'ENABLE_MEDIA_PROCESSING=false'); + } + // Backup Module + if (flags.ENABLE_BACKUP) { + const startTime = Date.now(); + featureModules.push(BackupModule); + startupLogger.recordModuleLoaded('BackupModule', startTime); + } + else { + startupLogger.recordModuleSkipped('BackupModule', 'ENABLE_BACKUP=false'); + } + // GraphQL Module + if (flags.ENABLE_GRAPHQL) { + const startTime = Date.now(); + featureModules.push(GraphQLModule); + startupLogger.recordModuleLoaded('GraphQLModule', startTime); + } + else { + startupLogger.recordModuleSkipped('GraphQLModule', 'ENABLE_GRAPHQL=false'); + } + // Sync Module + if (flags.ENABLE_SYNC) { + const startTime = Date.now(); + featureModules.push(SyncModule); + startupLogger.recordModuleLoaded('SyncModule', startTime); + } + else { + startupLogger.recordModuleSkipped('SyncModule', 'ENABLE_SYNC=false'); + } + // Migration Module + if (flags.ENABLE_MIGRATIONS) { + const startTime = Date.now(); + featureModules.push(MigrationModule); + startupLogger.recordModuleLoaded('MigrationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('MigrationModule', 'ENABLE_MIGRATIONS=false'); + } + // Rate Limiting Module + if (flags.ENABLE_RATE_LIMITING) { + const startTime = Date.now(); + featureModules.push(RateLimitingModule); + startupLogger.recordModuleLoaded('RateLimitingModule', startTime); + } + else { + startupLogger.recordModuleSkipped('RateLimitingModule', 'ENABLE_RATE_LIMITING=false'); + } + // Observability Module + if (flags.ENABLE_OBSERVABILITY) { + const startTime = Date.now(); + featureModules.push(ObservabilityModule); + startupLogger.recordModuleLoaded('ObservabilityModule', startTime); + } + else { + startupLogger.recordModuleSkipped('ObservabilityModule', 'ENABLE_OBSERVABILITY=false'); + } + // Caching Module + if (flags.ENABLE_CACHING) { + const startTime = Date.now(); + featureModules.push(CachingModule); + startupLogger.recordModuleLoaded('CachingModule', startTime); + } + else { + startupLogger.recordModuleSkipped('CachingModule', 'ENABLE_CACHING=false'); + } + // Feature Flags Module + if (flags.ENABLE_FEATURE_FLAGS) { + const startTime = Date.now(); + featureModules.push(FeatureFlagsModule); + startupLogger.recordModuleLoaded('FeatureFlagsModule', startTime); + } + else { + startupLogger.recordModuleSkipped('FeatureFlagsModule', 'ENABLE_FEATURE_FLAGS=false'); + } + // Search Module + if (flags.ENABLE_SEARCH) { + const startTime = Date.now(); + featureModules.push(SearchModule); + startupLogger.recordModuleLoaded('SearchModule', startTime); + } + else { + startupLogger.recordModuleSkipped('SearchModule', 'ENABLE_SEARCH=false'); + } + // Notifications Module + if (flags.ENABLE_NOTIFICATIONS) { + const startTime = Date.now(); + featureModules.push(NotificationsModule); + startupLogger.recordModuleLoaded('NotificationsModule', startTime); + } + else { + startupLogger.recordModuleSkipped('NotificationsModule', 'ENABLE_NOTIFICATIONS=false'); + } + // Email Marketing Module + if (flags.ENABLE_EMAIL_MARKETING) { + const startTime = Date.now(); + featureModules.push(EmailMarketingModule); + startupLogger.recordModuleLoaded('EmailMarketingModule', startTime); + } + else { + startupLogger.recordModuleSkipped('EmailMarketingModule', 'ENABLE_EMAIL_MARKETING=false'); + } + // Gamification Module + if (flags.ENABLE_GAMIFICATION) { + const startTime = Date.now(); + featureModules.push(GamificationModule); + startupLogger.recordModuleLoaded('GamificationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('GamificationModule', 'ENABLE_GAMIFICATION=false'); + } + // Assessment Module + if (flags.ENABLE_ASSESSMENT) { + const startTime = Date.now(); + featureModules.push(AssessmentsModule); + startupLogger.recordModuleLoaded('AssessmentModule', startTime); + } + else { + startupLogger.recordModuleSkipped('AssessmentModule', 'ENABLE_ASSESSMENT=false'); + } + // Learning Paths Module + if (flags.ENABLE_LEARNING_PATHS) { + const startTime = Date.now(); + featureModules.push(LearningPathsModule); + startupLogger.recordModuleLoaded('LearningPathsModule', startTime); + } + else { + startupLogger.recordModuleSkipped('LearningPathsModule', 'ENABLE_LEARNING_PATHS=false'); + } + // Moderation Module + if (flags.ENABLE_MODERATION) { + const startTime = Date.now(); + featureModules.push(ModerationModule); + startupLogger.recordModuleLoaded('ModerationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('ModerationModule', 'ENABLE_MODERATION=false'); + } + // Orchestration Module + if (flags.ENABLE_ORCHESTRATION) { + const startTime = Date.now(); + featureModules.push(OrchestrationModule); + startupLogger.recordModuleLoaded('OrchestrationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('OrchestrationModule', 'ENABLE_ORCHESTRATION=false'); + } + // Security Module + if (flags.ENABLE_SECURITY) { + const startTime = Date.now(); + featureModules.push(SecurityModule); + startupLogger.recordModuleLoaded('SecurityModule', startTime); + } + else { + startupLogger.recordModuleSkipped('SecurityModule', 'ENABLE_SECURITY=false'); + } + // Tenancy Module + if (flags.ENABLE_TENANCY) { + const startTime = Date.now(); + featureModules.push(TenancyModule); + startupLogger.recordModuleLoaded('TenancyModule', startTime); + } + else { + startupLogger.recordModuleSkipped('TenancyModule', 'ENABLE_TENANCY=false'); + } + // CDN Module + if (flags.ENABLE_CDN) { + const startTime = Date.now(); + featureModules.push(CdnModule); + startupLogger.recordModuleLoaded('CDNModule', startTime); + } + else { + startupLogger.recordModuleSkipped('CDNModule', 'ENABLE_CDN=false'); + } + // Localization Module + if (flags.ENABLE_LOCALIZATION) { + const startTime = Date.now(); + featureModules.push(LocalizationModule); + startupLogger.recordModuleLoaded('LocalizationModule', startTime); + } + else { + startupLogger.recordModuleSkipped('LocalizationModule', 'ENABLE_LOCALIZATION=false'); + } + // Queue Module (always loaded for Bull) + featureModules.push(QueueModule); + return { + module: AppModule, + imports: [...coreModules, ...featureModules], + controllers: [AppController], + providers: [ + AppService, + StartupLogger, + ShutdownStateService, + { + provide: APP_INTERCEPTOR, + useClass: MonitoringInterceptor, + }, + { + provide: APP_INTERCEPTOR, + useClass: TimeoutInterceptor, + }, + { + provide: APP_GUARD, + useClass: CustomThrottleGuard, + }, + ], + exports: [ShutdownStateService], + }; } - - // Payments Module - if (flags.ENABLE_PAYMENTS) { - const startTime = Date.now(); - featureModules.push(PaymentsModule); - startupLogger.recordModuleLoaded('PaymentsModule', startTime); - } else { - startupLogger.recordModuleSkipped('PaymentsModule', 'ENABLE_PAYMENTS=false'); - } - - // AB Testing Module - if (flags.ENABLE_AB_TESTING) { - const startTime = Date.now(); - featureModules.push(ABTestingModule); - startupLogger.recordModuleLoaded('ABTestingModule', startTime); - } else { - startupLogger.recordModuleSkipped('ABTestingModule', 'ENABLE_AB_TESTING=false'); - } - - // Data Warehouse Module - if (flags.ENABLE_DATA_WAREHOUSE) { - const startTime = Date.now(); - featureModules.push(DataWarehouseModule); - startupLogger.recordModuleLoaded('DataWarehouseModule', startTime); - } else { - startupLogger.recordModuleSkipped('DataWarehouseModule', 'ENABLE_DATA_WAREHOUSE=false'); - } - - // Collaboration Module - if (flags.ENABLE_COLLABORATION) { - const startTime = Date.now(); - featureModules.push(CollaborationModule); - startupLogger.recordModuleLoaded('CollaborationModule', startTime); - } else { - startupLogger.recordModuleSkipped('CollaborationModule', 'ENABLE_COLLABORATION=false'); - } - - // Media Module - if (flags.ENABLE_MEDIA_PROCESSING) { - const startTime = Date.now(); - featureModules.push(MediaModule); - startupLogger.recordModuleLoaded('MediaModule', startTime); - } else { - startupLogger.recordModuleSkipped('MediaModule', 'ENABLE_MEDIA_PROCESSING=false'); - } - - // Backup Module - if (flags.ENABLE_BACKUP) { - const startTime = Date.now(); - featureModules.push(BackupModule); - startupLogger.recordModuleLoaded('BackupModule', startTime); - } else { - startupLogger.recordModuleSkipped('BackupModule', 'ENABLE_BACKUP=false'); - } - - // GraphQL Module - if (flags.ENABLE_GRAPHQL) { - const startTime = Date.now(); - featureModules.push(GraphQLModule); - startupLogger.recordModuleLoaded('GraphQLModule', startTime); - } else { - startupLogger.recordModuleSkipped('GraphQLModule', 'ENABLE_GRAPHQL=false'); - } - - // Sync Module - if (flags.ENABLE_SYNC) { - const startTime = Date.now(); - featureModules.push(SyncModule); - startupLogger.recordModuleLoaded('SyncModule', startTime); - } else { - startupLogger.recordModuleSkipped('SyncModule', 'ENABLE_SYNC=false'); - } - - // Migration Module - if (flags.ENABLE_MIGRATIONS) { - const startTime = Date.now(); - featureModules.push(MigrationModule); - startupLogger.recordModuleLoaded('MigrationModule', startTime); - } else { - startupLogger.recordModuleSkipped('MigrationModule', 'ENABLE_MIGRATIONS=false'); - } - - // Rate Limiting Module - if (flags.ENABLE_RATE_LIMITING) { - const startTime = Date.now(); - featureModules.push(RateLimitingModule); - startupLogger.recordModuleLoaded('RateLimitingModule', startTime); - } else { - startupLogger.recordModuleSkipped('RateLimitingModule', 'ENABLE_RATE_LIMITING=false'); - } - - // Observability Module - if (flags.ENABLE_OBSERVABILITY) { - const startTime = Date.now(); - featureModules.push(ObservabilityModule); - startupLogger.recordModuleLoaded('ObservabilityModule', startTime); - } else { - startupLogger.recordModuleSkipped('ObservabilityModule', 'ENABLE_OBSERVABILITY=false'); - } - - // Caching Module - if (flags.ENABLE_CACHING) { - const startTime = Date.now(); - featureModules.push(CachingModule); - startupLogger.recordModuleLoaded('CachingModule', startTime); - } else { - startupLogger.recordModuleSkipped('CachingModule', 'ENABLE_CACHING=false'); - } - - // Feature Flags Module - if (flags.ENABLE_FEATURE_FLAGS) { - const startTime = Date.now(); - featureModules.push(FeatureFlagsModule); - startupLogger.recordModuleLoaded('FeatureFlagsModule', startTime); - } else { - startupLogger.recordModuleSkipped('FeatureFlagsModule', 'ENABLE_FEATURE_FLAGS=false'); - } - - // Search Module - if (flags.ENABLE_SEARCH) { - const startTime = Date.now(); - featureModules.push(SearchModule); - startupLogger.recordModuleLoaded('SearchModule', startTime); - } else { - startupLogger.recordModuleSkipped('SearchModule', 'ENABLE_SEARCH=false'); - } - - // Notifications Module - if (flags.ENABLE_NOTIFICATIONS) { - const startTime = Date.now(); - featureModules.push(NotificationsModule); - startupLogger.recordModuleLoaded('NotificationsModule', startTime); - } else { - startupLogger.recordModuleSkipped('NotificationsModule', 'ENABLE_NOTIFICATIONS=false'); - } - - // Email Marketing Module - if (flags.ENABLE_EMAIL_MARKETING) { - const startTime = Date.now(); - featureModules.push(EmailMarketingModule); - startupLogger.recordModuleLoaded('EmailMarketingModule', startTime); - } else { - startupLogger.recordModuleSkipped('EmailMarketingModule', 'ENABLE_EMAIL_MARKETING=false'); - } - - // Gamification Module - if (flags.ENABLE_GAMIFICATION) { - const startTime = Date.now(); - featureModules.push(GamificationModule); - startupLogger.recordModuleLoaded('GamificationModule', startTime); - } else { - startupLogger.recordModuleSkipped('GamificationModule', 'ENABLE_GAMIFICATION=false'); - } - - // Assessment Module - if (flags.ENABLE_ASSESSMENT) { - const startTime = Date.now(); - featureModules.push(AssessmentsModule); - startupLogger.recordModuleLoaded('AssessmentModule', startTime); - } else { - startupLogger.recordModuleSkipped('AssessmentModule', 'ENABLE_ASSESSMENT=false'); - } - - // Learning Paths Module - if (flags.ENABLE_LEARNING_PATHS) { - const startTime = Date.now(); - featureModules.push(LearningPathsModule); - startupLogger.recordModuleLoaded('LearningPathsModule', startTime); - } else { - startupLogger.recordModuleSkipped('LearningPathsModule', 'ENABLE_LEARNING_PATHS=false'); - } - - // Moderation Module - if (flags.ENABLE_MODERATION) { - const startTime = Date.now(); - featureModules.push(ModerationModule); - startupLogger.recordModuleLoaded('ModerationModule', startTime); - } else { - startupLogger.recordModuleSkipped('ModerationModule', 'ENABLE_MODERATION=false'); - } - - // Orchestration Module - if (flags.ENABLE_ORCHESTRATION) { - const startTime = Date.now(); - featureModules.push(OrchestrationModule); - startupLogger.recordModuleLoaded('OrchestrationModule', startTime); - } else { - startupLogger.recordModuleSkipped('OrchestrationModule', 'ENABLE_ORCHESTRATION=false'); - } - - // Security Module - if (flags.ENABLE_SECURITY) { - const startTime = Date.now(); - featureModules.push(SecurityModule); - startupLogger.recordModuleLoaded('SecurityModule', startTime); - } else { - startupLogger.recordModuleSkipped('SecurityModule', 'ENABLE_SECURITY=false'); - } - - // Tenancy Module - if (flags.ENABLE_TENANCY) { - const startTime = Date.now(); - featureModules.push(TenancyModule); - startupLogger.recordModuleLoaded('TenancyModule', startTime); - } else { - startupLogger.recordModuleSkipped('TenancyModule', 'ENABLE_TENANCY=false'); - } - - // CDN Module - if (flags.ENABLE_CDN) { - const startTime = Date.now(); - featureModules.push(CdnModule); - startupLogger.recordModuleLoaded('CDNModule', startTime); - } else { - startupLogger.recordModuleSkipped('CDNModule', 'ENABLE_CDN=false'); - } - - // Localization Module - if (flags.ENABLE_LOCALIZATION) { - const startTime = Date.now(); - featureModules.push(LocalizationModule); - startupLogger.recordModuleLoaded('LocalizationModule', startTime); - } else { - startupLogger.recordModuleSkipped('LocalizationModule', 'ENABLE_LOCALIZATION=false'); - } - - // Queue Module (always loaded for Bull) - featureModules.push(QueueModule); - - return { - module: AppModule, - imports: [...coreModules, ...featureModules], - controllers: [AppController], - providers: [ - AppService, - StartupLogger, - ShutdownStateService, - { - provide: APP_INTERCEPTOR, - useClass: MonitoringInterceptor, - }, - { - provide: APP_INTERCEPTOR, - useClass: TimeoutInterceptor, - }, - { - provide: APP_GUARD, - useClass: CustomThrottleGuard, - }, - ], - exports: [ShutdownStateService], - }; - } } diff --git a/src/app.service.spec.ts b/src/app.service.spec.ts index b9fd4695..0824f2ea 100644 --- a/src/app.service.spec.ts +++ b/src/app.service.spec.ts @@ -1,35 +1,28 @@ import { Test, TestingModule } from '@nestjs/testing'; import { AppService } from './app.service'; - describe('AppService', () => { - let service: AppService; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [AppService], - }).compile(); - - service = module.get(AppService); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('getHello', () => { - it('should return "Hello World!"', () => { - const result = service.getHello(); - expect(result).toBe('Hello World!'); + let service: AppService; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + providers: [AppService], + }).compile(); + service = module.get(AppService); }); - - it('should return a string', () => { - const result = service.getHello(); - expect(typeof result).toBe('string'); + it('should be defined', () => { + expect(service).toBeDefined(); }); - - it('should return the expected greeting message', () => { - const result = service.getHello(); - expect(result).toMatch(/^Hello/); + describe('getHello', () => { + it('should return "Hello World!"', () => { + const result = service.getHello(); + expect(result).toBe('Hello World!'); + }); + it('should return a string', () => { + const result = service.getHello(); + expect(typeof result).toBe('string'); + }); + it('should return the expected greeting message', () => { + const result = service.getHello(); + expect(result).toMatch(/^Hello/); + }); }); - }); -}); \ No newline at end of file +}); diff --git a/src/app.service.ts b/src/app.service.ts index 927d7cca..e33b07c2 100644 --- a/src/app.service.ts +++ b/src/app.service.ts @@ -1,8 +1,7 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class AppService { - getHello(): string { - return 'Hello World!'; - } + getHello(): string { + return 'Hello World!'; + } } diff --git a/src/assessment/assessment.controller.ts b/src/assessment/assessment.controller.ts index 7af50f2b..55945ec1 100644 --- a/src/assessment/assessment.controller.ts +++ b/src/assessment/assessment.controller.ts @@ -1,30 +1,40 @@ import { Body, Controller, Get, Param, Post, UseGuards, Request } from '@nestjs/common'; import { AssessmentsService } from './assessments.service'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; - @Controller('assessments') export class AssessmentsController { - constructor(private readonly service: AssessmentsService) {} - - @Post(':id/start') - @UseGuards(JwtAuthGuard) - start(@Request() req: any, @Param('id') id: string): any { - const studentId = req.user.id; - if (!studentId) { - throw new Error('User not authenticated'); + constructor(private readonly service: AssessmentsService) { } + @Post(':id/start') + @UseGuards(JwtAuthGuard) + start( + @Request() + req: unknown, + @Param('id') + id: string): unknown { + const studentId = req.user.id; + if (!studentId) { + throw new Error('User not authenticated'); + } + return this.service.startAssessment(studentId, id); + } + @Post('attempts/:id/submit') + @UseGuards(JwtAuthGuard) + submit( + @Request() + req: unknown, + @Param('id') + id: string, + @Body('answers') + answers: unknown[]): unknown { + return this.service.submitAssessment(id, answers); + } + @Get('attempts/:id') + @UseGuards(JwtAuthGuard) + results( + @Request() + req: unknown, + @Param('id') + id: string): unknown { + return this.service.getResults(id); } - return this.service.startAssessment(studentId, id); - } - - @Post('attempts/:id/submit') - @UseGuards(JwtAuthGuard) - submit(@Request() req: any, @Param('id') id: string, @Body('answers') answers: any[]): any { - return this.service.submitAssessment(id, answers); - } - - @Get('attempts/:id') - @UseGuards(JwtAuthGuard) - results(@Request() req: any, @Param('id') id: string): any { - return this.service.getResults(id); - } } diff --git a/src/assessment/assessment.module.ts b/src/assessment/assessment.module.ts index 50a1df71..045b2af1 100644 --- a/src/assessment/assessment.module.ts +++ b/src/assessment/assessment.module.ts @@ -9,16 +9,16 @@ import { FeedbackGenerationService } from './feedback/feedback-generation.servic import { QuestionBankService } from './questions/question-bank.service'; import { ScoreCalculationService } from './scoring/score-calculation.service'; import { Module } from '@nestjs/common'; - @Module({ - imports: [TypeOrmModule.forFeature([Assessment, Question, AssessmentAttempt, Answer])], - controllers: [AssessmentsController], - providers: [ - AssessmentsService, - QuestionBankService, - ScoreCalculationService, - FeedbackGenerationService, - ], - exports: [AssessmentsService], + imports: [TypeOrmModule.forFeature([Assessment, Question, AssessmentAttempt, Answer])], + controllers: [AssessmentsController], + providers: [ + AssessmentsService, + QuestionBankService, + ScoreCalculationService, + FeedbackGenerationService, + ], + exports: [AssessmentsService], }) -export class AssessmentsModule {} +export class AssessmentsModule { +} diff --git a/src/assessment/assessments.service.spec.ts b/src/assessment/assessments.service.spec.ts index dbcef856..e73e5dd7 100644 --- a/src/assessment/assessments.service.spec.ts +++ b/src/assessment/assessments.service.spec.ts @@ -8,397 +8,331 @@ import { Question } from './entities/question.entity'; import { AssessmentStatus } from './enums/assessment-status.enum'; import { ScoreCalculationService } from './scoring/score-calculation.service'; import { FeedbackGenerationService } from './feedback/feedback-generation.service'; -import { - createMockRepository, - createMockQueryBuilder, -} from 'test/utils/mock-factories'; +import { createMockRepository, createMockQueryBuilder, } from 'test/utils/mock-factories'; import { Repository } from 'typeorm'; - describe('AssessmentsService', () => { - // ───────────────────────────────────────────────────────────────────────── - // DECLARATIONS - // ───────────────────────────────────────────────────────────────────────── - - let service: AssessmentsService; - let mockAssessmentRepo: jest.Mocked>; - let mockAttemptRepo: jest.Mocked>; - let mockAnswerRepo: jest.Mocked>; - let mockScoringService: jest.Mocked; - let mockFeedbackService: jest.Mocked; - - // ───────────────────────────────────────────────────────────────────────── - // SETUP & TEARDOWN - // ───────────────────────────────────────────────────────────────────────── - - beforeEach(async () => { - // Initialize all dependency mocks - mockAssessmentRepo = createMockRepository(); - mockAttemptRepo = createMockRepository(); - mockAnswerRepo = createMockRepository(); - - mockScoringService = { - calculate: jest.fn(), - } as jest.Mocked; - - mockFeedbackService = { - generate: jest.fn(), - } as jest.Mocked; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - AssessmentsService, - { - provide: getRepositoryToken(Assessment), - useValue: mockAssessmentRepo, - }, - { - provide: getRepositoryToken(AssessmentAttempt), - useValue: mockAttemptRepo, - }, - { - provide: getRepositoryToken(Answer), - useValue: mockAnswerRepo, - }, - { - provide: ScoreCalculationService, - useValue: mockScoringService, - }, - { - provide: FeedbackGenerationService, - useValue: mockFeedbackService, - }, - ], - }).compile(); - - service = module.get(AssessmentsService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - // ───────────────────────────────────────────────────────────────────────── - // TEST SUITES - // ───────────────────────────────────────────────────────────────────────── - - describe('startAssessment', () => { - const studentId = 'student-1'; - const assessmentId = 'assessment-1'; - const mockAssessment = { - id: assessmentId, - title: 'Test Assessment', - questions: [ - { id: 'q1', question: 'Question 1' }, - { id: 'q2', question: 'Question 2' }, - ], - }; - - beforeEach(() => { - mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); - mockAttemptRepo.save.mockImplementation(async (attempt) => attempt); + // ───────────────────────────────────────────────────────────────────────── + // DECLARATIONS + // ───────────────────────────────────────────────────────────────────────── + let service: AssessmentsService; + let mockAssessmentRepo: jest.Mocked>; + let mockAttemptRepo: jest.Mocked>; + let mockAnswerRepo: jest.Mocked>; + let mockScoringService: jest.Mocked; + let mockFeedbackService: jest.Mocked; + // ───────────────────────────────────────────────────────────────────────── + // SETUP & TEARDOWN + // ───────────────────────────────────────────────────────────────────────── + beforeEach(async () => { + // Initialize all dependency mocks + mockAssessmentRepo = createMockRepository(); + mockAttemptRepo = createMockRepository(); + mockAnswerRepo = createMockRepository(); + mockScoringService = { + calculate: jest.fn(), + } as jest.Mocked; + mockFeedbackService = { + generate: jest.fn(), + } as jest.Mocked; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + AssessmentsService, + { + provide: getRepositoryToken(Assessment), + useValue: mockAssessmentRepo, + }, + { + provide: getRepositoryToken(AssessmentAttempt), + useValue: mockAttemptRepo, + }, + { + provide: getRepositoryToken(Answer), + useValue: mockAnswerRepo, + }, + { + provide: ScoreCalculationService, + useValue: mockScoringService, + }, + { + provide: FeedbackGenerationService, + useValue: mockFeedbackService, + }, + ], + }).compile(); + service = module.get(AssessmentsService); }); - - it('should start an assessment and create an attempt', async () => { - const result = await service.startAssessment(studentId, assessmentId); - - expect(result).toEqual({ - studentId, - assessment: mockAssessment, - status: AssessmentStatus.IN_PROGRESS, - startedAt: expect.any(Date), - }); - - expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ - where: { id: assessmentId }, - relations: ['questions'], - }); - expect(mockAttemptRepo.save).toHaveBeenCalled(); + afterEach(() => { + jest.clearAllMocks(); }); - - it('should load assessment with questions relation', async () => { - await service.startAssessment(studentId, assessmentId); - - expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ - where: { id: assessmentId }, - relations: ['questions'], - }); + // ───────────────────────────────────────────────────────────────────────── + // TEST SUITES + // ───────────────────────────────────────────────────────────────────────── + describe('startAssessment', () => { + const studentId = 'student-1'; + const assessmentId = 'assessment-1'; + const mockAssessment = { + id: assessmentId, + title: 'Test Assessment', + questions: [ + { id: 'q1', question: 'Question 1' }, + { id: 'q2', question: 'Question 2' }, + ], + }; + beforeEach(() => { + mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); + mockAttemptRepo.save.mockImplementation(async (attempt) => attempt); + }); + it('should start an assessment and create an attempt', async () => { + const result = await service.startAssessment(studentId, assessmentId); + expect(result).toEqual({ + studentId, + assessment: mockAssessment, + status: AssessmentStatus.IN_PROGRESS, + startedAt: expect.any(Date), + }); + expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ + where: { id: assessmentId }, + relations: ['questions'], + }); + expect(mockAttemptRepo.save).toHaveBeenCalled(); + }); + it('should load assessment with questions relation', async () => { + await service.startAssessment(studentId, assessmentId); + expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ + where: { id: assessmentId }, + relations: ['questions'], + }); + }); }); - }); - - describe('findAll', () => { - const mockAssessments = [ - { id: '1', title: 'Assessment 1' }, - { id: '2', title: 'Assessment 2' }, - ]; - - beforeEach(() => { - mockAssessmentRepo.find.mockResolvedValue(mockAssessments); + describe('findAll', () => { + const mockAssessments = [ + { id: '1', title: 'Assessment 1' }, + { id: '2', title: 'Assessment 2' }, + ]; + beforeEach(() => { + mockAssessmentRepo.find.mockResolvedValue(mockAssessments); + }); + it('should return all assessments with questions relation', async () => { + const result = await service.findAll(); + expect(result).toEqual(mockAssessments); + expect(mockAssessmentRepo.find).toHaveBeenCalledWith({ + relations: ['questions'], + }); + }); }); - - it('should return all assessments with questions relation', async () => { - const result = await service.findAll(); - - expect(result).toEqual(mockAssessments); - expect(mockAssessmentRepo.find).toHaveBeenCalledWith({ - relations: ['questions'], - }); + describe('findOne', () => { + const assessmentId = 'assessment-1'; + const mockAssessment = { + id: assessmentId, + title: 'Test Assessment', + questions: [], + }; + beforeEach(() => { + mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); + }); + it('should return assessment by id with questions relation', async () => { + const result = await service.findOne(assessmentId); + expect(result).toEqual(mockAssessment); + expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ + where: { id: assessmentId }, + relations: ['questions'], + }); + }); }); - }); - - describe('findOne', () => { - const assessmentId = 'assessment-1'; - const mockAssessment = { - id: assessmentId, - title: 'Test Assessment', - questions: [], - }; - - beforeEach(() => { - mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); + describe('findByIds', () => { + const ids = ['1', '2', '3']; + const mockAssessments = [ + { id: '1', title: 'Assessment 1' }, + { id: '2', title: 'Assessment 2' }, + { id: '3', title: 'Assessment 3' }, + ]; + beforeEach(() => { + mockAssessmentRepo.findByIds.mockResolvedValue(mockAssessments); + }); + it('should return assessments by ids', async () => { + const result = await service.findByIds(ids); + expect(result).toEqual(mockAssessments); + expect(mockAssessmentRepo.findByIds).toHaveBeenCalledWith(ids); + }); + it('should return empty array for empty ids', async () => { + const result = await service.findByIds([]); + expect(result).toEqual([]); + expect(mockAssessmentRepo.findByIds).not.toHaveBeenCalled(); + }); }); - - it('should return assessment by id with questions relation', async () => { - const result = await service.findOne(assessmentId); - - expect(result).toEqual(mockAssessment); - expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ - where: { id: assessmentId }, - relations: ['questions'], - }); + describe('create', () => { + const createData = { + title: 'New Assessment', + description: 'Test assessment', + durationMinutes: 60, + }; + const mockAssessment = { + id: 'new-assessment-1', + ...createData, + }; + beforeEach(() => { + mockAssessmentRepo.create.mockReturnValue(mockAssessment as Assessment); + mockAssessmentRepo.save.mockResolvedValue(mockAssessment as Assessment); + }); + it('should create and save a new assessment', async () => { + const result = await service.create(createData); + expect(result).toEqual(mockAssessment); + expect(mockAssessmentRepo.create).toHaveBeenCalledWith(createData); + expect(mockAssessmentRepo.save).toHaveBeenCalledWith(mockAssessment); + }); + it('should handle array return from save', async () => { + const savedArray = [mockAssessment]; + mockAssessmentRepo.save.mockResolvedValue(savedArray as unknown); + const result = await service.create(createData); + expect(result).toEqual(mockAssessment); + }); }); - }); - - describe('findByIds', () => { - const ids = ['1', '2', '3']; - const mockAssessments = [ - { id: '1', title: 'Assessment 1' }, - { id: '2', title: 'Assessment 2' }, - { id: '3', title: 'Assessment 3' }, - ]; - - beforeEach(() => { - mockAssessmentRepo.findByIds.mockResolvedValue(mockAssessments); + describe('update', () => { + const assessmentId = 'assessment-1'; + const updateData = { title: 'Updated Title' }; + const mockAssessment = { + id: assessmentId, + title: 'Updated Title', + description: 'Original description', + }; + beforeEach(() => { + mockAssessmentRepo.update.mockResolvedValue({ affected: 1, raw: {}, generatedMaps: [] }); + mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); + }); + it('should update assessment and return updated entity', async () => { + const result = await service.update(assessmentId, updateData); + expect(result).toEqual(mockAssessment); + expect(mockAssessmentRepo.update).toHaveBeenCalledWith(assessmentId, updateData); + expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ + where: { id: assessmentId }, + relations: ['questions'], + }); + }); }); - - it('should return assessments by ids', async () => { - const result = await service.findByIds(ids); - - expect(result).toEqual(mockAssessments); - expect(mockAssessmentRepo.findByIds).toHaveBeenCalledWith(ids); + describe('remove', () => { + const assessmentId = 'assessment-1'; + const mockAssessment = { + id: assessmentId, + title: 'Test Assessment', + }; + beforeEach(() => { + mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); + mockAssessmentRepo.manager = { + transaction: jest.fn().mockImplementation(async (fn) => fn({ + getRepository: jest.fn().mockReturnValue({ + createQueryBuilder: jest.fn().mockReturnValue({ + softDelete: jest.fn().mockReturnThis(), + where: jest.fn().mockReturnThis(), + execute: jest.fn().mockResolvedValue(undefined), + }), + }), + })), + } as unknown; + }); + it('should soft delete assessment and its questions', async () => { + await service.remove(assessmentId); + expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ + where: { id: assessmentId }, + relations: ['questions'], + }); + expect(mockAssessmentRepo.manager.transaction).toHaveBeenCalled(); + }); + it('should do nothing if assessment not found', async () => { + mockAssessmentRepo.findOne.mockResolvedValue(null); + await service.remove(assessmentId); + expect(mockAssessmentRepo.manager.transaction).not.toHaveBeenCalled(); + }); }); - - it('should return empty array for empty ids', async () => { - const result = await service.findByIds([]); - - expect(result).toEqual([]); - expect(mockAssessmentRepo.findByIds).not.toHaveBeenCalled(); + describe('submitAssessment', () => { + const attemptId = 'attempt-1'; + const answers = [ + { questionId: 'q1', response: 'Answer 1' }, + { questionId: 'q2', response: 'Answer 2' }, + ]; + const mockAttempt = { + id: attemptId, + startedAt: new Date(Date.now() - 30 * 60 * 1000), // 30 minutes ago + assessment: { + id: 'assessment-1', + durationMinutes: 60, + questions: [ + { id: 'q1', points: 10 }, + { id: 'q2', points: 15 }, + ], + }, + }; + beforeEach(() => { + mockAttemptRepo.findOne.mockResolvedValue(mockAttempt as unknown); + mockScoringService.calculate + .mockReturnValueOnce(8) // Question 1: 8/10 points + .mockReturnValueOnce(12); // Question 2: 12/15 points + mockAnswerRepo.save.mockResolvedValue({} as unknown); + mockAttemptRepo.save.mockImplementation(async (attempt) => attempt); + mockFeedbackService.generate.mockReturnValue({ + overall: 'Good performance', + strengths: ['Good understanding'], + improvements: ['Need more practice'], + }); + }); + it('should submit assessment and calculate score', async () => { + const result = await service.submitAssessment(attemptId, answers); + expect(result).toEqual({ + attempt: { + ...mockAttempt, + score: 20, // 8 + 12 + status: AssessmentStatus.GRADED, + submittedAt: expect.any(Date), + }, + feedback: { + overall: 'Good performance', + strengths: ['Good understanding'], + improvements: ['Need more practice'], + }, + }); + expect(mockAttemptRepo.findOne).toHaveBeenCalledWith({ + where: { id: attemptId }, + relations: ['assessment', 'assessment.questions'], + }); + expect(mockScoringService.calculate).toHaveBeenCalledTimes(2); + expect(mockAnswerRepo.save).toHaveBeenCalledTimes(2); + expect(mockFeedbackService.generate).toHaveBeenCalledWith(20, 25); // totalScore, maxScore + }); + it('should mark as timed out if submitted after duration', async () => { + // Set startedAt to 2 hours ago, duration is 60 minutes + const oldAttempt = { + ...mockAttempt, + startedAt: new Date(Date.now() - 2 * 60 * 60 * 1000), // 2 hours ago + }; + mockAttemptRepo.findOne.mockResolvedValue(oldAttempt as unknown); + const result = await service.submitAssessment(attemptId, answers); + expect(result.status).toBe(AssessmentStatus.TIMED_OUT); + expect(mockScoringService.calculate).not.toHaveBeenCalled(); + }); }); - }); - - describe('create', () => { - const createData = { - title: 'New Assessment', - description: 'Test assessment', - durationMinutes: 60, - }; - - const mockAssessment = { - id: 'new-assessment-1', - ...createData, - }; - - beforeEach(() => { - mockAssessmentRepo.create.mockReturnValue(mockAssessment as Assessment); - mockAssessmentRepo.save.mockResolvedValue(mockAssessment as Assessment); + describe('getResults', () => { + const attemptId = 'attempt-1'; + const mockResults = { + id: attemptId, + score: 85, + answers: [ + { + id: 'answer-1', + question: { id: 'q1', question: 'Question 1' }, + response: 'Answer 1', + awardedPoints: 8, + }, + ], + }; + beforeEach(() => { + mockAttemptRepo.findOne.mockResolvedValue(mockResults as unknown); + }); + it('should return assessment results with answers and questions', async () => { + const result = await service.getResults(attemptId); + expect(result).toEqual(mockResults); + expect(mockAttemptRepo.findOne).toHaveBeenCalledWith({ + where: { id: attemptId }, + relations: ['answers', 'answers.question'], + }); + }); }); - - it('should create and save a new assessment', async () => { - const result = await service.create(createData); - - expect(result).toEqual(mockAssessment); - expect(mockAssessmentRepo.create).toHaveBeenCalledWith(createData); - expect(mockAssessmentRepo.save).toHaveBeenCalledWith(mockAssessment); - }); - - it('should handle array return from save', async () => { - const savedArray = [mockAssessment]; - mockAssessmentRepo.save.mockResolvedValue(savedArray as any); - - const result = await service.create(createData); - - expect(result).toEqual(mockAssessment); - }); - }); - - describe('update', () => { - const assessmentId = 'assessment-1'; - const updateData = { title: 'Updated Title' }; - const mockAssessment = { - id: assessmentId, - title: 'Updated Title', - description: 'Original description', - }; - - beforeEach(() => { - mockAssessmentRepo.update.mockResolvedValue({ affected: 1, raw: {}, generatedMaps: [] }); - mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); - }); - - it('should update assessment and return updated entity', async () => { - const result = await service.update(assessmentId, updateData); - - expect(result).toEqual(mockAssessment); - expect(mockAssessmentRepo.update).toHaveBeenCalledWith(assessmentId, updateData); - expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ - where: { id: assessmentId }, - relations: ['questions'], - }); - }); - }); - - describe('remove', () => { - const assessmentId = 'assessment-1'; - const mockAssessment = { - id: assessmentId, - title: 'Test Assessment', - }; - - beforeEach(() => { - mockAssessmentRepo.findOne.mockResolvedValue(mockAssessment); - mockAssessmentRepo.manager = { - transaction: jest.fn().mockImplementation(async (fn) => fn({ - getRepository: jest.fn().mockReturnValue({ - createQueryBuilder: jest.fn().mockReturnValue({ - softDelete: jest.fn().mockReturnThis(), - where: jest.fn().mockReturnThis(), - execute: jest.fn().mockResolvedValue(undefined), - }), - }), - })), - } as any; - }); - - it('should soft delete assessment and its questions', async () => { - await service.remove(assessmentId); - - expect(mockAssessmentRepo.findOne).toHaveBeenCalledWith({ - where: { id: assessmentId }, - relations: ['questions'], - }); - expect(mockAssessmentRepo.manager.transaction).toHaveBeenCalled(); - }); - - it('should do nothing if assessment not found', async () => { - mockAssessmentRepo.findOne.mockResolvedValue(null); - - await service.remove(assessmentId); - - expect(mockAssessmentRepo.manager.transaction).not.toHaveBeenCalled(); - }); - }); - - describe('submitAssessment', () => { - const attemptId = 'attempt-1'; - const answers = [ - { questionId: 'q1', response: 'Answer 1' }, - { questionId: 'q2', response: 'Answer 2' }, - ]; - - const mockAttempt = { - id: attemptId, - startedAt: new Date(Date.now() - 30 * 60 * 1000), // 30 minutes ago - assessment: { - id: 'assessment-1', - durationMinutes: 60, - questions: [ - { id: 'q1', points: 10 }, - { id: 'q2', points: 15 }, - ], - }, - }; - - beforeEach(() => { - mockAttemptRepo.findOne.mockResolvedValue(mockAttempt as any); - mockScoringService.calculate - .mockReturnValueOnce(8) // Question 1: 8/10 points - .mockReturnValueOnce(12); // Question 2: 12/15 points - mockAnswerRepo.save.mockResolvedValue({} as any); - mockAttemptRepo.save.mockImplementation(async (attempt) => attempt); - mockFeedbackService.generate.mockReturnValue({ - overall: 'Good performance', - strengths: ['Good understanding'], - improvements: ['Need more practice'], - }); - }); - - it('should submit assessment and calculate score', async () => { - const result = await service.submitAssessment(attemptId, answers); - - expect(result).toEqual({ - attempt: { - ...mockAttempt, - score: 20, // 8 + 12 - status: AssessmentStatus.GRADED, - submittedAt: expect.any(Date), - }, - feedback: { - overall: 'Good performance', - strengths: ['Good understanding'], - improvements: ['Need more practice'], - }, - }); - - expect(mockAttemptRepo.findOne).toHaveBeenCalledWith({ - where: { id: attemptId }, - relations: ['assessment', 'assessment.questions'], - }); - expect(mockScoringService.calculate).toHaveBeenCalledTimes(2); - expect(mockAnswerRepo.save).toHaveBeenCalledTimes(2); - expect(mockFeedbackService.generate).toHaveBeenCalledWith(20, 25); // totalScore, maxScore - }); - - it('should mark as timed out if submitted after duration', async () => { - // Set startedAt to 2 hours ago, duration is 60 minutes - const oldAttempt = { - ...mockAttempt, - startedAt: new Date(Date.now() - 2 * 60 * 60 * 1000), // 2 hours ago - }; - mockAttemptRepo.findOne.mockResolvedValue(oldAttempt as any); - - const result = await service.submitAssessment(attemptId, answers); - - expect(result.status).toBe(AssessmentStatus.TIMED_OUT); - expect(mockScoringService.calculate).not.toHaveBeenCalled(); - }); - }); - - describe('getResults', () => { - const attemptId = 'attempt-1'; - const mockResults = { - id: attemptId, - score: 85, - answers: [ - { - id: 'answer-1', - question: { id: 'q1', question: 'Question 1' }, - response: 'Answer 1', - awardedPoints: 8, - }, - ], - }; - - beforeEach(() => { - mockAttemptRepo.findOne.mockResolvedValue(mockResults as any); - }); - - it('should return assessment results with answers and questions', async () => { - const result = await service.getResults(attemptId); - - expect(result).toEqual(mockResults); - expect(mockAttemptRepo.findOne).toHaveBeenCalledWith({ - where: { id: attemptId }, - relations: ['answers', 'answers.question'], - }); - }); - }); -}); \ No newline at end of file +}); diff --git a/src/assessment/assessments.service.ts b/src/assessment/assessments.service.ts index 1f2f9487..7e7fec27 100644 --- a/src/assessment/assessments.service.ts +++ b/src/assessment/assessments.service.ts @@ -8,128 +8,104 @@ import { FeedbackGenerationService } from './feedback/feedback-generation.servic import { Answer } from './entities/answer.entity'; import { ScoreCalculationService } from './scoring/score-calculation.service'; import { Question } from './entities/question.entity'; - @Injectable() export class AssessmentsService { - constructor( + constructor( @InjectRepository(Assessment) - private readonly assessmentRepo: Repository, + private readonly assessmentRepo: Repository, @InjectRepository(AssessmentAttempt) - private readonly attemptRepo: Repository, + private readonly attemptRepo: Repository, @InjectRepository(Answer) - private readonly answerRepo: Repository, - private readonly scoringService: ScoreCalculationService, - private readonly feedbackService: FeedbackGenerationService, - ) {} - - async startAssessment(studentId: string, assessmentId: string) { - const assessment = await this.assessmentRepo.findOne({ - where: { id: assessmentId }, - relations: ['questions'], - }); - - return this.attemptRepo.save({ - studentId, - assessment, - status: AssessmentStatus.IN_PROGRESS, - startedAt: new Date(), - }); - } - - async findAll(): Promise { - return await this.assessmentRepo.find({ - relations: ['questions'], - }); - } - - async findOne(id: string): Promise { - return await this.assessmentRepo.findOne({ - where: { id }, - relations: ['questions'], - }); - } - - async findByIds(ids: string[]): Promise { - if (ids.length === 0) return []; - return await this.assessmentRepo.findByIds(ids); - } - - async create(data: any): Promise { - const assessment = this.assessmentRepo.create(data); - const saved = await this.assessmentRepo.save(assessment); - return Array.isArray(saved) ? saved[0] : saved; - } - - async update(id: string, data: any): Promise { - await this.assessmentRepo.update(id, data); - return this.findOne(id); - } - - async remove(id: string): Promise { - const assessment = await this.findOne(id); - if (!assessment) { - return; + private readonly answerRepo: Repository, private readonly scoringService: ScoreCalculationService, private readonly feedbackService: FeedbackGenerationService) { } + async startAssessment(studentId: string, assessmentId: string) { + const assessment = await this.assessmentRepo.findOne({ + where: { id: assessmentId }, + relations: ['questions'], + }); + return this.attemptRepo.save({ + studentId, + assessment, + status: AssessmentStatus.IN_PROGRESS, + startedAt: new Date(), + }); } - - await this.assessmentRepo.manager.transaction(async (manager) => { - await manager - .getRepository(Question) - .createQueryBuilder() - .softDelete() - .where('"assessmentId" = :assessmentId', { assessmentId: id }) - .execute(); - await manager.getRepository(Assessment).softDelete(id); - }); - } - - async submitAssessment(attemptId: string, answers: any[]) { - const attempt = await this.attemptRepo.findOne({ - where: { id: attemptId }, - relations: ['assessment', 'assessment.questions'], - }); - - const endTime = - new Date(attempt.startedAt).getTime() + attempt.assessment.durationMinutes * 60000; - - if (Date.now() > endTime) { - attempt.status = AssessmentStatus.TIMED_OUT; - return this.attemptRepo.save(attempt); + async findAll(): Promise { + return await this.assessmentRepo.find({ + relations: ['questions'], + }); } - - let totalScore = 0; - let maxScore = 0; - - for (const question of attempt.assessment.questions) { - const response = answers.find((a) => a.questionId === question.id)?.response; - - const score = this.scoringService.calculate(question, response); - maxScore += question.points; - totalScore += score; - - await this.answerRepo.save({ - attempt, - question, - response, - awardedPoints: score, - }); + async findOne(id: string): Promise { + return await this.assessmentRepo.findOne({ + where: { id }, + relations: ['questions'], + }); + } + async findByIds(ids: string[]): Promise { + if (ids.length === 0) + return []; + return await this.assessmentRepo.findByIds(ids); + } + async create(data: unknown): Promise { + const assessment = this.assessmentRepo.create(data); + const saved = await this.assessmentRepo.save(assessment); + return Array.isArray(saved) ? saved[0] : saved; + } + async update(id: string, data: unknown): Promise { + await this.assessmentRepo.update(id, data); + return this.findOne(id); + } + async remove(id: string): Promise { + const assessment = await this.findOne(id); + if (!assessment) { + return; + } + await this.assessmentRepo.manager.transaction(async (manager) => { + await manager + .getRepository(Question) + .createQueryBuilder() + .softDelete() + .where('"assessmentId" = :assessmentId', { assessmentId: id }) + .execute(); + await manager.getRepository(Assessment).softDelete(id); + }); + } + async submitAssessment(attemptId: string, answers: unknown[]) { + const attempt = await this.attemptRepo.findOne({ + where: { id: attemptId }, + relations: ['assessment', 'assessment.questions'], + }); + const endTime = new Date(attempt.startedAt).getTime() + attempt.assessment.durationMinutes * 60000; + if (Date.now() > endTime) { + attempt.status = AssessmentStatus.TIMED_OUT; + return this.attemptRepo.save(attempt); + } + let totalScore = 0; + let maxScore = 0; + for (const question of attempt.assessment.questions) { + const response = answers.find((a) => a.questionId === question.id)?.response; + const score = this.scoringService.calculate(question, response); + maxScore += question.points; + totalScore += score; + await this.answerRepo.save({ + attempt, + question, + response, + awardedPoints: score, + }); + } + attempt.score = totalScore; + attempt.status = AssessmentStatus.GRADED; + attempt.submittedAt = new Date(); + const feedback = this.feedbackService.generate(totalScore, maxScore); + return { + attempt: await this.attemptRepo.save(attempt), + feedback, + }; + } + getResults(attemptId: string) { + return this.attemptRepo.findOne({ + where: { id: attemptId }, + relations: ['answers', 'answers.question'], + }); } - - attempt.score = totalScore; - attempt.status = AssessmentStatus.GRADED; - attempt.submittedAt = new Date(); - - const feedback = this.feedbackService.generate(totalScore, maxScore); - - return { - attempt: await this.attemptRepo.save(attempt), - feedback, - }; - } - - getResults(attemptId: string) { - return this.attemptRepo.findOne({ - where: { id: attemptId }, - relations: ['answers', 'answers.question'], - }); - } } diff --git a/src/assessment/dto/create-assessment.dto.ts b/src/assessment/dto/create-assessment.dto.ts index 8188cab5..d7b37fc5 100644 --- a/src/assessment/dto/create-assessment.dto.ts +++ b/src/assessment/dto/create-assessment.dto.ts @@ -1,112 +1,91 @@ -import { - IsString, - IsNotEmpty, - IsOptional, - IsArray, - IsUUID, - IsEnum, - IsNumber, - Min, - Max, -} from 'class-validator'; +import { IsString, IsNotEmpty, IsOptional, IsArray, IsUUID, IsEnum, IsNumber, Min, Max, } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; - export enum AssessmentType { - QUIZ = 'quiz', - EXAM = 'exam', - ASSIGNMENT = 'assignment', - PROJECT = 'project', + QUIZ = 'quiz', + EXAM = 'exam', + ASSIGNMENT = 'assignment', + PROJECT = 'project' } - export enum AssessmentStatus { - DRAFT = 'draft', - PUBLISHED = 'published', - ARCHIVED = 'archived', + DRAFT = 'draft', + PUBLISHED = 'published', + ARCHIVED = 'archived' } - export class CreateAssessmentDto { - @ApiProperty({ - description: 'Assessment title', - example: 'JavaScript Fundamentals Quiz', - }) - @IsString({ message: 'Title must be a string' }) - @IsNotEmpty({ message: 'Title is required' }) - title: string; - - @ApiProperty({ - description: 'Assessment description', - example: 'Test your knowledge of JavaScript basics', - }) - @IsString({ message: 'Description must be a string' }) - @IsNotEmpty({ message: 'Description is required' }) - description: string; - - @ApiPropertyOptional({ - description: 'Type of assessment', - enum: AssessmentType, - default: AssessmentType.QUIZ, - }) - @IsOptional() - @IsEnum(AssessmentType, { message: 'Type must be a valid assessment type' }) - type?: AssessmentType; - - @ApiPropertyOptional({ - description: 'Course ID this assessment belongs to', - example: '123e4567-e89b-12d3-a456-426614174000', - }) - @IsOptional() - @IsUUID('4', { message: 'Course ID must be a valid UUID' }) - courseId?: string; - - @ApiPropertyOptional({ - description: 'Maximum score for this assessment', - example: 100, - minimum: 1, - maximum: 1000, - }) - @IsOptional() - @IsNumber({}, { message: 'Max score must be a number' }) - @Min(1, { message: 'Max score must be at least 1' }) - @Max(1000, { message: 'Max score cannot exceed 1000' }) - maxScore?: number; - - @ApiPropertyOptional({ - description: 'Time limit in minutes', - example: 60, - minimum: 1, - maximum: 1440, - }) - @IsOptional() - @IsNumber({}, { message: 'Time limit must be a number' }) - @Min(1, { message: 'Time limit must be at least 1 minute' }) - @Max(1440, { message: 'Time limit cannot exceed 24 hours' }) - timeLimitMinutes?: number; - - @ApiPropertyOptional({ - description: 'Whether this assessment is published', - default: false, - }) - @IsOptional() - @IsEnum(AssessmentStatus, { message: 'Status must be a valid assessment status' }) - status?: AssessmentStatus; - - @ApiPropertyOptional({ - description: 'Array of question IDs', - type: [String], - }) - @IsOptional() - @IsArray({ message: 'Questions must be an array' }) - @IsUUID('4', { each: true, message: 'Each question ID must be a valid UUID' }) - questionIds?: string[]; - - @ApiPropertyOptional({ - description: 'Assessment settings', - example: { - allowRetakes: true, - showCorrectAnswers: false, - randomizeQuestions: true, - }, - }) - @IsOptional() - settings?: Record; + @ApiProperty({ + description: 'Assessment title', + example: 'JavaScript Fundamentals Quiz', + }) + @IsString({ message: 'Title must be a string' }) + @IsNotEmpty({ message: 'Title is required' }) + title: string; + @ApiProperty({ + description: 'Assessment description', + example: 'Test your knowledge of JavaScript basics', + }) + @IsString({ message: 'Description must be a string' }) + @IsNotEmpty({ message: 'Description is required' }) + description: string; + @ApiPropertyOptional({ + description: 'Type of assessment', + enum: AssessmentType, + default: AssessmentType.QUIZ, + }) + @IsOptional() + @IsEnum(AssessmentType, { message: 'Type must be a valid assessment type' }) + type?: AssessmentType; + @ApiPropertyOptional({ + description: 'Course ID this assessment belongs to', + example: '123e4567-e89b-12d3-a456-426614174000', + }) + @IsOptional() + @IsUUID('4', { message: 'Course ID must be a valid UUID' }) + courseId?: string; + @ApiPropertyOptional({ + description: 'Maximum score for this assessment', + example: 100, + minimum: 1, + maximum: 1000, + }) + @IsOptional() + @IsNumber({}, { message: 'Max score must be a number' }) + @Min(1, { message: 'Max score must be at least 1' }) + @Max(1000, { message: 'Max score cannot exceed 1000' }) + maxScore?: number; + @ApiPropertyOptional({ + description: 'Time limit in minutes', + example: 60, + minimum: 1, + maximum: 1440, + }) + @IsOptional() + @IsNumber({}, { message: 'Time limit must be a number' }) + @Min(1, { message: 'Time limit must be at least 1 minute' }) + @Max(1440, { message: 'Time limit cannot exceed 24 hours' }) + timeLimitMinutes?: number; + @ApiPropertyOptional({ + description: 'Whether this assessment is published', + default: false, + }) + @IsOptional() + @IsEnum(AssessmentStatus, { message: 'Status must be a valid assessment status' }) + status?: AssessmentStatus; + @ApiPropertyOptional({ + description: 'Array of question IDs', + type: [String], + }) + @IsOptional() + @IsArray({ message: 'Questions must be an array' }) + @IsUUID('4', { each: true, message: 'Each question ID must be a valid UUID' }) + questionIds?: string[]; + @ApiPropertyOptional({ + description: 'Assessment settings', + example: { + allowRetakes: true, + showCorrectAnswers: false, + randomizeQuestions: true, + }, + }) + @IsOptional() + settings?: Record; } diff --git a/src/assessment/dto/update-assessment.dto.ts b/src/assessment/dto/update-assessment.dto.ts index 8034569e..ed357369 100644 --- a/src/assessment/dto/update-assessment.dto.ts +++ b/src/assessment/dto/update-assessment.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateAssessmentDto } from './create-assessment.dto'; - -export class UpdateAssessmentDto extends PartialType(CreateAssessmentDto) {} +export class UpdateAssessmentDto extends PartialType(CreateAssessmentDto) { +} diff --git a/src/assessment/entities/answer.entity.ts b/src/assessment/entities/answer.entity.ts index e6ebc7b0..816ce181 100644 --- a/src/assessment/entities/answer.entity.ts +++ b/src/assessment/entities/answer.entity.ts @@ -3,18 +3,14 @@ import { AssessmentAttempt } from './assessment-attempt.entity'; import { Question } from './question.entity'; @Entity() export class Answer { - @PrimaryGeneratedColumn('uuid') - id: string; - - @ManyToOne(() => AssessmentAttempt, (a) => a.answers) - attempt: AssessmentAttempt; - - @ManyToOne(() => Question) - question: Question; - - @Column({ type: 'json' }) - response: string | any; - - @Column({ nullable: true }) - awardedPoints?: number; + @PrimaryGeneratedColumn('uuid') + id: string; + @ManyToOne(() => AssessmentAttempt, (a) => a.answers) + attempt: AssessmentAttempt; + @ManyToOne(() => Question) + question: Question; + @Column({ type: 'json' }) + response: string | unknown; + @Column({ nullable: true }) + awardedPoints?: number; } diff --git a/src/assessment/entities/assessment-attempt.entity.ts b/src/assessment/entities/assessment-attempt.entity.ts index aa37f7dc..0f9900cc 100644 --- a/src/assessment/entities/assessment-attempt.entity.ts +++ b/src/assessment/entities/assessment-attempt.entity.ts @@ -2,32 +2,24 @@ import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany } from 'ty import { AssessmentStatus } from '../enums/assessment-status.enum'; import { Answer } from './answer.entity'; import { Assessment } from './assessment.entity'; - @Entity() export class AssessmentAttempt { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - studentId: string; - - @ManyToOne(() => Assessment) - assessment: Assessment; - - @Column({ type: 'enum', enum: AssessmentStatus }) - status: AssessmentStatus; - - @Column({ nullable: true }) - score?: number; - - @Column({ type: 'timestamp' }) - startedAt: Date; - - @Column({ type: 'timestamp', nullable: true }) - submittedAt?: Date; - - @OneToMany(() => Answer, (a) => a.attempt, { - cascade: true, - }) - answers: Answer[]; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + studentId: string; + @ManyToOne(() => Assessment) + assessment: Assessment; + @Column({ type: 'enum', enum: AssessmentStatus }) + status: AssessmentStatus; + @Column({ nullable: true }) + score?: number; + @Column({ type: 'timestamp' }) + startedAt: Date; + @Column({ type: 'timestamp', nullable: true }) + submittedAt?: Date; + @OneToMany(() => Answer, (a) => a.attempt, { + cascade: true, + }) + answers: Answer[]; } diff --git a/src/assessment/entities/assessment.entity.ts b/src/assessment/entities/assessment.entity.ts index 4f0ea2a9..d442b644 100644 --- a/src/assessment/entities/assessment.entity.ts +++ b/src/assessment/entities/assessment.entity.ts @@ -1,38 +1,23 @@ -import { - Column, - CreateDateColumn, - DeleteDateColumn, - Entity, - Index, - OneToMany, - PrimaryGeneratedColumn, -} from 'typeorm'; +import { Column, CreateDateColumn, DeleteDateColumn, Entity, Index, OneToMany, PrimaryGeneratedColumn, } from 'typeorm'; import { Question } from './question.entity'; - @Entity() export class Assessment { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - title: string; - - @Column({ nullable: true }) - description?: string; - - @Column() - @Index() - durationMinutes: number; - - @OneToMany(() => Question, (q) => q.assessment, { - cascade: true, - }) - questions: Question[]; - - @CreateDateColumn() - createdAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + title: string; + @Column({ nullable: true }) + description?: string; + @Column() + @Index() + durationMinutes: number; + @OneToMany(() => Question, (q) => q.assessment, { + cascade: true, + }) + questions: Question[]; + @CreateDateColumn() + createdAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/assessment/entities/question.entity.ts b/src/assessment/entities/question.entity.ts index 0237443a..90d55977 100644 --- a/src/assessment/entities/question.entity.ts +++ b/src/assessment/entities/question.entity.ts @@ -1,41 +1,26 @@ -import { - Column, - DeleteDateColumn, - Entity, - Index, - ManyToOne, - PrimaryGeneratedColumn, -} from 'typeorm'; +import { Column, DeleteDateColumn, Entity, Index, ManyToOne, PrimaryGeneratedColumn, } from 'typeorm'; import { QuestionType } from '../enums/question-type.enum'; import { Assessment } from './assessment.entity'; - @Entity() export class Question { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'enum', enum: QuestionType }) - @Index() - type: QuestionType; - - @Column() - @Index() - prompt: string; - - @Column({ type: 'json', nullable: true }) - options?: string[]; // MCQ - - @Column({ type: 'json', nullable: true }) - correctAnswer?: any; - - @Column({ default: 1 }) - points: number; - - @ManyToOne(() => Assessment, (a) => a.questions, { - onDelete: 'CASCADE', - }) - assessment: Assessment; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'enum', enum: QuestionType }) + @Index() + type: QuestionType; + @Column() + @Index() + prompt: string; + @Column({ type: 'json', nullable: true }) + options?: string[]; // MCQ + @Column({ type: 'json', nullable: true }) + correctAnswer?: unknown; + @Column({ default: 1 }) + points: number; + @ManyToOne(() => Assessment, (a) => a.questions, { + onDelete: 'CASCADE', + }) + assessment: Assessment; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/assessment/enums/assessment-status.enum.ts b/src/assessment/enums/assessment-status.enum.ts index e02c3cff..50347d52 100644 --- a/src/assessment/enums/assessment-status.enum.ts +++ b/src/assessment/enums/assessment-status.enum.ts @@ -1,6 +1,6 @@ export enum AssessmentStatus { - IN_PROGRESS = 'in_progress', - SUBMITTED = 'submitted', - GRADED = 'graded', - TIMED_OUT = 'timed_out', + IN_PROGRESS = 'in_progress', + SUBMITTED = 'submitted', + GRADED = 'graded', + TIMED_OUT = 'timed_out' } diff --git a/src/assessment/enums/question-type.enum.ts b/src/assessment/enums/question-type.enum.ts index 3cf90b53..fd77df5b 100644 --- a/src/assessment/enums/question-type.enum.ts +++ b/src/assessment/enums/question-type.enum.ts @@ -1,5 +1,5 @@ export enum QuestionType { - MULTIPLE_CHOICE = 'multiple_choice', - TRUE_FALSE = 'true_false', - CODING = 'coding', + MULTIPLE_CHOICE = 'multiple_choice', + TRUE_FALSE = 'true_false', + CODING = 'coding' } diff --git a/src/assessment/feedback/feedback-generation.service.ts b/src/assessment/feedback/feedback-generation.service.ts index e977d130..3cd28450 100644 --- a/src/assessment/feedback/feedback-generation.service.ts +++ b/src/assessment/feedback/feedback-generation.service.ts @@ -1,12 +1,12 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class FeedbackGenerationService { - generate(score: number, maxScore: number): string { - const percentage = (score / maxScore) * 100; - - if (percentage >= 80) return 'Excellent performance 🎉'; - if (percentage >= 50) return 'Good job, but there is room to improve 👍'; - return 'Keep practicing, you can do better 💪'; - } + generate(score: number, maxScore: number): string { + const percentage = (score / maxScore) * 100; + if (percentage >= 80) + return 'Excellent performance 🎉'; + if (percentage >= 50) + return 'Good job, but there is room to improve 👍'; + return 'Keep practicing, you can do better 💪'; + } } diff --git a/src/assessment/questions/question-bank.service.ts b/src/assessment/questions/question-bank.service.ts index 58c2fa7d..b662ade4 100644 --- a/src/assessment/questions/question-bank.service.ts +++ b/src/assessment/questions/question-bank.service.ts @@ -2,21 +2,17 @@ import { Question } from '../entities/question.entity'; import { Repository } from 'typeorm'; import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; - @Injectable() export class QuestionBankService { - constructor( + constructor( @InjectRepository(Question) - private readonly questionRepo: Repository, - ) {} - - create(question: Partial): Promise { - return this.questionRepo.save(question); - } - - findByAssessment(assessmentId: string): Promise { - return this.questionRepo.find({ - where: { assessment: { id: assessmentId } }, - }); - } + private readonly questionRepo: Repository) { } + create(question: Partial): Promise { + return this.questionRepo.save(question); + } + findByAssessment(assessmentId: string): Promise { + return this.questionRepo.find({ + where: { assessment: { id: assessmentId } }, + }); + } } diff --git a/src/assessment/scoring/score-calculation.service.ts b/src/assessment/scoring/score-calculation.service.ts index 060be7be..215e3aca 100644 --- a/src/assessment/scoring/score-calculation.service.ts +++ b/src/assessment/scoring/score-calculation.service.ts @@ -1,21 +1,18 @@ import { Injectable } from '@nestjs/common'; import { Question } from '../entities/question.entity'; import { QuestionType } from '../enums/question-type.enum'; - @Injectable() export class ScoreCalculationService { - calculate(question: Question, response: any): number { - switch (question.type) { - case QuestionType.MULTIPLE_CHOICE: - case QuestionType.TRUE_FALSE: - return response === question.correctAnswer ? question.points : 0; - - case QuestionType.CODING: - // Placeholder (extend with judge later) - return response?.passed ? question.points : 0; - - default: - return 0; + calculate(question: Question, response: unknown): number { + switch (question.type) { + case QuestionType.MULTIPLE_CHOICE: + case QuestionType.TRUE_FALSE: + return response === question.correctAnswer ? question.points : 0; + case QuestionType.CODING: + // Placeholder (extend with judge later) + return response?.passed ? question.points : 0; + default: + return 0; + } } - } } diff --git a/src/audit-log/audit-log.controller.ts b/src/audit-log/audit-log.controller.ts index 75386f31..d5d5661b 100644 --- a/src/audit-log/audit-log.controller.ts +++ b/src/audit-log/audit-log.controller.ts @@ -1,260 +1,276 @@ -import { - Controller, - Get, - Post, - Query, - Param, - Res, - UseGuards, - HttpException, - HttpStatus, - Logger, - ParseIntPipe, - DefaultValuePipe, -} from '@nestjs/common'; -import { - ApiTags, - ApiOperation, - ApiResponse, - ApiQuery, - ApiParam, - ApiBearerAuth, -} from '@nestjs/swagger'; +import { Controller, Get, Post, Query, Param, Res, UseGuards, HttpException, HttpStatus, Logger, ParseIntPipe, DefaultValuePipe, } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiQuery, ApiParam, ApiBearerAuth, } from '@nestjs/swagger'; import { Response } from 'express'; import { AuditLogService, AuditLogSearchFilters } from './audit-log.service'; import { AuditLog } from './audit-log.entity'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { AuditAction, AuditCategory, AuditSeverity } from './enums/audit-action.enum'; - @ApiTags('Audit Logs') @ApiBearerAuth() @UseGuards(JwtAuthGuard) @Controller('audit-logs') export class AuditLogController { - private readonly logger = new Logger(AuditLogController.name); - - constructor(private readonly auditLogService: AuditLogService) {} - - @Get() - @ApiOperation({ summary: 'Search audit logs with filters' }) - @ApiQuery({ name: 'userId', required: false, description: 'Filter by user ID' }) - @ApiQuery({ name: 'userEmail', required: false, description: 'Filter by user email' }) - @ApiQuery({ - name: 'actions', - required: false, - description: 'Filter by actions (comma-separated)', - }) - @ApiQuery({ - name: 'categories', - required: false, - description: 'Filter by categories (comma-separated)', - }) - @ApiQuery({ - name: 'severities', - required: false, - description: 'Filter by severities (comma-separated)', - }) - @ApiQuery({ name: 'entityType', required: false, description: 'Filter by entity type' }) - @ApiQuery({ name: 'entityId', required: false, description: 'Filter by entity ID' }) - @ApiQuery({ name: 'ipAddress', required: false, description: 'Filter by IP address' }) - @ApiQuery({ name: 'sessionId', required: false, description: 'Filter by session ID' }) - @ApiQuery({ name: 'tenantId', required: false, description: 'Filter by tenant ID' }) - @ApiQuery({ name: 'apiEndpoint', required: false, description: 'Filter by API endpoint' }) - @ApiQuery({ name: 'httpMethod', required: false, description: 'Filter by HTTP method' }) - @ApiQuery({ name: 'statusCode', required: false, description: 'Filter by HTTP status code' }) - @ApiQuery({ name: 'startDate', required: false, description: 'Start date (ISO 8601)' }) - @ApiQuery({ name: 'endDate', required: false, description: 'End date (ISO 8601)' }) - @ApiQuery({ name: 'page', required: false, description: 'Page number', type: Number }) - @ApiQuery({ name: 'limit', required: false, description: 'Items per page', type: Number }) - @ApiResponse({ status: 200, description: 'Search results' }) - async search( - @Query('userId') userId?: string, - @Query('userEmail') userEmail?: string, - @Query('actions') actions?: string, - @Query('categories') categories?: string, - @Query('severities') severities?: string, - @Query('entityType') entityType?: string, - @Query('entityId') entityId?: string, - @Query('ipAddress') ipAddress?: string, - @Query('sessionId') sessionId?: string, - @Query('tenantId') tenantId?: string, - @Query('apiEndpoint') apiEndpoint?: string, - @Query('httpMethod') httpMethod?: string, - @Query('statusCode') statusCode?: string, - @Query('startDate') startDate?: string, - @Query('endDate') endDate?: string, - @Query('page', new DefaultValuePipe(1), ParseIntPipe) page?: number, - @Query('limit', new DefaultValuePipe(50), ParseIntPipe) limit?: number, - ) { - const filters: AuditLogSearchFilters = {}; - - if (userId) filters.userId = userId; - if (userEmail) filters.userEmail = userEmail; - if (actions) filters.actions = actions.split(',') as AuditAction[]; - if (categories) filters.categories = categories.split(',') as AuditCategory[]; - if (severities) filters.severities = severities.split(',') as AuditSeverity[]; - if (entityType) filters.entityType = entityType; - if (entityId) filters.entityId = entityId; - if (ipAddress) filters.ipAddress = ipAddress; - if (sessionId) filters.sessionId = sessionId; - if (tenantId) filters.tenantId = tenantId; - if (apiEndpoint) filters.apiEndpoint = apiEndpoint; - if (httpMethod) filters.httpMethod = httpMethod; - if (statusCode) filters.statusCode = parseInt(statusCode, 10); - if (startDate) filters.startDate = new Date(startDate); - if (endDate) filters.endDate = new Date(endDate); - - return this.auditLogService.search(filters, page, limit); - } - - @Get('recent') - @ApiOperation({ summary: 'Get recent audit logs' }) - @ApiQuery({ - name: 'limit', - required: false, - description: 'Number of logs to return', - type: Number, - }) - @ApiResponse({ status: 200, description: 'Recent audit logs' }) - async getRecent( - @Query('limit', new DefaultValuePipe(100), ParseIntPipe) limit?: number, - ): Promise { - return this.auditLogService.findAll(limit); - } - - @Get('user/:userId') - @ApiOperation({ summary: 'Get audit logs for a specific user' }) - @ApiParam({ name: 'userId', description: 'User ID' }) - @ApiQuery({ - name: 'limit', - required: false, - description: 'Number of logs to return', - type: Number, - }) - @ApiResponse({ status: 200, description: 'User audit logs' }) - async getByUser( - @Param('userId') userId: string, - @Query('limit', new DefaultValuePipe(100), ParseIntPipe) limit?: number, - ): Promise { - return this.auditLogService.findByUser(userId, limit); - } - - @Get('entity/:entityType/:entityId') - @ApiOperation({ summary: 'Get audit logs for a specific entity' }) - @ApiParam({ name: 'entityType', description: 'Entity type (e.g., user, course)' }) - @ApiParam({ name: 'entityId', description: 'Entity ID' }) - @ApiQuery({ - name: 'limit', - required: false, - description: 'Number of logs to return', - type: Number, - }) - @ApiResponse({ status: 200, description: 'Entity audit logs' }) - async getByEntity( - @Param('entityType') entityType: string, - @Param('entityId') entityId: string, - @Query('limit', new DefaultValuePipe(100), ParseIntPipe) limit?: number, - ): Promise { - return this.auditLogService.findByEntity(entityType, entityId, limit); - } - - @Get('ip/:ipAddress') - @ApiOperation({ summary: 'Get audit logs by IP address' }) - @ApiParam({ name: 'ipAddress', description: 'IP address' }) - @ApiQuery({ - name: 'limit', - required: false, - description: 'Number of logs to return', - type: Number, - }) - @ApiResponse({ status: 200, description: 'IP audit logs' }) - async getByIpAddress( - @Param('ipAddress') ipAddress: string, - @Query('limit', new DefaultValuePipe(100), ParseIntPipe) limit?: number, - ): Promise { - return this.auditLogService.findByIpAddress(ipAddress, limit); - } - - @Get('statistics') - @ApiOperation({ summary: 'Get audit log statistics' }) - @ApiResponse({ status: 200, description: 'Statistics' }) - async getStatistics() { - return this.auditLogService.getStatistics(); - } - - @Get('report') - @ApiOperation({ summary: 'Generate audit report' }) - @ApiQuery({ name: 'startDate', required: true, description: 'Start date (ISO 8601)' }) - @ApiQuery({ name: 'endDate', required: true, description: 'End date (ISO 8601)' }) - @ApiResponse({ status: 200, description: 'Audit report' }) - async generateReport(@Query('startDate') startDate: string, @Query('endDate') endDate: string) { - if (!startDate || !endDate) { - throw new HttpException('Start date and end date are required', HttpStatus.BAD_REQUEST); + private readonly logger = new Logger(AuditLogController.name); + constructor(private readonly auditLogService: AuditLogService) { } + @Get() + @ApiOperation({ summary: 'Search audit logs with filters' }) + @ApiQuery({ name: 'userId', required: false, description: 'Filter by user ID' }) + @ApiQuery({ name: 'userEmail', required: false, description: 'Filter by user email' }) + @ApiQuery({ + name: 'actions', + required: false, + description: 'Filter by actions (comma-separated)', + }) + @ApiQuery({ + name: 'categories', + required: false, + description: 'Filter by categories (comma-separated)', + }) + @ApiQuery({ + name: 'severities', + required: false, + description: 'Filter by severities (comma-separated)', + }) + @ApiQuery({ name: 'entityType', required: false, description: 'Filter by entity type' }) + @ApiQuery({ name: 'entityId', required: false, description: 'Filter by entity ID' }) + @ApiQuery({ name: 'ipAddress', required: false, description: 'Filter by IP address' }) + @ApiQuery({ name: 'sessionId', required: false, description: 'Filter by session ID' }) + @ApiQuery({ name: 'tenantId', required: false, description: 'Filter by tenant ID' }) + @ApiQuery({ name: 'apiEndpoint', required: false, description: 'Filter by API endpoint' }) + @ApiQuery({ name: 'httpMethod', required: false, description: 'Filter by HTTP method' }) + @ApiQuery({ name: 'statusCode', required: false, description: 'Filter by HTTP status code' }) + @ApiQuery({ name: 'startDate', required: false, description: 'Start date (ISO 8601)' }) + @ApiQuery({ name: 'endDate', required: false, description: 'End date (ISO 8601)' }) + @ApiQuery({ name: 'page', required: false, description: 'Page number', type: Number }) + @ApiQuery({ name: 'limit', required: false, description: 'Items per page', type: Number }) + @ApiResponse({ status: 200, description: 'Search results' }) + async search( + @Query('userId') + userId?: string, + @Query('userEmail') + userEmail?: string, + @Query('actions') + actions?: string, + @Query('categories') + categories?: string, + @Query('severities') + severities?: string, + @Query('entityType') + entityType?: string, + @Query('entityId') + entityId?: string, + @Query('ipAddress') + ipAddress?: string, + @Query('sessionId') + sessionId?: string, + @Query('tenantId') + tenantId?: string, + @Query('apiEndpoint') + apiEndpoint?: string, + @Query('httpMethod') + httpMethod?: string, + @Query('statusCode') + statusCode?: string, + @Query('startDate') + startDate?: string, + @Query('endDate') + endDate?: string, + @Query('page', new DefaultValuePipe(1), ParseIntPipe) + page?: number, + @Query('limit', new DefaultValuePipe(50), ParseIntPipe) + limit?: number) { + const filters: AuditLogSearchFilters = {}; + if (userId) + filters.userId = userId; + if (userEmail) + filters.userEmail = userEmail; + if (actions) + filters.actions = actions.split(',') as AuditAction[]; + if (categories) + filters.categories = categories.split(',') as AuditCategory[]; + if (severities) + filters.severities = severities.split(',') as AuditSeverity[]; + if (entityType) + filters.entityType = entityType; + if (entityId) + filters.entityId = entityId; + if (ipAddress) + filters.ipAddress = ipAddress; + if (sessionId) + filters.sessionId = sessionId; + if (tenantId) + filters.tenantId = tenantId; + if (apiEndpoint) + filters.apiEndpoint = apiEndpoint; + if (httpMethod) + filters.httpMethod = httpMethod; + if (statusCode) + filters.statusCode = parseInt(statusCode, 10); + if (startDate) + filters.startDate = new Date(startDate); + if (endDate) + filters.endDate = new Date(endDate); + return this.auditLogService.search(filters, page, limit); + } + @Get('recent') + @ApiOperation({ summary: 'Get recent audit logs' }) + @ApiQuery({ + name: 'limit', + required: false, + description: 'Number of logs to return', + type: Number, + }) + @ApiResponse({ status: 200, description: 'Recent audit logs' }) + async getRecent( + @Query('limit', new DefaultValuePipe(100), ParseIntPipe) + limit?: number): Promise { + return this.auditLogService.findAll(limit); + } + @Get('user/:userId') + @ApiOperation({ summary: 'Get audit logs for a specific user' }) + @ApiParam({ name: 'userId', description: 'User ID' }) + @ApiQuery({ + name: 'limit', + required: false, + description: 'Number of logs to return', + type: Number, + }) + @ApiResponse({ status: 200, description: 'User audit logs' }) + async getByUser( + @Param('userId') + userId: string, + @Query('limit', new DefaultValuePipe(100), ParseIntPipe) + limit?: number): Promise { + return this.auditLogService.findByUser(userId, limit); + } + @Get('entity/:entityType/:entityId') + @ApiOperation({ summary: 'Get audit logs for a specific entity' }) + @ApiParam({ name: 'entityType', description: 'Entity type (e.g., user, course)' }) + @ApiParam({ name: 'entityId', description: 'Entity ID' }) + @ApiQuery({ + name: 'limit', + required: false, + description: 'Number of logs to return', + type: Number, + }) + @ApiResponse({ status: 200, description: 'Entity audit logs' }) + async getByEntity( + @Param('entityType') + entityType: string, + @Param('entityId') + entityId: string, + @Query('limit', new DefaultValuePipe(100), ParseIntPipe) + limit?: number): Promise { + return this.auditLogService.findByEntity(entityType, entityId, limit); + } + @Get('ip/:ipAddress') + @ApiOperation({ summary: 'Get audit logs by IP address' }) + @ApiParam({ name: 'ipAddress', description: 'IP address' }) + @ApiQuery({ + name: 'limit', + required: false, + description: 'Number of logs to return', + type: Number, + }) + @ApiResponse({ status: 200, description: 'IP audit logs' }) + async getByIpAddress( + @Param('ipAddress') + ipAddress: string, + @Query('limit', new DefaultValuePipe(100), ParseIntPipe) + limit?: number): Promise { + return this.auditLogService.findByIpAddress(ipAddress, limit); + } + @Get('statistics') + @ApiOperation({ summary: 'Get audit log statistics' }) + @ApiResponse({ status: 200, description: 'Statistics' }) + async getStatistics() { + return this.auditLogService.getStatistics(); + } + @Get('report') + @ApiOperation({ summary: 'Generate audit report' }) + @ApiQuery({ name: 'startDate', required: true, description: 'Start date (ISO 8601)' }) + @ApiQuery({ name: 'endDate', required: true, description: 'End date (ISO 8601)' }) + @ApiResponse({ status: 200, description: 'Audit report' }) + async generateReport( + @Query('startDate') + startDate: string, + @Query('endDate') + endDate: string) { + if (!startDate || !endDate) { + throw new HttpException('Start date and end date are required', HttpStatus.BAD_REQUEST); + } + return this.auditLogService.generateReport(new Date(startDate), new Date(endDate)); + } + @Post('export/json') + @ApiOperation({ summary: 'Export audit logs to JSON' }) + @ApiQuery({ name: 'userId', required: false }) + @ApiQuery({ name: 'actions', required: false }) + @ApiQuery({ name: 'startDate', required: false }) + @ApiQuery({ name: 'endDate', required: false }) + async exportToJson( + @Res() + res: Response, + @Query('userId') + userId?: string, + @Query('actions') + actions?: string, + @Query('startDate') + startDate?: string, + @Query('endDate') + endDate?: string) { + const filters: AuditLogSearchFilters = {}; + if (userId) + filters.userId = userId; + if (actions) + filters.actions = actions.split(',') as AuditAction[]; + if (startDate) + filters.startDate = new Date(startDate); + if (endDate) + filters.endDate = new Date(endDate); + const json = await this.auditLogService.exportToJson(filters); + res.setHeader('Content-Type', 'application/json'); + res.setHeader('Content-Disposition', 'attachment; filename=audit-logs.json'); + res.send(json); + } + @Post('export/csv') + @ApiOperation({ summary: 'Export audit logs to CSV' }) + @ApiQuery({ name: 'userId', required: false }) + @ApiQuery({ name: 'actions', required: false }) + @ApiQuery({ name: 'startDate', required: false }) + @ApiQuery({ name: 'endDate', required: false }) + async exportToCsv( + @Res() + res: Response, + @Query('userId') + userId?: string, + @Query('actions') + actions?: string, + @Query('startDate') + startDate?: string, + @Query('endDate') + endDate?: string) { + const filters: AuditLogSearchFilters = {}; + if (userId) + filters.userId = userId; + if (actions) + filters.actions = actions.split(',') as AuditAction[]; + if (startDate) + filters.startDate = new Date(startDate); + if (endDate) + filters.endDate = new Date(endDate); + const csv = await this.auditLogService.exportToCsv(filters); + res.setHeader('Content-Type', 'text/csv'); + res.setHeader('Content-Disposition', 'attachment; filename=audit-logs.csv'); + res.send(csv); + } + @Post('retention/apply') + @ApiOperation({ summary: 'Apply retention policy (delete old logs)' }) + @ApiResponse({ status: 200, description: 'Retention policy applied' }) + async applyRetentionPolicy() { + const deletedCount = await this.auditLogService.applyRetentionPolicy(); + return { + message: 'Retention policy applied successfully', + deletedCount, + }; } - - return this.auditLogService.generateReport(new Date(startDate), new Date(endDate)); - } - - @Post('export/json') - @ApiOperation({ summary: 'Export audit logs to JSON' }) - @ApiQuery({ name: 'userId', required: false }) - @ApiQuery({ name: 'actions', required: false }) - @ApiQuery({ name: 'startDate', required: false }) - @ApiQuery({ name: 'endDate', required: false }) - async exportToJson( - @Res() res: Response, - @Query('userId') userId?: string, - @Query('actions') actions?: string, - @Query('startDate') startDate?: string, - @Query('endDate') endDate?: string, - ) { - const filters: AuditLogSearchFilters = {}; - if (userId) filters.userId = userId; - if (actions) filters.actions = actions.split(',') as AuditAction[]; - if (startDate) filters.startDate = new Date(startDate); - if (endDate) filters.endDate = new Date(endDate); - - const json = await this.auditLogService.exportToJson(filters); - - res.setHeader('Content-Type', 'application/json'); - res.setHeader('Content-Disposition', 'attachment; filename=audit-logs.json'); - res.send(json); - } - - @Post('export/csv') - @ApiOperation({ summary: 'Export audit logs to CSV' }) - @ApiQuery({ name: 'userId', required: false }) - @ApiQuery({ name: 'actions', required: false }) - @ApiQuery({ name: 'startDate', required: false }) - @ApiQuery({ name: 'endDate', required: false }) - async exportToCsv( - @Res() res: Response, - @Query('userId') userId?: string, - @Query('actions') actions?: string, - @Query('startDate') startDate?: string, - @Query('endDate') endDate?: string, - ) { - const filters: AuditLogSearchFilters = {}; - if (userId) filters.userId = userId; - if (actions) filters.actions = actions.split(',') as AuditAction[]; - if (startDate) filters.startDate = new Date(startDate); - if (endDate) filters.endDate = new Date(endDate); - - const csv = await this.auditLogService.exportToCsv(filters); - - res.setHeader('Content-Type', 'text/csv'); - res.setHeader('Content-Disposition', 'attachment; filename=audit-logs.csv'); - res.send(csv); - } - - @Post('retention/apply') - @ApiOperation({ summary: 'Apply retention policy (delete old logs)' }) - @ApiResponse({ status: 200, description: 'Retention policy applied' }) - async applyRetentionPolicy() { - const deletedCount = await this.auditLogService.applyRetentionPolicy(); - return { - message: 'Retention policy applied successfully', - deletedCount, - }; - } } diff --git a/src/audit-log/audit-log.entity.ts b/src/audit-log/audit-log.entity.ts index 675ead3d..21408575 100644 --- a/src/audit-log/audit-log.entity.ts +++ b/src/audit-log/audit-log.entity.ts @@ -1,6 +1,5 @@ import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, Index } from 'typeorm'; import { AuditAction, AuditSeverity, AuditCategory } from './enums/audit-action.enum'; - @Entity('audit_logs') @Index(['userId', 'timestamp']) @Index(['action', 'timestamp']) @@ -10,82 +9,60 @@ import { AuditAction, AuditSeverity, AuditCategory } from './enums/audit-action. @Index(['ipAddress']) @Index(['timestamp']) export class AuditLog { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ name: 'user_id', nullable: true }) - userId: string | null; - - @Column({ name: 'user_email', nullable: true }) - userEmail: string | null; - - @Column({ - type: 'enum', - enum: AuditAction, - }) - action: AuditAction; - - @Column({ - type: 'enum', - enum: AuditCategory, - }) - category: AuditCategory; - - @Column({ - type: 'enum', - enum: AuditSeverity, - default: AuditSeverity.INFO, - }) - severity: AuditSeverity; - - @Column({ name: 'entity_type', nullable: true }) - entityType: string | null; - - @Column({ name: 'entity_id', nullable: true }) - entityId: string | null; - - @Column({ type: 'text', nullable: true }) - description: string | null; - - @Column({ type: 'jsonb', nullable: true }) - metadata: Record | null; - - @Column({ name: 'old_values', type: 'jsonb', nullable: true }) - oldValues: Record | null; - - @Column({ name: 'new_values', type: 'jsonb', nullable: true }) - newValues: Record | null; - - @Column({ name: 'ip_address', nullable: true }) - ipAddress: string | null; - - @Column({ name: 'user_agent', nullable: true }) - userAgent: string | null; - - @Column({ name: 'session_id', nullable: true }) - sessionId: string | null; - - @Column({ name: 'request_id', nullable: true }) - requestId: string | null; - - @Column({ name: 'api_endpoint', nullable: true }) - apiEndpoint: string | null; - - @Column({ name: 'http_method', nullable: true }) - httpMethod: string | null; - - @Column({ name: 'status_code', nullable: true }) - statusCode: number | null; - - @Column({ name: 'response_time_ms', nullable: true }) - responseTimeMs: number | null; - - @Column({ name: 'tenant_id', nullable: true }) - tenantId: string | null; - - @CreateDateColumn({ name: 'timestamp' }) - timestamp: Date; - - @Column({ name: 'retention_until', nullable: true }) - retentionUntil: Date | null; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ name: 'user_id', nullable: true }) + userId: string | null; + @Column({ name: 'user_email', nullable: true }) + userEmail: string | null; + @Column({ + type: 'enum', + enum: AuditAction, + }) + action: AuditAction; + @Column({ + type: 'enum', + enum: AuditCategory, + }) + category: AuditCategory; + @Column({ + type: 'enum', + enum: AuditSeverity, + default: AuditSeverity.INFO, + }) + severity: AuditSeverity; + @Column({ name: 'entity_type', nullable: true }) + entityType: string | null; + @Column({ name: 'entity_id', nullable: true }) + entityId: string | null; + @Column({ type: 'text', nullable: true }) + description: string | null; + @Column({ type: 'jsonb', nullable: true }) + metadata: Record | null; + @Column({ name: 'old_values', type: 'jsonb', nullable: true }) + oldValues: Record | null; + @Column({ name: 'new_values', type: 'jsonb', nullable: true }) + newValues: Record | null; + @Column({ name: 'ip_address', nullable: true }) + ipAddress: string | null; + @Column({ name: 'user_agent', nullable: true }) + userAgent: string | null; + @Column({ name: 'session_id', nullable: true }) + sessionId: string | null; + @Column({ name: 'request_id', nullable: true }) + requestId: string | null; + @Column({ name: 'api_endpoint', nullable: true }) + apiEndpoint: string | null; + @Column({ name: 'http_method', nullable: true }) + httpMethod: string | null; + @Column({ name: 'status_code', nullable: true }) + statusCode: number | null; + @Column({ name: 'response_time_ms', nullable: true }) + responseTimeMs: number | null; + @Column({ name: 'tenant_id', nullable: true }) + tenantId: string | null; + @CreateDateColumn({ name: 'timestamp' }) + timestamp: Date; + @Column({ name: 'retention_until', nullable: true }) + retentionUntil: Date | null; } diff --git a/src/audit-log/audit-log.module.ts b/src/audit-log/audit-log.module.ts index f3696692..b583bb33 100644 --- a/src/audit-log/audit-log.module.ts +++ b/src/audit-log/audit-log.module.ts @@ -7,11 +7,11 @@ import { AuditLogService } from './audit-log.service'; import { AuditLogController } from './audit-log.controller'; import { AuditLogInterceptor } from './interceptors/audit-log.interceptor'; import { AuditRetentionTask } from './tasks/audit-retention.task'; - @Module({ - imports: [TypeOrmModule.forFeature([AuditLog]), ConfigModule, ScheduleModule.forRoot()], - controllers: [AuditLogController], - providers: [AuditLogService, AuditLogInterceptor, AuditRetentionTask], - exports: [AuditLogService, AuditLogInterceptor], + imports: [TypeOrmModule.forFeature([AuditLog]), ConfigModule, ScheduleModule.forRoot()], + controllers: [AuditLogController], + providers: [AuditLogService, AuditLogInterceptor, AuditRetentionTask], + exports: [AuditLogService, AuditLogInterceptor], }) -export class AuditLogModule {} +export class AuditLogModule { +} diff --git a/src/audit-log/audit-log.service.spec.ts b/src/audit-log/audit-log.service.spec.ts index 65785930..4cc418c6 100644 --- a/src/audit-log/audit-log.service.spec.ts +++ b/src/audit-log/audit-log.service.spec.ts @@ -3,765 +3,550 @@ import { getRepositoryToken } from '@nestjs/typeorm'; import { AuditLogService } from './audit-log.service'; import { AuditLog } from './audit-log.entity'; import { AuditAction, AuditSeverity, AuditCategory } from './enums/audit-action.enum'; -import { - createMockRepository, - createMockConfigService, - createMockQueryBuilder, -} from 'test/utils/mock-factories'; +import { createMockRepository, createMockConfigService, createMockQueryBuilder, } from 'test/utils/mock-factories'; import { Repository } from 'typeorm'; - describe('AuditLogService', () => { - // ───────────────────────────────────────────────────────────────────────── - // DECLARATIONS - // ───────────────────────────────────────────────────────────────────────── - - let service: AuditLogService; - let mockAuditRepo: jest.Mocked>; - let mockConfigService: jest.Mocked; - - // ───────────────────────────────────────────────────────────────────────── - // SETUP & TEARDOWN - // ───────────────────────────────────────────────────────────────────────── - - beforeEach(async () => { - // Initialize dependency mocks - mockAuditRepo = createMockRepository(); - mockConfigService = createMockConfigService({ - AUDIT_LOG_RETENTION_DAYS: 365, - }); - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - AuditLogService, - { - provide: getRepositoryToken(AuditLog), - useValue: mockAuditRepo, - }, - { - provide: 'ConfigService', - useValue: mockConfigService, - }, - ], - }).compile(); - - service = module.get(AuditLogService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - // ───────────────────────────────────────────────────────────────────────── - // TEST SUITES - // ───────────────────────────────────────────────────────────────────────── - - describe('constructor', () => { - it('should initialize with default retention days', () => { - const defaultService = new AuditLogService(mockAuditRepo, { - get: jest.fn().mockReturnValue(undefined), - } as any); - - expect((defaultService as any).retentionDays).toBe(365); - }); - - it('should use configured retention days', () => { - expect((service as any).retentionDays).toBe(365); - }); - }); - - describe('log', () => { - const auditEntry = { - userId: 'user-1', - userEmail: 'test@example.com', - action: AuditAction.LOGIN, - category: AuditCategory.AUTHENTICATION, - severity: AuditSeverity.INFO, - description: 'User logged in', - ipAddress: '127.0.0.1', - userAgent: 'TestAgent', - metadata: { sessionId: 'session-1' }, - }; - - const mockSavedLog = { - id: 'log-1', - ...auditEntry, - timestamp: new Date(), - retentionUntil: new Date(Date.now() + 365 * 24 * 60 * 60 * 1000), - }; - - beforeEach(() => { - mockAuditRepo.create.mockReturnValue(mockSavedLog as AuditLog); - mockAuditRepo.save.mockResolvedValue(mockSavedLog as AuditLog); - }); - - it('should create and save audit log entry', async () => { - const result = await service.log(auditEntry); - - expect(result).toEqual(mockSavedLog); - expect(mockAuditRepo.create).toHaveBeenCalledWith({ - ...auditEntry, - severity: AuditSeverity.INFO, - retentionUntil: expect.any(Date), - }); - expect(mockAuditRepo.save).toHaveBeenCalledWith(mockSavedLog); - }); - - it('should use default severity when not provided', async () => { - const entryWithoutSeverity = { ...auditEntry }; - delete entryWithoutSeverity.severity; - - await service.log(entryWithoutSeverity); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - severity: AuditSeverity.INFO, - }), - ); - }); - - it('should handle save errors gracefully', async () => { - const error = new Error('Database error'); - mockAuditRepo.save.mockRejectedValue(error); - - const consoleSpy = jest.spyOn(console, 'error').mockImplementation(); - - const result = await service.log(auditEntry); - - expect(result).toEqual(mockSavedLog); // Returns created log even on error - expect(consoleSpy).toHaveBeenCalledWith('Failed to create audit log:', error); - - consoleSpy.mockRestore(); - }); - - it('should set retention date correctly', async () => { - const retentionDays = 365; - const expectedRetentionDate = new Date(); - expectedRetentionDate.setDate(expectedRetentionDate.getDate() + retentionDays); - - await service.log(auditEntry); - - const createCall = mockAuditRepo.create.mock.calls[0][0]; - expect(createCall.retentionUntil).toBeInstanceOf(Date); - expect(createCall.retentionUntil.getDate()).toBe(expectedRetentionDate.getDate()); - }); - }); - - describe('logAuth', () => { - const authParams = { - action: AuditAction.LOGIN, - userId: 'user-1', - userEmail: 'test@example.com', - ipAddress: '127.0.0.1', - userAgent: 'TestAgent', - metadata: { sessionId: 'session-1' }, - }; - - beforeEach(() => { - mockAuditRepo.create.mockReturnValue({} as AuditLog); - mockAuditRepo.save.mockResolvedValue({} as AuditLog); - }); - - it('should log authentication event with correct category', async () => { - await service.logAuth( - authParams.action, - authParams.userId, - authParams.userEmail, - authParams.ipAddress, - authParams.userAgent, - authParams.metadata, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: authParams.userId, - userEmail: authParams.userEmail, - action: authParams.action, - category: AuditCategory.AUTHENTICATION, - severity: AuditSeverity.INFO, - ipAddress: authParams.ipAddress, - userAgent: authParams.userAgent, - metadata: authParams.metadata, - }), - ); - }); - - it('should handle null userId and userEmail', async () => { - await service.logAuth( - AuditAction.LOGIN_FAILED, - null, - null, - '127.0.0.1', - 'TestAgent', - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: undefined, - userEmail: undefined, - action: AuditAction.LOGIN_FAILED, - category: AuditCategory.AUTHENTICATION, - }), - ); - }); - - it('should use provided severity', async () => { - await service.logAuth( - AuditAction.LOGIN_FAILED, - 'user-1', - 'test@example.com', - '127.0.0.1', - 'TestAgent', - undefined, - AuditSeverity.WARNING, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - severity: AuditSeverity.WARNING, - }), - ); - }); - }); - - describe('logDataChange', () => { - const dataChangeParams = { - action: AuditAction.UPDATE, - userId: 'user-1', - userEmail: 'test@example.com', - entityType: 'User', - entityId: 'user-1', - oldValues: { name: 'Old Name' }, - newValues: { name: 'New Name' }, - ipAddress: '127.0.0.1', - description: 'User profile updated', - }; - - beforeEach(() => { - mockAuditRepo.create.mockReturnValue({} as AuditLog); - mockAuditRepo.save.mockResolvedValue({} as AuditLog); - }); - - it('should log data change event with correct category', async () => { - await service.logDataChange( - dataChangeParams.action, - dataChangeParams.userId, - dataChangeParams.userEmail, - dataChangeParams.entityType, - dataChangeParams.entityId, - dataChangeParams.oldValues, - dataChangeParams.newValues, - dataChangeParams.ipAddress, - dataChangeParams.description, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: dataChangeParams.userId, - userEmail: dataChangeParams.userEmail, - action: dataChangeParams.action, - category: AuditCategory.DATA_MODIFICATION, - severity: AuditSeverity.INFO, - entityType: dataChangeParams.entityType, - entityId: dataChangeParams.entityId, - oldValues: dataChangeParams.oldValues, - newValues: dataChangeParams.newValues, - ipAddress: dataChangeParams.ipAddress, - description: dataChangeParams.description, - }), - ); - }); - }); - - describe('logApiAccess', () => { - const apiAccessParams = { - userId: 'user-1', - userEmail: 'test@example.com', - apiEndpoint: '/api/users', - httpMethod: 'GET', - statusCode: 200, - responseTimeMs: 150, - ipAddress: '127.0.0.1', - userAgent: 'TestAgent', - requestId: 'req-123', - }; - - beforeEach(() => { - mockAuditRepo.create.mockReturnValue({} as AuditLog); - mockAuditRepo.save.mockResolvedValue({} as AuditLog); - }); - - it('should log API access with INFO severity for 2xx status', async () => { - await service.logApiAccess( - apiAccessParams.userId, - apiAccessParams.userEmail, - apiAccessParams.apiEndpoint, - apiAccessParams.httpMethod, - apiAccessParams.statusCode, - apiAccessParams.responseTimeMs, - apiAccessParams.ipAddress, - apiAccessParams.userAgent, - apiAccessParams.requestId, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: apiAccessParams.userId, - userEmail: apiAccessParams.userEmail, - action: AuditAction.API_CALLED, - category: AuditCategory.DATA_ACCESS, - severity: AuditSeverity.INFO, - apiEndpoint: apiAccessParams.apiEndpoint, - httpMethod: apiAccessParams.httpMethod, - statusCode: apiAccessParams.statusCode, - responseTimeMs: apiAccessParams.responseTimeMs, - ipAddress: apiAccessParams.ipAddress, - userAgent: apiAccessParams.userAgent, - requestId: apiAccessParams.requestId, - }), - ); - }); - - it('should log API access with WARNING severity for 4xx status', async () => { - await service.logApiAccess( - apiAccessParams.userId, - apiAccessParams.userEmail, - apiAccessParams.apiEndpoint, - apiAccessParams.httpMethod, - 404, - apiAccessParams.responseTimeMs, - apiAccessParams.ipAddress, - apiAccessParams.userAgent, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - severity: AuditSeverity.WARNING, - statusCode: 404, - }), - ); - }); - - it('should log API access with ERROR severity for 5xx status', async () => { - await service.logApiAccess( - apiAccessParams.userId, - apiAccessParams.userEmail, - apiAccessParams.apiEndpoint, - apiAccessParams.httpMethod, - 500, - apiAccessParams.responseTimeMs, - apiAccessParams.ipAddress, - apiAccessParams.userAgent, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - severity: AuditSeverity.ERROR, - statusCode: 500, - }), - ); - }); - - it('should handle null userId and userEmail', async () => { - await service.logApiAccess( - null, - null, - apiAccessParams.apiEndpoint, - apiAccessParams.httpMethod, - apiAccessParams.statusCode, - apiAccessParams.responseTimeMs, - apiAccessParams.ipAddress, - apiAccessParams.userAgent, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: undefined, - userEmail: undefined, - }), - ); - }); - }); - - describe('logSecurityEvent', () => { - const securityParams = { - action: AuditAction.SECURITY_ALERT, - userId: 'user-1', - userEmail: 'test@example.com', - ipAddress: '127.0.0.1', - userAgent: 'TestAgent', - description: 'Suspicious activity detected', - metadata: { threatLevel: 'high' }, - }; - - beforeEach(() => { - mockAuditRepo.create.mockReturnValue({} as AuditLog); - mockAuditRepo.save.mockResolvedValue({} as AuditLog); - }); - - it('should log security event with WARNING severity', async () => { - await service.logSecurityEvent( - securityParams.action, - securityParams.userId, - securityParams.userEmail, - securityParams.ipAddress, - securityParams.userAgent, - securityParams.description, - securityParams.metadata, - ); - - expect(mockAuditRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - userId: securityParams.userId, - userEmail: securityParams.userEmail, - action: securityParams.action, - category: AuditCategory.SECURITY, - severity: AuditSeverity.WARNING, - ipAddress: securityParams.ipAddress, - userAgent: securityParams.userAgent, - description: securityParams.description, - metadata: securityParams.metadata, - }), - ); - }); - }); - - describe('search', () => { - const mockQueryBuilder = createMockQueryBuilder(); - const mockLogs = [ - { id: 'log-1', action: AuditAction.LOGIN }, - { id: 'log-2', action: AuditAction.LOGOUT }, - ]; - - beforeEach(() => { - mockAuditRepo.createQueryBuilder.mockReturnValue(mockQueryBuilder as any); - mockQueryBuilder.getCount.mockResolvedValue(2); - mockQueryBuilder.getMany.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should search audit logs with filters', async () => { - const filters = { - userId: 'user-1', - actions: [AuditAction.LOGIN], - startDate: new Date('2024-01-01'), - endDate: new Date('2024-01-31'), - }; - - const result = await service.search(filters, 1, 10); - - expect(result).toEqual({ - logs: mockLogs, - total: 2, - page: 1, - limit: 10, - totalPages: 1, - }); - - expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith( - 'audit.userId = :userId', - { userId: 'user-1' }, - ); - expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith( - 'audit.action IN (:...actions)', - { actions: [AuditAction.LOGIN] }, - ); - expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith( - 'audit.timestamp BETWEEN :startDate AND :endDate', - { startDate: filters.startDate, endDate: filters.endDate }, - ); - }); - - it('should apply pagination correctly', async () => { - await service.search({}, 2, 20); - - expect(mockQueryBuilder.skip).toHaveBeenCalledWith(20); // (page-1) * limit - expect(mockQueryBuilder.take).toHaveBeenCalledWith(20); - }); - - it('should handle empty filters', async () => { - await service.search({}); - - expect(mockQueryBuilder.andWhere).not.toHaveBeenCalled(); - }); - }); - - describe('findAll', () => { - const mockLogs = [ - { id: 'log-1', timestamp: new Date() }, - { id: 'log-2', timestamp: new Date() }, - ]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should return all logs ordered by timestamp desc', async () => { - const result = await service.findAll(50); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - order: { timestamp: 'DESC' }, - take: 50, - }); - }); - - it('should use default limit of 100', async () => { - await service.findAll(); - - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - order: { timestamp: 'DESC' }, - take: 100, - }); - }); - }); - - describe('findByUser', () => { - const userId = 'user-1'; - const mockLogs = [{ id: 'log-1', userId }]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should find logs by user ID', async () => { - const result = await service.findByUser(userId, 25); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - where: { userId }, - order: { timestamp: 'DESC' }, - take: 25, - }); - }); - }); - - describe('findByAction', () => { - const action = AuditAction.LOGIN; - const mockLogs = [{ id: 'log-1', action }]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should find logs by action', async () => { - const result = await service.findByAction(action, 30); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - where: { action }, - order: { timestamp: 'DESC' }, - take: 30, - }); - }); - }); - - describe('findByEntity', () => { - const entityType = 'User'; - const entityId = 'user-1'; - const mockLogs = [{ id: 'log-1', entityType, entityId }]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should find logs by entity', async () => { - const result = await service.findByEntity(entityType, entityId, 40); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - where: { entityType, entityId }, - order: { timestamp: 'DESC' }, - take: 40, - }); - }); - }); - - describe('findByIpAddress', () => { - const ipAddress = '127.0.0.1'; - const mockLogs = [{ id: 'log-1', ipAddress }]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should find logs by IP address', async () => { - const result = await service.findByIpAddress(ipAddress, 35); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - where: { ipAddress }, - order: { timestamp: 'DESC' }, - take: 35, - }); - }); - }); - - describe('findByDateRange', () => { - const startDate = new Date('2024-01-01'); - const endDate = new Date('2024-01-31'); - const mockLogs = [{ id: 'log-1', timestamp: new Date('2024-01-15') }]; - - beforeEach(() => { - mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); - }); - - it('should find logs by date range', async () => { - const result = await service.findByDateRange(startDate, endDate, 1000); - - expect(result).toEqual(mockLogs); - expect(mockAuditRepo.find).toHaveBeenCalledWith({ - where: { - timestamp: expect.any(Object), // Between operator - }, - order: { timestamp: 'DESC' }, - take: 1000, - }); - }); - }); - - describe('generateReport', () => { - const startDate = new Date('2024-01-01'); - const endDate = new Date('2024-01-31'); - - const mockCategoryStats = [ - { category: AuditCategory.AUTHENTICATION, count: '10' }, - { category: AuditCategory.DATA_MODIFICATION, count: '5' }, - ]; - - const mockActionStats = [ - { action: AuditAction.LOGIN, count: '8' }, - { action: AuditAction.LOGOUT, count: '2' }, - ]; - - const mockSeverityStats = [ - { severity: AuditSeverity.INFO, count: '12' }, - { severity: AuditSeverity.WARNING, count: '3' }, - ]; - - const mockTopUsers = [ - { userId: 'user-1', userEmail: 'test@example.com', count: '5' }, - ]; - - const mockTopEndpoints = [ - { endpoint: '/api/users', count: '10' }, - ]; - - const mockFailedActions = [ - { action: AuditAction.API_CALLED, count: '2' }, - ]; - - beforeEach(() => { - // Mock the main query builder - const mainQueryBuilder = createMockQueryBuilder(); - mainQueryBuilder.where.mockReturnThis(); - mainQueryBuilder.getCount.mockResolvedValue(15); - - // Mock category stats query - const categoryQueryBuilder = createMockQueryBuilder(); - categoryQueryBuilder.select.mockReturnThis(); - categoryQueryBuilder.addSelect.mockReturnThis(); - categoryQueryBuilder.where.mockReturnThis(); - categoryQueryBuilder.groupBy.mockReturnThis(); - categoryQueryBuilder.getRawMany.mockResolvedValue(mockCategoryStats); - - // Mock action stats query - const actionQueryBuilder = createMockQueryBuilder(); - actionQueryBuilder.select.mockReturnThis(); - actionQueryBuilder.addSelect.mockReturnThis(); - actionQueryBuilder.where.mockReturnThis(); - actionQueryBuilder.groupBy.mockReturnThis(); - actionQueryBuilder.getRawMany.mockResolvedValue(mockActionStats); - - // Mock severity stats query - const severityQueryBuilder = createMockQueryBuilder(); - severityQueryBuilder.select.mockReturnThis(); - severityQueryBuilder.addSelect.mockReturnThis(); - severityQueryBuilder.where.mockReturnThis(); - severityQueryBuilder.groupBy.mockReturnThis(); - severityQueryBuilder.getRawMany.mockResolvedValue(mockSeverityStats); - - // Mock top users query - const topUsersQueryBuilder = createMockQueryBuilder(); - topUsersQueryBuilder.select.mockReturnThis(); - topUsersQueryBuilder.addSelect.mockReturnThis(); - topUsersQueryBuilder.where.mockReturnThis(); - topUsersQueryBuilder.andWhere.mockReturnThis(); - topUsersQueryBuilder.groupBy.mockReturnThis(); - topUsersQueryBuilder.addGroupBy.mockReturnThis(); - topUsersQueryBuilder.orderBy.mockReturnThis(); - topUsersQueryBuilder.limit.mockReturnThis(); - topUsersQueryBuilder.getRawMany.mockResolvedValue(mockTopUsers); - - // Mock top endpoints query - const topEndpointsQueryBuilder = createMockQueryBuilder(); - topEndpointsQueryBuilder.select.mockReturnThis(); - topEndpointsQueryBuilder.addSelect.mockReturnThis(); - topEndpointsQueryBuilder.where.mockReturnThis(); - topEndpointsQueryBuilder.andWhere.mockReturnThis(); - topEndpointsQueryBuilder.groupBy.mockReturnThis(); - topEndpointsQueryBuilder.orderBy.mockReturnThis(); - topEndpointsQueryBuilder.limit.mockReturnThis(); - topEndpointsQueryBuilder.getRawMany.mockResolvedValue(mockTopEndpoints); - - // Mock failed actions query - const failedActionsQueryBuilder = createMockQueryBuilder(); - failedActionsQueryBuilder.select.mockReturnThis(); - failedActionsQueryBuilder.addSelect.mockReturnThis(); - failedActionsQueryBuilder.where.mockReturnThis(); - failedActionsQueryBuilder.andWhere.mockReturnThis(); - failedActionsQueryBuilder.groupBy.mockReturnThis(); - failedActionsQueryBuilder.orderBy.mockReturnThis(); - failedActionsQueryBuilder.limit.mockReturnThis(); - failedActionsQueryBuilder.getRawMany.mockResolvedValue(mockFailedActions); - - // Set up the mock to return different query builders for different calls - mockAuditRepo.createQueryBuilder - .mockReturnValueOnce(mainQueryBuilder as any) - .mockReturnValueOnce(categoryQueryBuilder as any) - .mockReturnValueOnce(actionQueryBuilder as any) - .mockReturnValueOnce(severityQueryBuilder as any) - .mockReturnValueOnce(topUsersQueryBuilder as any) - .mockReturnValueOnce(topEndpointsQueryBuilder as any) - .mockReturnValueOnce(failedActionsQueryBuilder as any); - }); - - it('should generate comprehensive audit report', async () => { - const result = await service.generateReport(startDate, endDate); - - expect(result).toEqual({ - period: { start: startDate, end: endDate }, - totalEvents: 15, - eventsByCategory: { - [AuditCategory.AUTHENTICATION]: 10, - [AuditCategory.DATA_MODIFICATION]: 5, - }, - eventsByAction: { - [AuditAction.LOGIN]: 8, - [AuditAction.LOGOUT]: 2, - }, - eventsBySeverity: { - [AuditSeverity.INFO]: 12, - [AuditSeverity.WARNING]: 3, - }, - topUsers: [ - { + // ───────────────────────────────────────────────────────────────────────── + // DECLARATIONS + // ───────────────────────────────────────────────────────────────────────── + let service: AuditLogService; + let mockAuditRepo: jest.Mocked>; + let mockConfigService: jest.Mocked; + // ───────────────────────────────────────────────────────────────────────── + // SETUP & TEARDOWN + // ───────────────────────────────────────────────────────────────────────── + beforeEach(async () => { + // Initialize dependency mocks + mockAuditRepo = createMockRepository(); + mockConfigService = createMockConfigService({ + AUDIT_LOG_RETENTION_DAYS: 365, + }); + const module: TestingModule = await Test.createTestingModule({ + providers: [ + AuditLogService, + { + provide: getRepositoryToken(AuditLog), + useValue: mockAuditRepo, + }, + { + provide: 'ConfigService', + useValue: mockConfigService, + }, + ], + }).compile(); + service = module.get(AuditLogService); + }); + afterEach(() => { + jest.clearAllMocks(); + }); + // ───────────────────────────────────────────────────────────────────────── + // TEST SUITES + // ───────────────────────────────────────────────────────────────────────── + describe('constructor', () => { + it('should initialize with default retention days', () => { + const defaultService = new AuditLogService(mockAuditRepo, { + get: jest.fn().mockReturnValue(undefined), + } as unknown); + expect((defaultService as unknown).retentionDays).toBe(365); + }); + it('should use configured retention days', () => { + expect((service as unknown).retentionDays).toBe(365); + }); + }); + describe('log', () => { + const auditEntry = { userId: 'user-1', userEmail: 'test@example.com', - count: 5, - }, - ], - topEndpoints: [ - { - endpoint: '/api/users', - count: 10, - }, - ], - failedActions: [ - { - action: AuditAction.API_CALLED, - count: 2, - }, - ], - }); - }); - }); -}); \ No newline at end of file + action: AuditAction.LOGIN, + category: AuditCategory.AUTHENTICATION, + severity: AuditSeverity.INFO, + description: 'User logged in', + ipAddress: '127.0.0.1', + userAgent: 'TestAgent', + metadata: { sessionId: 'session-1' }, + }; + const mockSavedLog = { + id: 'log-1', + ...auditEntry, + timestamp: new Date(), + retentionUntil: new Date(Date.now() + 365 * 24 * 60 * 60 * 1000), + }; + beforeEach(() => { + mockAuditRepo.create.mockReturnValue(mockSavedLog as AuditLog); + mockAuditRepo.save.mockResolvedValue(mockSavedLog as AuditLog); + }); + it('should create and save audit log entry', async () => { + const result = await service.log(auditEntry); + expect(result).toEqual(mockSavedLog); + expect(mockAuditRepo.create).toHaveBeenCalledWith({ + ...auditEntry, + severity: AuditSeverity.INFO, + retentionUntil: expect.any(Date), + }); + expect(mockAuditRepo.save).toHaveBeenCalledWith(mockSavedLog); + }); + it('should use default severity when not provided', async () => { + const entryWithoutSeverity = { ...auditEntry }; + delete entryWithoutSeverity.severity; + await service.log(entryWithoutSeverity); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + severity: AuditSeverity.INFO, + })); + }); + it('should handle save errors gracefully', async () => { + const error = new Error('Database error'); + mockAuditRepo.save.mockRejectedValue(error); + const consoleSpy = jest.spyOn(console, 'error').mockImplementation(); + const result = await service.log(auditEntry); + expect(result).toEqual(mockSavedLog); // Returns created log even on error + expect(consoleSpy).toHaveBeenCalledWith('Failed to create audit log:', error); + consoleSpy.mockRestore(); + }); + it('should set retention date correctly', async () => { + const retentionDays = 365; + const expectedRetentionDate = new Date(); + expectedRetentionDate.setDate(expectedRetentionDate.getDate() + retentionDays); + await service.log(auditEntry); + const createCall = mockAuditRepo.create.mock.calls[0][0]; + expect(createCall.retentionUntil).toBeInstanceOf(Date); + expect(createCall.retentionUntil.getDate()).toBe(expectedRetentionDate.getDate()); + }); + }); + describe('logAuth', () => { + const authParams = { + action: AuditAction.LOGIN, + userId: 'user-1', + userEmail: 'test@example.com', + ipAddress: '127.0.0.1', + userAgent: 'TestAgent', + metadata: { sessionId: 'session-1' }, + }; + beforeEach(() => { + mockAuditRepo.create.mockReturnValue({} as AuditLog); + mockAuditRepo.save.mockResolvedValue({} as AuditLog); + }); + it('should log authentication event with correct category', async () => { + await service.logAuth(authParams.action, authParams.userId, authParams.userEmail, authParams.ipAddress, authParams.userAgent, authParams.metadata); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: authParams.userId, + userEmail: authParams.userEmail, + action: authParams.action, + category: AuditCategory.AUTHENTICATION, + severity: AuditSeverity.INFO, + ipAddress: authParams.ipAddress, + userAgent: authParams.userAgent, + metadata: authParams.metadata, + })); + }); + it('should handle null userId and userEmail', async () => { + await service.logAuth(AuditAction.LOGIN_FAILED, null, null, '127.0.0.1', 'TestAgent'); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: undefined, + userEmail: undefined, + action: AuditAction.LOGIN_FAILED, + category: AuditCategory.AUTHENTICATION, + })); + }); + it('should use provided severity', async () => { + await service.logAuth(AuditAction.LOGIN_FAILED, 'user-1', 'test@example.com', '127.0.0.1', 'TestAgent', undefined, AuditSeverity.WARNING); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + severity: AuditSeverity.WARNING, + })); + }); + }); + describe('logDataChange', () => { + const dataChangeParams = { + action: AuditAction.UPDATE, + userId: 'user-1', + userEmail: 'test@example.com', + entityType: 'User', + entityId: 'user-1', + oldValues: { name: 'Old Name' }, + newValues: { name: 'New Name' }, + ipAddress: '127.0.0.1', + description: 'User profile updated', + }; + beforeEach(() => { + mockAuditRepo.create.mockReturnValue({} as AuditLog); + mockAuditRepo.save.mockResolvedValue({} as AuditLog); + }); + it('should log data change event with correct category', async () => { + await service.logDataChange(dataChangeParams.action, dataChangeParams.userId, dataChangeParams.userEmail, dataChangeParams.entityType, dataChangeParams.entityId, dataChangeParams.oldValues, dataChangeParams.newValues, dataChangeParams.ipAddress, dataChangeParams.description); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: dataChangeParams.userId, + userEmail: dataChangeParams.userEmail, + action: dataChangeParams.action, + category: AuditCategory.DATA_MODIFICATION, + severity: AuditSeverity.INFO, + entityType: dataChangeParams.entityType, + entityId: dataChangeParams.entityId, + oldValues: dataChangeParams.oldValues, + newValues: dataChangeParams.newValues, + ipAddress: dataChangeParams.ipAddress, + description: dataChangeParams.description, + })); + }); + }); + describe('logApiAccess', () => { + const apiAccessParams = { + userId: 'user-1', + userEmail: 'test@example.com', + apiEndpoint: '/api/users', + httpMethod: 'GET', + statusCode: 200, + responseTimeMs: 150, + ipAddress: '127.0.0.1', + userAgent: 'TestAgent', + requestId: 'req-123', + }; + beforeEach(() => { + mockAuditRepo.create.mockReturnValue({} as AuditLog); + mockAuditRepo.save.mockResolvedValue({} as AuditLog); + }); + it('should log API access with INFO severity for 2xx status', async () => { + await service.logApiAccess(apiAccessParams.userId, apiAccessParams.userEmail, apiAccessParams.apiEndpoint, apiAccessParams.httpMethod, apiAccessParams.statusCode, apiAccessParams.responseTimeMs, apiAccessParams.ipAddress, apiAccessParams.userAgent, apiAccessParams.requestId); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: apiAccessParams.userId, + userEmail: apiAccessParams.userEmail, + action: AuditAction.API_CALLED, + category: AuditCategory.DATA_ACCESS, + severity: AuditSeverity.INFO, + apiEndpoint: apiAccessParams.apiEndpoint, + httpMethod: apiAccessParams.httpMethod, + statusCode: apiAccessParams.statusCode, + responseTimeMs: apiAccessParams.responseTimeMs, + ipAddress: apiAccessParams.ipAddress, + userAgent: apiAccessParams.userAgent, + requestId: apiAccessParams.requestId, + })); + }); + it('should log API access with WARNING severity for 4xx status', async () => { + await service.logApiAccess(apiAccessParams.userId, apiAccessParams.userEmail, apiAccessParams.apiEndpoint, apiAccessParams.httpMethod, 404, apiAccessParams.responseTimeMs, apiAccessParams.ipAddress, apiAccessParams.userAgent); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + severity: AuditSeverity.WARNING, + statusCode: 404, + })); + }); + it('should log API access with ERROR severity for 5xx status', async () => { + await service.logApiAccess(apiAccessParams.userId, apiAccessParams.userEmail, apiAccessParams.apiEndpoint, apiAccessParams.httpMethod, 500, apiAccessParams.responseTimeMs, apiAccessParams.ipAddress, apiAccessParams.userAgent); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + severity: AuditSeverity.ERROR, + statusCode: 500, + })); + }); + it('should handle null userId and userEmail', async () => { + await service.logApiAccess(null, null, apiAccessParams.apiEndpoint, apiAccessParams.httpMethod, apiAccessParams.statusCode, apiAccessParams.responseTimeMs, apiAccessParams.ipAddress, apiAccessParams.userAgent); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: undefined, + userEmail: undefined, + })); + }); + }); + describe('logSecurityEvent', () => { + const securityParams = { + action: AuditAction.SECURITY_ALERT, + userId: 'user-1', + userEmail: 'test@example.com', + ipAddress: '127.0.0.1', + userAgent: 'TestAgent', + description: 'Suspicious activity detected', + metadata: { threatLevel: 'high' }, + }; + beforeEach(() => { + mockAuditRepo.create.mockReturnValue({} as AuditLog); + mockAuditRepo.save.mockResolvedValue({} as AuditLog); + }); + it('should log security event with WARNING severity', async () => { + await service.logSecurityEvent(securityParams.action, securityParams.userId, securityParams.userEmail, securityParams.ipAddress, securityParams.userAgent, securityParams.description, securityParams.metadata); + expect(mockAuditRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + userId: securityParams.userId, + userEmail: securityParams.userEmail, + action: securityParams.action, + category: AuditCategory.SECURITY, + severity: AuditSeverity.WARNING, + ipAddress: securityParams.ipAddress, + userAgent: securityParams.userAgent, + description: securityParams.description, + metadata: securityParams.metadata, + })); + }); + }); + describe('search', () => { + const mockQueryBuilder = createMockQueryBuilder(); + const mockLogs = [ + { id: 'log-1', action: AuditAction.LOGIN }, + { id: 'log-2', action: AuditAction.LOGOUT }, + ]; + beforeEach(() => { + mockAuditRepo.createQueryBuilder.mockReturnValue(mockQueryBuilder as unknown); + mockQueryBuilder.getCount.mockResolvedValue(2); + mockQueryBuilder.getMany.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should search audit logs with filters', async () => { + const filters = { + userId: 'user-1', + actions: [AuditAction.LOGIN], + startDate: new Date('2024-01-01'), + endDate: new Date('2024-01-31'), + }; + const result = await service.search(filters, 1, 10); + expect(result).toEqual({ + logs: mockLogs, + total: 2, + page: 1, + limit: 10, + totalPages: 1, + }); + expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith('audit.userId = :userId', { userId: 'user-1' }); + expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith('audit.action IN (:...actions)', { actions: [AuditAction.LOGIN] }); + expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith('audit.timestamp BETWEEN :startDate AND :endDate', { startDate: filters.startDate, endDate: filters.endDate }); + }); + it('should apply pagination correctly', async () => { + await service.search({}, 2, 20); + expect(mockQueryBuilder.skip).toHaveBeenCalledWith(20); // (page-1) * limit + expect(mockQueryBuilder.take).toHaveBeenCalledWith(20); + }); + it('should handle empty filters', async () => { + await service.search({}); + expect(mockQueryBuilder.andWhere).not.toHaveBeenCalled(); + }); + }); + describe('findAll', () => { + const mockLogs = [ + { id: 'log-1', timestamp: new Date() }, + { id: 'log-2', timestamp: new Date() }, + ]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should return all logs ordered by timestamp desc', async () => { + const result = await service.findAll(50); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + order: { timestamp: 'DESC' }, + take: 50, + }); + }); + it('should use default limit of 100', async () => { + await service.findAll(); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + order: { timestamp: 'DESC' }, + take: 100, + }); + }); + }); + describe('findByUser', () => { + const userId = 'user-1'; + const mockLogs = [{ id: 'log-1', userId }]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should find logs by user ID', async () => { + const result = await service.findByUser(userId, 25); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + where: { userId }, + order: { timestamp: 'DESC' }, + take: 25, + }); + }); + }); + describe('findByAction', () => { + const action = AuditAction.LOGIN; + const mockLogs = [{ id: 'log-1', action }]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should find logs by action', async () => { + const result = await service.findByAction(action, 30); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + where: { action }, + order: { timestamp: 'DESC' }, + take: 30, + }); + }); + }); + describe('findByEntity', () => { + const entityType = 'User'; + const entityId = 'user-1'; + const mockLogs = [{ id: 'log-1', entityType, entityId }]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should find logs by entity', async () => { + const result = await service.findByEntity(entityType, entityId, 40); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + where: { entityType, entityId }, + order: { timestamp: 'DESC' }, + take: 40, + }); + }); + }); + describe('findByIpAddress', () => { + const ipAddress = '127.0.0.1'; + const mockLogs = [{ id: 'log-1', ipAddress }]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should find logs by IP address', async () => { + const result = await service.findByIpAddress(ipAddress, 35); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + where: { ipAddress }, + order: { timestamp: 'DESC' }, + take: 35, + }); + }); + }); + describe('findByDateRange', () => { + const startDate = new Date('2024-01-01'); + const endDate = new Date('2024-01-31'); + const mockLogs = [{ id: 'log-1', timestamp: new Date('2024-01-15') }]; + beforeEach(() => { + mockAuditRepo.find.mockResolvedValue(mockLogs as AuditLog[]); + }); + it('should find logs by date range', async () => { + const result = await service.findByDateRange(startDate, endDate, 1000); + expect(result).toEqual(mockLogs); + expect(mockAuditRepo.find).toHaveBeenCalledWith({ + where: { + timestamp: expect.any(Object), // Between operator + }, + order: { timestamp: 'DESC' }, + take: 1000, + }); + }); + }); + describe('generateReport', () => { + const startDate = new Date('2024-01-01'); + const endDate = new Date('2024-01-31'); + const mockCategoryStats = [ + { category: AuditCategory.AUTHENTICATION, count: '10' }, + { category: AuditCategory.DATA_MODIFICATION, count: '5' }, + ]; + const mockActionStats = [ + { action: AuditAction.LOGIN, count: '8' }, + { action: AuditAction.LOGOUT, count: '2' }, + ]; + const mockSeverityStats = [ + { severity: AuditSeverity.INFO, count: '12' }, + { severity: AuditSeverity.WARNING, count: '3' }, + ]; + const mockTopUsers = [ + { userId: 'user-1', userEmail: 'test@example.com', count: '5' }, + ]; + const mockTopEndpoints = [ + { endpoint: '/api/users', count: '10' }, + ]; + const mockFailedActions = [ + { action: AuditAction.API_CALLED, count: '2' }, + ]; + beforeEach(() => { + // Mock the main query builder + const mainQueryBuilder = createMockQueryBuilder(); + mainQueryBuilder.where.mockReturnThis(); + mainQueryBuilder.getCount.mockResolvedValue(15); + // Mock category stats query + const categoryQueryBuilder = createMockQueryBuilder(); + categoryQueryBuilder.select.mockReturnThis(); + categoryQueryBuilder.addSelect.mockReturnThis(); + categoryQueryBuilder.where.mockReturnThis(); + categoryQueryBuilder.groupBy.mockReturnThis(); + categoryQueryBuilder.getRawMany.mockResolvedValue(mockCategoryStats); + // Mock action stats query + const actionQueryBuilder = createMockQueryBuilder(); + actionQueryBuilder.select.mockReturnThis(); + actionQueryBuilder.addSelect.mockReturnThis(); + actionQueryBuilder.where.mockReturnThis(); + actionQueryBuilder.groupBy.mockReturnThis(); + actionQueryBuilder.getRawMany.mockResolvedValue(mockActionStats); + // Mock severity stats query + const severityQueryBuilder = createMockQueryBuilder(); + severityQueryBuilder.select.mockReturnThis(); + severityQueryBuilder.addSelect.mockReturnThis(); + severityQueryBuilder.where.mockReturnThis(); + severityQueryBuilder.groupBy.mockReturnThis(); + severityQueryBuilder.getRawMany.mockResolvedValue(mockSeverityStats); + // Mock top users query + const topUsersQueryBuilder = createMockQueryBuilder(); + topUsersQueryBuilder.select.mockReturnThis(); + topUsersQueryBuilder.addSelect.mockReturnThis(); + topUsersQueryBuilder.where.mockReturnThis(); + topUsersQueryBuilder.andWhere.mockReturnThis(); + topUsersQueryBuilder.groupBy.mockReturnThis(); + topUsersQueryBuilder.addGroupBy.mockReturnThis(); + topUsersQueryBuilder.orderBy.mockReturnThis(); + topUsersQueryBuilder.limit.mockReturnThis(); + topUsersQueryBuilder.getRawMany.mockResolvedValue(mockTopUsers); + // Mock top endpoints query + const topEndpointsQueryBuilder = createMockQueryBuilder(); + topEndpointsQueryBuilder.select.mockReturnThis(); + topEndpointsQueryBuilder.addSelect.mockReturnThis(); + topEndpointsQueryBuilder.where.mockReturnThis(); + topEndpointsQueryBuilder.andWhere.mockReturnThis(); + topEndpointsQueryBuilder.groupBy.mockReturnThis(); + topEndpointsQueryBuilder.orderBy.mockReturnThis(); + topEndpointsQueryBuilder.limit.mockReturnThis(); + topEndpointsQueryBuilder.getRawMany.mockResolvedValue(mockTopEndpoints); + // Mock failed actions query + const failedActionsQueryBuilder = createMockQueryBuilder(); + failedActionsQueryBuilder.select.mockReturnThis(); + failedActionsQueryBuilder.addSelect.mockReturnThis(); + failedActionsQueryBuilder.where.mockReturnThis(); + failedActionsQueryBuilder.andWhere.mockReturnThis(); + failedActionsQueryBuilder.groupBy.mockReturnThis(); + failedActionsQueryBuilder.orderBy.mockReturnThis(); + failedActionsQueryBuilder.limit.mockReturnThis(); + failedActionsQueryBuilder.getRawMany.mockResolvedValue(mockFailedActions); + // Set up the mock to return different query builders for different calls + mockAuditRepo.createQueryBuilder + .mockReturnValueOnce(mainQueryBuilder as unknown) + .mockReturnValueOnce(categoryQueryBuilder as unknown) + .mockReturnValueOnce(actionQueryBuilder as unknown) + .mockReturnValueOnce(severityQueryBuilder as unknown) + .mockReturnValueOnce(topUsersQueryBuilder as unknown) + .mockReturnValueOnce(topEndpointsQueryBuilder as unknown) + .mockReturnValueOnce(failedActionsQueryBuilder as unknown); + }); + it('should generate comprehensive audit report', async () => { + const result = await service.generateReport(startDate, endDate); + expect(result).toEqual({ + period: { start: startDate, end: endDate }, + totalEvents: 15, + eventsByCategory: { + [AuditCategory.AUTHENTICATION]: 10, + [AuditCategory.DATA_MODIFICATION]: 5, + }, + eventsByAction: { + [AuditAction.LOGIN]: 8, + [AuditAction.LOGOUT]: 2, + }, + eventsBySeverity: { + [AuditSeverity.INFO]: 12, + [AuditSeverity.WARNING]: 3, + }, + topUsers: [ + { + userId: 'user-1', + userEmail: 'test@example.com', + count: 5, + }, + ], + topEndpoints: [ + { + endpoint: '/api/users', + count: 10, + }, + ], + failedActions: [ + { + action: AuditAction.API_CALLED, + count: 2, + }, + ], + }); + }); + }); +}); diff --git a/src/audit-log/audit-log.service.ts b/src/audit-log/audit-log.service.ts index 627faf0b..1a59abaa 100644 --- a/src/audit-log/audit-log.service.ts +++ b/src/audit-log/audit-log.service.ts @@ -5,627 +5,525 @@ import { AuditLog } from './audit-log.entity'; import { AuditAction, AuditSeverity, AuditCategory } from './enums/audit-action.enum'; import { ConfigService } from '@nestjs/config'; import { sanitizePii } from '../common/utils/pii-sanitizer.utils'; - export interface AuditLogEntry { - userId?: string; - userEmail?: string; - action: AuditAction; - category: AuditCategory; - severity?: AuditSeverity; - entityType?: string; - entityId?: string; - description?: string; - metadata?: Record; - oldValues?: Record; - newValues?: Record; - ipAddress?: string; - userAgent?: string; - sessionId?: string; - requestId?: string; - apiEndpoint?: string; - httpMethod?: string; - statusCode?: number; - responseTimeMs?: number; - tenantId?: string; + userId?: string; + userEmail?: string; + action: AuditAction; + category: AuditCategory; + severity?: AuditSeverity; + entityType?: string; + entityId?: string; + description?: string; + metadata?: Record; + oldValues?: Record; + newValues?: Record; + ipAddress?: string; + userAgent?: string; + sessionId?: string; + requestId?: string; + apiEndpoint?: string; + httpMethod?: string; + statusCode?: number; + responseTimeMs?: number; + tenantId?: string; } - export interface AuditLogSearchFilters { - userId?: string; - userEmail?: string; - actions?: AuditAction[]; - categories?: AuditCategory[]; - severities?: AuditSeverity[]; - entityType?: string; - entityId?: string; - ipAddress?: string; - sessionId?: string; - tenantId?: string; - startDate?: Date; - endDate?: Date; - apiEndpoint?: string; - httpMethod?: string; - statusCode?: number; + userId?: string; + userEmail?: string; + actions?: AuditAction[]; + categories?: AuditCategory[]; + severities?: AuditSeverity[]; + entityType?: string; + entityId?: string; + ipAddress?: string; + sessionId?: string; + tenantId?: string; + startDate?: Date; + endDate?: Date; + apiEndpoint?: string; + httpMethod?: string; + statusCode?: number; } - export interface AuditLogSearchResult { - logs: AuditLog[]; - total: number; - page: number; - limit: number; - totalPages: number; + logs: AuditLog[]; + total: number; + page: number; + limit: number; + totalPages: number; } - export interface AuditReport { - period: { start: Date; end: Date }; - totalEvents: number; - eventsByCategory: Record; - eventsByAction: Record; - eventsBySeverity: Record; - topUsers: Array<{ userId: string; userEmail: string; count: number }>; - topEndpoints: Array<{ endpoint: string; count: number }>; - failedActions: Array<{ action: string; count: number }>; + period: { + start: Date; + end: Date; + }; + totalEvents: number; + eventsByCategory: Record; + eventsByAction: Record; + eventsBySeverity: Record; + topUsers: Array<{ + userId: string; + userEmail: string; + count: number; + }>; + topEndpoints: Array<{ + endpoint: string; + count: number; + }>; + failedActions: Array<{ + action: string; + count: number; + }>; } - @Injectable() export class AuditLogService { - private readonly logger = new Logger(AuditLogService.name); - private readonly retentionDays: number; - - constructor( + private readonly logger = new Logger(AuditLogService.name); + private readonly retentionDays: number; + constructor( @InjectRepository(AuditLog) - private readonly auditRepo: Repository, - private readonly configService: ConfigService, - ) { - this.retentionDays = this.configService.get('AUDIT_LOG_RETENTION_DAYS', 365); - } - - /** - * Log an audit event - */ - async log(entry: AuditLogEntry): Promise { - const retentionUntil = new Date(); - retentionUntil.setDate(retentionUntil.getDate() + this.retentionDays); - - const log = this.auditRepo.create({ - ...entry, - severity: entry.severity || AuditSeverity.INFO, - retentionUntil, - }); - - try { - const saved = await this.auditRepo.save(log); - this.logger.debug( - `Audit log created: ${log.action} - ${sanitizePii(log.description || 'no description')}`, - ); - return saved; - } catch (error) { - this.logger.error('Failed to create audit log:', error); - // Don't throw - audit logging should not break main functionality - return log; + private readonly auditRepo: Repository, private readonly configService: ConfigService) { + this.retentionDays = this.configService.get('AUDIT_LOG_RETENTION_DAYS', 365); } - } - - /** - * Log authentication event - */ - async logAuth( - action: AuditAction, - userId: string | null, - userEmail: string | null, - ipAddress: string, - userAgent: string, - metadata?: Record, - severity: AuditSeverity = AuditSeverity.INFO, - ): Promise { - return this.log({ - userId: userId || undefined, - userEmail: userEmail || undefined, - action, - category: AuditCategory.AUTHENTICATION, - severity, - ipAddress, - userAgent, - metadata, - }); - } - - /** - * Log data modification - */ - async logDataChange( - action: AuditAction, - userId: string, - userEmail: string, - entityType: string, - entityId: string, - oldValues: Record, - newValues: Record, - ipAddress?: string, - description?: string, - ): Promise { - return this.log({ - userId, - userEmail, - action, - category: AuditCategory.DATA_MODIFICATION, - severity: AuditSeverity.INFO, - entityType, - entityId, - description, - oldValues, - newValues, - ipAddress, - }); - } - - /** - * Log API access - */ - async logApiAccess( - userId: string | null, - userEmail: string | null, - apiEndpoint: string, - httpMethod: string, - statusCode: number, - responseTimeMs: number, - ipAddress: string, - userAgent: string, - requestId?: string, - ): Promise { - const severity = - statusCode >= 500 - ? AuditSeverity.ERROR - : statusCode >= 400 - ? AuditSeverity.WARNING - : AuditSeverity.INFO; - - return this.log({ - userId: userId || undefined, - userEmail: userEmail || undefined, - action: AuditAction.API_CALLED, - category: AuditCategory.DATA_ACCESS, - severity, - apiEndpoint, - httpMethod, - statusCode, - responseTimeMs, - ipAddress, - userAgent, - requestId, - }); - } - - /** - * Log security event - */ - async logSecurityEvent( - action: AuditAction, - userId: string | null, - userEmail: string | null, - ipAddress: string, - userAgent: string, - description: string, - metadata?: Record, - ): Promise { - return this.log({ - userId: userId || undefined, - userEmail: userEmail || undefined, - action, - category: AuditCategory.SECURITY, - severity: AuditSeverity.WARNING, - ipAddress, - userAgent, - description, - metadata, - }); - } - - /** - * Search audit logs with filters - */ - async search( - filters: AuditLogSearchFilters, - page: number = 1, - limit: number = 50, - ): Promise { - const queryBuilder = this.auditRepo.createQueryBuilder('audit'); - - // Apply filters - if (filters.userId) { - queryBuilder.andWhere('audit.userId = :userId', { userId: filters.userId }); + /** + * Log an audit event + */ + async log(entry: AuditLogEntry): Promise { + const retentionUntil = new Date(); + retentionUntil.setDate(retentionUntil.getDate() + this.retentionDays); + const log = this.auditRepo.create({ + ...entry, + severity: entry.severity || AuditSeverity.INFO, + retentionUntil, + }); + try { + const saved = await this.auditRepo.save(log); + this.logger.debug(`Audit log created: ${log.action} - ${sanitizePii(log.description || 'no description')}`); + return saved; + } + catch (error) { + this.logger.error('Failed to create audit log:', error); + // Don't throw - audit logging should not break main functionality + return log; + } } - - if (filters.userEmail) { - queryBuilder.andWhere('audit.userEmail = :userEmail', { userEmail: filters.userEmail }); + /** + * Log authentication event + */ + async logAuth(action: AuditAction, userId: string | null, userEmail: string | null, ipAddress: string, userAgent: string, metadata?: Record, severity: AuditSeverity = AuditSeverity.INFO): Promise { + return this.log({ + userId: userId || undefined, + userEmail: userEmail || undefined, + action, + category: AuditCategory.AUTHENTICATION, + severity, + ipAddress, + userAgent, + metadata, + }); } - - if (filters.actions && filters.actions.length > 0) { - queryBuilder.andWhere('audit.action IN (:...actions)', { actions: filters.actions }); + /** + * Log data modification + */ + async logDataChange(action: AuditAction, userId: string, userEmail: string, entityType: string, entityId: string, oldValues: Record, newValues: Record, ipAddress?: string, description?: string): Promise { + return this.log({ + userId, + userEmail, + action, + category: AuditCategory.DATA_MODIFICATION, + severity: AuditSeverity.INFO, + entityType, + entityId, + description, + oldValues, + newValues, + ipAddress, + }); } - - if (filters.categories && filters.categories.length > 0) { - queryBuilder.andWhere('audit.category IN (:...categories)', { - categories: filters.categories, - }); + /** + * Log API access + */ + async logApiAccess(userId: string | null, userEmail: string | null, apiEndpoint: string, httpMethod: string, statusCode: number, responseTimeMs: number, ipAddress: string, userAgent: string, requestId?: string): Promise { + const severity = statusCode >= 500 + ? AuditSeverity.ERROR + : statusCode >= 400 + ? AuditSeverity.WARNING + : AuditSeverity.INFO; + return this.log({ + userId: userId || undefined, + userEmail: userEmail || undefined, + action: AuditAction.API_CALLED, + category: AuditCategory.DATA_ACCESS, + severity, + apiEndpoint, + httpMethod, + statusCode, + responseTimeMs, + ipAddress, + userAgent, + requestId, + }); } - - if (filters.severities && filters.severities.length > 0) { - queryBuilder.andWhere('audit.severity IN (:...severities)', { - severities: filters.severities, - }); + /** + * Log security event + */ + async logSecurityEvent(action: AuditAction, userId: string | null, userEmail: string | null, ipAddress: string, userAgent: string, description: string, metadata?: Record): Promise { + return this.log({ + userId: userId || undefined, + userEmail: userEmail || undefined, + action, + category: AuditCategory.SECURITY, + severity: AuditSeverity.WARNING, + ipAddress, + userAgent, + description, + metadata, + }); } - - if (filters.entityType) { - queryBuilder.andWhere('audit.entityType = :entityType', { entityType: filters.entityType }); + /** + * Search audit logs with filters + */ + async search(filters: AuditLogSearchFilters, page: number = 1, limit: number = 50): Promise { + const queryBuilder = this.auditRepo.createQueryBuilder('audit'); + // Apply filters + if (filters.userId) { + queryBuilder.andWhere('audit.userId = :userId', { userId: filters.userId }); + } + if (filters.userEmail) { + queryBuilder.andWhere('audit.userEmail = :userEmail', { userEmail: filters.userEmail }); + } + if (filters.actions && filters.actions.length > 0) { + queryBuilder.andWhere('audit.action IN (:...actions)', { actions: filters.actions }); + } + if (filters.categories && filters.categories.length > 0) { + queryBuilder.andWhere('audit.category IN (:...categories)', { + categories: filters.categories, + }); + } + if (filters.severities && filters.severities.length > 0) { + queryBuilder.andWhere('audit.severity IN (:...severities)', { + severities: filters.severities, + }); + } + if (filters.entityType) { + queryBuilder.andWhere('audit.entityType = :entityType', { entityType: filters.entityType }); + } + if (filters.entityId) { + queryBuilder.andWhere('audit.entityId = :entityId', { entityId: filters.entityId }); + } + if (filters.ipAddress) { + queryBuilder.andWhere('audit.ipAddress = :ipAddress', { ipAddress: filters.ipAddress }); + } + if (filters.sessionId) { + queryBuilder.andWhere('audit.sessionId = :sessionId', { sessionId: filters.sessionId }); + } + if (filters.tenantId) { + queryBuilder.andWhere('audit.tenantId = :tenantId', { tenantId: filters.tenantId }); + } + if (filters.apiEndpoint) { + queryBuilder.andWhere('audit.apiEndpoint LIKE :apiEndpoint', { + apiEndpoint: `%${filters.apiEndpoint}%`, + }); + } + if (filters.httpMethod) { + queryBuilder.andWhere('audit.httpMethod = :httpMethod', { httpMethod: filters.httpMethod }); + } + if (filters.statusCode) { + queryBuilder.andWhere('audit.statusCode = :statusCode', { statusCode: filters.statusCode }); + } + if (filters.startDate && filters.endDate) { + queryBuilder.andWhere('audit.timestamp BETWEEN :startDate AND :endDate', { + startDate: filters.startDate, + endDate: filters.endDate, + }); + } + else if (filters.startDate) { + queryBuilder.andWhere('audit.timestamp >= :startDate', { startDate: filters.startDate }); + } + else if (filters.endDate) { + queryBuilder.andWhere('audit.timestamp <= :endDate', { endDate: filters.endDate }); + } + // Order by timestamp desc + queryBuilder.orderBy('audit.timestamp', 'DESC'); + // Get total count + const total = await queryBuilder.getCount(); + // Apply pagination + const skip = (page - 1) * limit; + queryBuilder.skip(skip).take(limit); + const logs = await queryBuilder.getMany(); + return { + logs, + total, + page, + limit, + totalPages: Math.ceil(total / limit), + }; } - - if (filters.entityId) { - queryBuilder.andWhere('audit.entityId = :entityId', { entityId: filters.entityId }); + /** + * Find all logs (with limit) + */ + async findAll(limit: number = 100): Promise { + return this.auditRepo.find({ + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.ipAddress) { - queryBuilder.andWhere('audit.ipAddress = :ipAddress', { ipAddress: filters.ipAddress }); + /** + * Find logs by user + */ + async findByUser(userId: string, limit: number = 100): Promise { + return this.auditRepo.find({ + where: { userId }, + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.sessionId) { - queryBuilder.andWhere('audit.sessionId = :sessionId', { sessionId: filters.sessionId }); + /** + * Find logs by action + */ + async findByAction(action: AuditAction, limit: number = 100): Promise { + return this.auditRepo.find({ + where: { action }, + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.tenantId) { - queryBuilder.andWhere('audit.tenantId = :tenantId', { tenantId: filters.tenantId }); + /** + * Find logs by entity + */ + async findByEntity(entityType: string, entityId: string, limit: number = 100): Promise { + return this.auditRepo.find({ + where: { entityType, entityId }, + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.apiEndpoint) { - queryBuilder.andWhere('audit.apiEndpoint LIKE :apiEndpoint', { - apiEndpoint: `%${filters.apiEndpoint}%`, - }); + /** + * Find logs by IP address + */ + async findByIpAddress(ipAddress: string, limit: number = 100): Promise { + return this.auditRepo.find({ + where: { ipAddress }, + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.httpMethod) { - queryBuilder.andWhere('audit.httpMethod = :httpMethod', { httpMethod: filters.httpMethod }); + /** + * Find logs by date range + */ + async findByDateRange(startDate: Date, endDate: Date, limit: number = 1000): Promise { + return this.auditRepo.find({ + where: { + timestamp: Between(startDate, endDate), + }, + order: { timestamp: 'DESC' }, + take: limit, + }); } - - if (filters.statusCode) { - queryBuilder.andWhere('audit.statusCode = :statusCode', { statusCode: filters.statusCode }); + /** + * Generate audit report + */ + async generateReport(startDate: Date, endDate: Date): Promise { + const queryBuilder = this.auditRepo.createQueryBuilder('audit'); + queryBuilder.where('audit.timestamp BETWEEN :startDate AND :endDate', { + startDate, + endDate, + }); + const totalEvents = await queryBuilder.getCount(); + // Events by category + const categoryStats = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.category', 'category') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .groupBy('audit.category') + .getRawMany(); + const eventsByCategory: Record = {}; + categoryStats.forEach((stat) => { + eventsByCategory[stat.category] = parseInt(stat.count, 10); + }); + // Events by action + const actionStats = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.action', 'action') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .groupBy('audit.action') + .getRawMany(); + const eventsByAction: Record = {}; + actionStats.forEach((stat) => { + eventsByAction[stat.action] = parseInt(stat.count, 10); + }); + // Events by severity + const severityStats = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.severity', 'severity') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .groupBy('audit.severity') + .getRawMany(); + const eventsBySeverity: Record = {}; + severityStats.forEach((stat) => { + eventsBySeverity[stat.severity] = parseInt(stat.count, 10); + }); + // Top users + const topUsers = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.userId', 'userId') + .addSelect('audit.userEmail', 'userEmail') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .andWhere('audit.userId IS NOT NULL') + .groupBy('audit.userId') + .addGroupBy('audit.userEmail') + .orderBy('count', 'DESC') + .limit(10) + .getRawMany(); + // Top endpoints + const topEndpoints = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.apiEndpoint', 'endpoint') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .andWhere('audit.apiEndpoint IS NOT NULL') + .groupBy('audit.apiEndpoint') + .orderBy('count', 'DESC') + .limit(10) + .getRawMany(); + // Failed actions (status code >= 400) + const failedActions = await this.auditRepo + .createQueryBuilder('audit') + .select('audit.action', 'action') + .addSelect('COUNT(*)', 'count') + .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) + .andWhere('audit.statusCode >= 400') + .groupBy('audit.action') + .orderBy('count', 'DESC') + .limit(10) + .getRawMany(); + return { + period: { start: startDate, end: endDate }, + totalEvents, + eventsByCategory, + eventsByAction, + eventsBySeverity, + topUsers: topUsers.map((u) => ({ + userId: u.userId, + userEmail: u.userEmail || 'Unknown', + count: parseInt(u.count, 10), + })), + topEndpoints: topEndpoints.map((e) => ({ + endpoint: e.endpoint, + count: parseInt(e.count, 10), + })), + failedActions: failedActions.map((f) => ({ + action: f.action, + count: parseInt(f.count, 10), + })), + }; } - - if (filters.startDate && filters.endDate) { - queryBuilder.andWhere('audit.timestamp BETWEEN :startDate AND :endDate', { - startDate: filters.startDate, - endDate: filters.endDate, - }); - } else if (filters.startDate) { - queryBuilder.andWhere('audit.timestamp >= :startDate', { startDate: filters.startDate }); - } else if (filters.endDate) { - queryBuilder.andWhere('audit.timestamp <= :endDate', { endDate: filters.endDate }); + /** + * Apply retention policy - delete old logs + */ + async applyRetentionPolicy(): Promise { + const cutoffDate = new Date(); + cutoffDate.setDate(cutoffDate.getDate() - this.retentionDays); + const result = await this.auditRepo + .createQueryBuilder() + .delete() + .where('timestamp < :cutoffDate', { cutoffDate }) + .orWhere('retentionUntil < NOW()') + .execute(); + const deletedCount = result.affected || 0; + this.logger.log(`Applied retention policy: deleted ${deletedCount} old audit logs`); + return deletedCount; + } + /** + * Export logs to JSON + */ + async exportToJson(filters: AuditLogSearchFilters): Promise { + const { logs } = await this.search(filters, 1, 10000); + return JSON.stringify(logs, null, 2); + } + /** + * Export logs to CSV + */ + async exportToCsv(filters: AuditLogSearchFilters): Promise { + const { logs } = await this.search(filters, 1, 10000); + const headers = [ + 'timestamp', + 'userId', + 'userEmail', + 'action', + 'category', + 'severity', + 'entityType', + 'entityId', + 'description', + 'ipAddress', + 'userAgent', + 'apiEndpoint', + 'httpMethod', + 'statusCode', + ]; + const rows = logs.map((log) => [ + log.timestamp.toISOString(), + log.userId || '', + log.userEmail || '', + log.action, + log.category, + log.severity, + log.entityType || '', + log.entityId || '', + log.description || '', + log.ipAddress || '', + log.userAgent || '', + log.apiEndpoint || '', + log.httpMethod || '', + log.statusCode || '', + ]); + const escapeCsv = (value: string | number) => { + const str = String(value); + if (str.includes(',') || str.includes('"') || str.includes('\n')) { + return `"${str.replace(/"/g, '""')}"`; + } + return str; + }; + const csvContent = [headers.join(','), ...rows.map((row) => row.map(escapeCsv).join(','))].join('\n'); + return csvContent; + } + /** + * Get statistics + */ + async getStatistics(): Promise<{ + totalLogs: number; + logsToday: number; + logsThisWeek: number; + logsThisMonth: number; + criticalEvents: number; + errorEvents: number; + }> { + const now = new Date(); + const today = new Date(now.getFullYear(), now.getMonth(), now.getDate()); + const weekAgo = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000); + const monthAgo = new Date(now.getTime() - 30 * 24 * 60 * 60 * 1000); + const [totalLogs, logsToday, logsThisWeek, logsThisMonth, criticalEvents, errorEvents] = await Promise.all([ + this.auditRepo.count(), + this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(today) } }), + this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(weekAgo) } }), + this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(monthAgo) } }), + this.auditRepo.count({ where: { severity: AuditSeverity.CRITICAL } }), + this.auditRepo.count({ where: { severity: AuditSeverity.ERROR } }), + ]); + return { + totalLogs, + logsToday, + logsThisWeek, + logsThisMonth, + criticalEvents, + errorEvents, + }; } - - // Order by timestamp desc - queryBuilder.orderBy('audit.timestamp', 'DESC'); - - // Get total count - const total = await queryBuilder.getCount(); - - // Apply pagination - const skip = (page - 1) * limit; - queryBuilder.skip(skip).take(limit); - - const logs = await queryBuilder.getMany(); - - return { - logs, - total, - page, - limit, - totalPages: Math.ceil(total / limit), - }; - } - - /** - * Find all logs (with limit) - */ - async findAll(limit: number = 100): Promise { - return this.auditRepo.find({ - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Find logs by user - */ - async findByUser(userId: string, limit: number = 100): Promise { - return this.auditRepo.find({ - where: { userId }, - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Find logs by action - */ - async findByAction(action: AuditAction, limit: number = 100): Promise { - return this.auditRepo.find({ - where: { action }, - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Find logs by entity - */ - async findByEntity( - entityType: string, - entityId: string, - limit: number = 100, - ): Promise { - return this.auditRepo.find({ - where: { entityType, entityId }, - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Find logs by IP address - */ - async findByIpAddress(ipAddress: string, limit: number = 100): Promise { - return this.auditRepo.find({ - where: { ipAddress }, - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Find logs by date range - */ - async findByDateRange(startDate: Date, endDate: Date, limit: number = 1000): Promise { - return this.auditRepo.find({ - where: { - timestamp: Between(startDate, endDate), - }, - order: { timestamp: 'DESC' }, - take: limit, - }); - } - - /** - * Generate audit report - */ - async generateReport(startDate: Date, endDate: Date): Promise { - const queryBuilder = this.auditRepo.createQueryBuilder('audit'); - queryBuilder.where('audit.timestamp BETWEEN :startDate AND :endDate', { - startDate, - endDate, - }); - - const totalEvents = await queryBuilder.getCount(); - - // Events by category - const categoryStats = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.category', 'category') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .groupBy('audit.category') - .getRawMany(); - - const eventsByCategory: Record = {}; - categoryStats.forEach((stat) => { - eventsByCategory[stat.category] = parseInt(stat.count, 10); - }); - - // Events by action - const actionStats = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.action', 'action') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .groupBy('audit.action') - .getRawMany(); - - const eventsByAction: Record = {}; - actionStats.forEach((stat) => { - eventsByAction[stat.action] = parseInt(stat.count, 10); - }); - - // Events by severity - const severityStats = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.severity', 'severity') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .groupBy('audit.severity') - .getRawMany(); - - const eventsBySeverity: Record = {}; - severityStats.forEach((stat) => { - eventsBySeverity[stat.severity] = parseInt(stat.count, 10); - }); - - // Top users - const topUsers = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.userId', 'userId') - .addSelect('audit.userEmail', 'userEmail') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .andWhere('audit.userId IS NOT NULL') - .groupBy('audit.userId') - .addGroupBy('audit.userEmail') - .orderBy('count', 'DESC') - .limit(10) - .getRawMany(); - - // Top endpoints - const topEndpoints = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.apiEndpoint', 'endpoint') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .andWhere('audit.apiEndpoint IS NOT NULL') - .groupBy('audit.apiEndpoint') - .orderBy('count', 'DESC') - .limit(10) - .getRawMany(); - - // Failed actions (status code >= 400) - const failedActions = await this.auditRepo - .createQueryBuilder('audit') - .select('audit.action', 'action') - .addSelect('COUNT(*)', 'count') - .where('audit.timestamp BETWEEN :startDate AND :endDate', { startDate, endDate }) - .andWhere('audit.statusCode >= 400') - .groupBy('audit.action') - .orderBy('count', 'DESC') - .limit(10) - .getRawMany(); - - return { - period: { start: startDate, end: endDate }, - totalEvents, - eventsByCategory, - eventsByAction, - eventsBySeverity, - topUsers: topUsers.map((u) => ({ - userId: u.userId, - userEmail: u.userEmail || 'Unknown', - count: parseInt(u.count, 10), - })), - topEndpoints: topEndpoints.map((e) => ({ - endpoint: e.endpoint, - count: parseInt(e.count, 10), - })), - failedActions: failedActions.map((f) => ({ - action: f.action, - count: parseInt(f.count, 10), - })), - }; - } - - /** - * Apply retention policy - delete old logs - */ - async applyRetentionPolicy(): Promise { - const cutoffDate = new Date(); - cutoffDate.setDate(cutoffDate.getDate() - this.retentionDays); - - const result = await this.auditRepo - .createQueryBuilder() - .delete() - .where('timestamp < :cutoffDate', { cutoffDate }) - .orWhere('retentionUntil < NOW()') - .execute(); - - const deletedCount = result.affected || 0; - this.logger.log(`Applied retention policy: deleted ${deletedCount} old audit logs`); - return deletedCount; - } - - /** - * Export logs to JSON - */ - async exportToJson(filters: AuditLogSearchFilters): Promise { - const { logs } = await this.search(filters, 1, 10000); - return JSON.stringify(logs, null, 2); - } - - /** - * Export logs to CSV - */ - async exportToCsv(filters: AuditLogSearchFilters): Promise { - const { logs } = await this.search(filters, 1, 10000); - - const headers = [ - 'timestamp', - 'userId', - 'userEmail', - 'action', - 'category', - 'severity', - 'entityType', - 'entityId', - 'description', - 'ipAddress', - 'userAgent', - 'apiEndpoint', - 'httpMethod', - 'statusCode', - ]; - - const rows = logs.map((log) => [ - log.timestamp.toISOString(), - log.userId || '', - log.userEmail || '', - log.action, - log.category, - log.severity, - log.entityType || '', - log.entityId || '', - log.description || '', - log.ipAddress || '', - log.userAgent || '', - log.apiEndpoint || '', - log.httpMethod || '', - log.statusCode || '', - ]); - - const escapeCsv = (value: string | number) => { - const str = String(value); - if (str.includes(',') || str.includes('"') || str.includes('\n')) { - return `"${str.replace(/"/g, '""')}"`; - } - return str; - }; - - const csvContent = [headers.join(','), ...rows.map((row) => row.map(escapeCsv).join(','))].join( - '\n', - ); - - return csvContent; - } - - /** - * Get statistics - */ - async getStatistics(): Promise<{ - totalLogs: number; - logsToday: number; - logsThisWeek: number; - logsThisMonth: number; - criticalEvents: number; - errorEvents: number; - }> { - const now = new Date(); - const today = new Date(now.getFullYear(), now.getMonth(), now.getDate()); - const weekAgo = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000); - const monthAgo = new Date(now.getTime() - 30 * 24 * 60 * 60 * 1000); - - const [totalLogs, logsToday, logsThisWeek, logsThisMonth, criticalEvents, errorEvents] = - await Promise.all([ - this.auditRepo.count(), - this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(today) } }), - this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(weekAgo) } }), - this.auditRepo.count({ where: { timestamp: MoreThanOrEqual(monthAgo) } }), - this.auditRepo.count({ where: { severity: AuditSeverity.CRITICAL } }), - this.auditRepo.count({ where: { severity: AuditSeverity.ERROR } }), - ]); - - return { - totalLogs, - logsToday, - logsThisWeek, - logsThisMonth, - criticalEvents, - errorEvents, - }; - } } - // Helper function for date comparison function MoreThanOrEqual(date: Date) { - return MoreThan(date); + return MoreThan(date); } diff --git a/src/audit-log/decorators/audit.decorator.ts b/src/audit-log/decorators/audit.decorator.ts index 0c3c2063..76cff171 100644 --- a/src/audit-log/decorators/audit.decorator.ts +++ b/src/audit-log/decorators/audit.decorator.ts @@ -1,96 +1,67 @@ import { SetMetadata } from '@nestjs/common'; import { AuditAction, AuditCategory, AuditSeverity } from '../enums/audit-action.enum'; - export const AUDIT_LOG_KEY = 'audit_log'; - export interface AuditLogOptions { - action: AuditAction; - category?: AuditCategory; - severity?: AuditSeverity; - entityType?: string; - entityIdParam?: string; - description?: string; - logRequestBody?: boolean; - logResponseBody?: boolean; - sensitiveFields?: string[]; + action: AuditAction; + category?: AuditCategory; + severity?: AuditSeverity; + entityType?: string; + entityIdParam?: string; + description?: string; + logRequestBody?: boolean; + logResponseBody?: boolean; + sensitiveFields?: string[]; } - /** * Decorator to mark a method for audit logging * @param options Audit log configuration options */ export const AuditLog = (options: AuditLogOptions) => SetMetadata(AUDIT_LOG_KEY, options); - /** * Predefined audit log decorators for common operations */ -export const AuditCreate = (entityType: string, options?: Partial) => - AuditLog({ +export const AuditCreate = (entityType: string, options?: Partial) => AuditLog({ action: AuditAction.DATA_CREATED, category: AuditCategory.DATA_MODIFICATION, entityType, ...options, - }); - -export const AuditUpdate = ( - entityType: string, - entityIdParam: string, - options?: Partial, -) => - AuditLog({ +}); +export const AuditUpdate = (entityType: string, entityIdParam: string, options?: Partial) => AuditLog({ action: AuditAction.DATA_UPDATED, category: AuditCategory.DATA_MODIFICATION, entityType, entityIdParam, ...options, - }); - -export const AuditDelete = ( - entityType: string, - entityIdParam: string, - options?: Partial, -) => - AuditLog({ +}); +export const AuditDelete = (entityType: string, entityIdParam: string, options?: Partial) => AuditLog({ action: AuditAction.DATA_DELETED, category: AuditCategory.DATA_MODIFICATION, severity: AuditSeverity.WARNING, entityType, entityIdParam, ...options, - }); - -export const AuditView = ( - entityType: string, - entityIdParam?: string, - options?: Partial, -) => - AuditLog({ +}); +export const AuditView = (entityType: string, entityIdParam?: string, options?: Partial) => AuditLog({ action: AuditAction.DATA_VIEWED, category: AuditCategory.DATA_ACCESS, entityType, entityIdParam, ...options, - }); - -export const AuditExport = (entityType: string, options?: Partial) => - AuditLog({ +}); +export const AuditExport = (entityType: string, options?: Partial) => AuditLog({ action: AuditAction.DATA_EXPORTED, category: AuditCategory.DATA_ACCESS, severity: AuditSeverity.WARNING, entityType, ...options, - }); - -export const AuditLogin = (options?: Partial) => - AuditLog({ +}); +export const AuditLogin = (options?: Partial) => AuditLog({ action: AuditAction.LOGIN, category: AuditCategory.AUTHENTICATION, ...options, - }); - -export const AuditLogout = (options?: Partial) => - AuditLog({ +}); +export const AuditLogout = (options?: Partial) => AuditLog({ action: AuditAction.LOGOUT, category: AuditCategory.AUTHENTICATION, ...options, - }); +}); diff --git a/src/audit-log/enums/audit-action.enum.ts b/src/audit-log/enums/audit-action.enum.ts index d6d0d63a..0cfc5393 100644 --- a/src/audit-log/enums/audit-action.enum.ts +++ b/src/audit-log/enums/audit-action.enum.ts @@ -1,76 +1,67 @@ export enum AuditAction { - // Authentication events - LOGIN = 'LOGIN', - LOGIN_FAILED = 'LOGIN_FAILED', - LOGOUT = 'LOGOUT', - REGISTER = 'REGISTER', - PASSWORD_RESET_REQUEST = 'PASSWORD_RESET_REQUEST', - PASSWORD_RESET = 'PASSWORD_RESET', - PASSWORD_CHANGE = 'PASSWORD_CHANGE', - EMAIL_VERIFIED = 'EMAIL_VERIFIED', - TOKEN_REFRESH = 'TOKEN_REFRESH', - SESSION_EXPIRED = 'SESSION_EXPIRED', - SESSION_REVOKED = 'SESSION_REVOKED', - - // User management - USER_CREATED = 'USER_CREATED', - USER_UPDATED = 'USER_UPDATED', - USER_DELETED = 'USER_DELETED', - USER_ROLE_CHANGED = 'USER_ROLE_CHANGED', - USER_STATUS_CHANGED = 'USER_STATUS_CHANGED', - - // Data access - DATA_VIEWED = 'DATA_VIEWED', - DATA_CREATED = 'DATA_CREATED', - DATA_UPDATED = 'DATA_UPDATED', - DATA_DELETED = 'DATA_DELETED', - DATA_EXPORTED = 'DATA_EXPORTED', - DATA_IMPORTED = 'DATA_IMPORTED', - - // File operations - FILE_UPLOADED = 'FILE_UPLOADED', - FILE_DOWNLOADED = 'FILE_DOWNLOADED', - FILE_DELETED = 'FILE_DELETED', - FILE_SHARED = 'FILE_SHARED', - - // API access - API_CALLED = 'API_CALLED', - API_RATE_LIMITED = 'API_RATE_LIMITED', - API_ERROR = 'API_ERROR', - - // Security events - PERMISSION_DENIED = 'PERMISSION_DENIED', - SUSPICIOUS_ACTIVITY = 'SUSPICIOUS_ACTIVITY', - MFA_ENABLED = 'MFA_ENABLED', - MFA_DISABLED = 'MFA_DISABLED', - MFA_FAILED = 'MFA_FAILED', - - // Admin operations - CONFIG_CHANGED = 'CONFIG_CHANGED', - SETTING_UPDATED = 'SETTING_UPDATED', - BACKUP_CREATED = 'BACKUP_CREATED', - BACKUP_RESTORED = 'BACKUP_RESTORED', - - // Compliance - DATA_RETENTION_APPLIED = 'DATA_RETENTION_APPLIED', - AUDIT_LOG_EXPORTED = 'AUDIT_LOG_EXPORTED', - REPORT_GENERATED = 'REPORT_GENERATED', + // Authentication events + LOGIN = 'LOGIN', + LOGIN_FAILED = 'LOGIN_FAILED', + LOGOUT = 'LOGOUT', + REGISTER = 'REGISTER', + PASSWORD_RESET_REQUEST = 'PASSWORD_RESET_REQUEST', + PASSWORD_RESET = 'PASSWORD_RESET', + PASSWORD_CHANGE = 'PASSWORD_CHANGE', + EMAIL_VERIFIED = 'EMAIL_VERIFIED', + TOKEN_REFRESH = 'TOKEN_REFRESH', + SESSION_EXPIRED = 'SESSION_EXPIRED', + SESSION_REVOKED = 'SESSION_REVOKED', + // User management + USER_CREATED = 'USER_CREATED', + USER_UPDATED = 'USER_UPDATED', + USER_DELETED = 'USER_DELETED', + USER_ROLE_CHANGED = 'USER_ROLE_CHANGED', + USER_STATUS_CHANGED = 'USER_STATUS_CHANGED', + // Data access + DATA_VIEWED = 'DATA_VIEWED', + DATA_CREATED = 'DATA_CREATED', + DATA_UPDATED = 'DATA_UPDATED', + DATA_DELETED = 'DATA_DELETED', + DATA_EXPORTED = 'DATA_EXPORTED', + DATA_IMPORTED = 'DATA_IMPORTED', + // File operations + FILE_UPLOADED = 'FILE_UPLOADED', + FILE_DOWNLOADED = 'FILE_DOWNLOADED', + FILE_DELETED = 'FILE_DELETED', + FILE_SHARED = 'FILE_SHARED', + // API access + API_CALLED = 'API_CALLED', + API_RATE_LIMITED = 'API_RATE_LIMITED', + API_ERROR = 'API_ERROR', + // Security events + PERMISSION_DENIED = 'PERMISSION_DENIED', + SUSPICIOUS_ACTIVITY = 'SUSPICIOUS_ACTIVITY', + MFA_ENABLED = 'MFA_ENABLED', + MFA_DISABLED = 'MFA_DISABLED', + MFA_FAILED = 'MFA_FAILED', + // Admin operations + CONFIG_CHANGED = 'CONFIG_CHANGED', + SETTING_UPDATED = 'SETTING_UPDATED', + BACKUP_CREATED = 'BACKUP_CREATED', + BACKUP_RESTORED = 'BACKUP_RESTORED', + // Compliance + DATA_RETENTION_APPLIED = 'DATA_RETENTION_APPLIED', + AUDIT_LOG_EXPORTED = 'AUDIT_LOG_EXPORTED', + REPORT_GENERATED = 'REPORT_GENERATED' } - export enum AuditSeverity { - INFO = 'INFO', - WARNING = 'WARNING', - ERROR = 'ERROR', - CRITICAL = 'CRITICAL', + INFO = 'INFO', + WARNING = 'WARNING', + ERROR = 'ERROR', + CRITICAL = 'CRITICAL' } - export enum AuditCategory { - AUTHENTICATION = 'AUTHENTICATION', - AUTHORIZATION = 'AUTHORIZATION', - DATA_ACCESS = 'DATA_ACCESS', - DATA_MODIFICATION = 'DATA_MODIFICATION', - FILE_OPERATION = 'FILE_OPERATION', - SYSTEM = 'SYSTEM', - SECURITY = 'SECURITY', - COMPLIANCE = 'COMPLIANCE', + AUTHENTICATION = 'AUTHENTICATION', + AUTHORIZATION = 'AUTHORIZATION', + DATA_ACCESS = 'DATA_ACCESS', + DATA_MODIFICATION = 'DATA_MODIFICATION', + FILE_OPERATION = 'FILE_OPERATION', + SYSTEM = 'SYSTEM', + SECURITY = 'SECURITY', + COMPLIANCE = 'COMPLIANCE' } diff --git a/src/audit-log/interceptors/audit-log.interceptor.ts b/src/audit-log/interceptors/audit-log.interceptor.ts index 388ffcd3..3157aaf5 100644 --- a/src/audit-log/interceptors/audit-log.interceptor.ts +++ b/src/audit-log/interceptors/audit-log.interceptor.ts @@ -4,119 +4,73 @@ import { tap } from 'rxjs/operators'; import { AuditLogService } from '../audit-log.service'; import { AuditAction, AuditCategory, AuditSeverity } from '../enums/audit-action.enum'; import { Request } from 'express'; - interface RequestWithUser extends Request { - user?: { - id: string; - email: string; - role?: string; - }; - requestId?: string; + user?: { + id: string; + email: string; + role?: string; + }; + requestId?: string; } - @Injectable() export class AuditLogInterceptor implements NestInterceptor { - private readonly logger = new Logger(AuditLogInterceptor.name); - - constructor(private readonly auditLogService: AuditLogService) {} - - intercept(context: ExecutionContext, next: CallHandler): Observable { - const request = context.switchToHttp().getRequest(); - const response = context.switchToHttp().getResponse(); - const startTime = Date.now(); - - const { method, path, ip, headers, user, requestId } = request; - - const userAgent = headers['user-agent'] || 'Unknown'; - const userId = user?.id || null; - const userEmail = user?.email || null; - - return next.handle().pipe( - tap({ - next: () => { - this.logRequest( - userId, - userEmail, - path, - method, - response.statusCode, - Date.now() - startTime, - ip, - userAgent, - requestId, - ); - }, - error: (error) => { - const statusCode = error.status || 500; - this.logRequest( - userId, - userEmail, - path, - method, - statusCode, - Date.now() - startTime, - ip, - userAgent, - requestId, - error.message, - ); - }, - }), - ); - } - - private async logRequest( - userId: string | null, - userEmail: string | null, - apiEndpoint: string, - httpMethod: string, - statusCode: number, - responseTimeMs: number, - ipAddress: string, - userAgent: string, - requestId?: string, - errorMessage?: string, - ): Promise { - try { - // Skip logging for health checks and static assets - if (this.shouldSkipLogging(apiEndpoint)) { - return; - } - - const severity = - statusCode >= 500 - ? AuditSeverity.ERROR - : statusCode >= 400 - ? AuditSeverity.WARNING - : AuditSeverity.INFO; - - await this.auditLogService.log({ - userId: userId || undefined, - userEmail: userEmail || undefined, - action: AuditAction.API_CALLED, - category: AuditCategory.DATA_ACCESS, - severity, - apiEndpoint, - httpMethod, - statusCode, - responseTimeMs, - ipAddress, - userAgent, - requestId, - description: errorMessage || `${httpMethod} ${apiEndpoint} - ${statusCode}`, - }); - } catch (error) { - this.logger.error('Failed to log audit entry:', error); + private readonly logger = new Logger(AuditLogInterceptor.name); + constructor(private readonly auditLogService: AuditLogService) { } + intercept(context: ExecutionContext, next: CallHandler): Observable { + const request = context.switchToHttp().getRequest(); + const response = context.switchToHttp().getResponse(); + const startTime = Date.now(); + const { method, path, ip, headers, user, requestId } = request; + const userAgent = headers['user-agent'] || 'Unknown'; + const userId = user?.id || null; + const userEmail = user?.email || null; + return next.handle().pipe(tap({ + next: () => { + this.logRequest(userId, userEmail, path, method, response.statusCode, Date.now() - startTime, ip, userAgent, requestId); + }, + error: (error) => { + const statusCode = error.status || 500; + this.logRequest(userId, userEmail, path, method, statusCode, Date.now() - startTime, ip, userAgent, requestId, error.message); + }, + })); + } + private async logRequest(userId: string | null, userEmail: string | null, apiEndpoint: string, httpMethod: string, statusCode: number, responseTimeMs: number, ipAddress: string, userAgent: string, requestId?: string, errorMessage?: string): Promise { + try { + // Skip logging for health checks and static assets + if (this.shouldSkipLogging(apiEndpoint)) { + return; + } + const severity = statusCode >= 500 + ? AuditSeverity.ERROR + : statusCode >= 400 + ? AuditSeverity.WARNING + : AuditSeverity.INFO; + await this.auditLogService.log({ + userId: userId || undefined, + userEmail: userEmail || undefined, + action: AuditAction.API_CALLED, + category: AuditCategory.DATA_ACCESS, + severity, + apiEndpoint, + httpMethod, + statusCode, + responseTimeMs, + ipAddress, + userAgent, + requestId, + description: errorMessage || `${httpMethod} ${apiEndpoint} - ${statusCode}`, + }); + } + catch (error) { + this.logger.error('Failed to log audit entry:', error); + } + } + private shouldSkipLogging(endpoint: string): boolean { + const skipPatterns = [ + /^\/health/, + /^\/favicon/, + /\.(css|js|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$/, + ]; + return skipPatterns.some((pattern) => pattern.test(endpoint)); } - } - - private shouldSkipLogging(endpoint: string): boolean { - const skipPatterns = [ - /^\/health/, - /^\/favicon/, - /\.(css|js|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$/, - ]; - - return skipPatterns.some((pattern) => pattern.test(endpoint)); - } } diff --git a/src/audit-log/tasks/audit-retention.task.ts b/src/audit-log/tasks/audit-retention.task.ts index 5b73e583..a7aa509e 100644 --- a/src/audit-log/tasks/audit-retention.task.ts +++ b/src/audit-log/tasks/audit-retention.task.ts @@ -1,50 +1,45 @@ import { Injectable, Logger } from '@nestjs/common'; import { Cron, CronExpression } from '@nestjs/schedule'; import { AuditLogService } from '../audit-log.service'; - @Injectable() export class AuditRetentionTask { - private readonly logger = new Logger(AuditRetentionTask.name); - - constructor(private readonly auditLogService: AuditLogService) {} - - /** - * Run retention policy daily at 2 AM - */ - @Cron(CronExpression.EVERY_DAY_AT_2AM) - async handleDailyRetention(): Promise { - this.logger.log('Starting daily audit log retention policy...'); - try { - const deletedCount = await this.auditLogService.applyRetentionPolicy(); - this.logger.log(`Daily retention policy completed. Deleted ${deletedCount} old audit logs.`); - } catch (error) { - this.logger.error('Failed to apply retention policy:', error); + private readonly logger = new Logger(AuditRetentionTask.name); + constructor(private readonly auditLogService: AuditLogService) { } + /** + * Run retention policy daily at 2 AM + */ + @Cron(CronExpression.EVERY_DAY_AT_2AM) + async handleDailyRetention(): Promise { + this.logger.log('Starting daily audit log retention policy...'); + try { + const deletedCount = await this.auditLogService.applyRetentionPolicy(); + this.logger.log(`Daily retention policy completed. Deleted ${deletedCount} old audit logs.`); + } + catch (error) { + this.logger.error('Failed to apply retention policy:', error); + } } - } - - /** - * Generate weekly report every Monday at 3 AM - */ - @Cron('0 3 * * 1') // Every Monday at 3 AM - async handleWeeklyReport(): Promise { - this.logger.log('Generating weekly audit report...'); - try { - const endDate = new Date(); - const startDate = new Date(); - startDate.setDate(startDate.getDate() - 7); - - const report = await this.auditLogService.generateReport(startDate, endDate); - - this.logger.log('Weekly report generated:', { - totalEvents: report.totalEvents, - criticalEvents: report.eventsBySeverity['CRITICAL'] || 0, - errorEvents: report.eventsBySeverity['ERROR'] || 0, - }); - - // In a real implementation, you might send this report via email - // or store it for compliance purposes - } catch (error) { - this.logger.error('Failed to generate weekly report:', error); + /** + * Generate weekly report every Monday at 3 AM + */ + @Cron('0 3 * * 1') // Every Monday at 3 AM + async handleWeeklyReport(): Promise { + this.logger.log('Generating weekly audit report...'); + try { + const endDate = new Date(); + const startDate = new Date(); + startDate.setDate(startDate.getDate() - 7); + const report = await this.auditLogService.generateReport(startDate, endDate); + this.logger.log('Weekly report generated:', { + totalEvents: report.totalEvents, + criticalEvents: report.eventsBySeverity['CRITICAL'] || 0, + errorEvents: report.eventsBySeverity['ERROR'] || 0, + }); + // In a real implementation, you might send this report via email + // or store it for compliance purposes + } + catch (error) { + this.logger.error('Failed to generate weekly report:', error); + } } - } } diff --git a/src/audit-log/tests/audit-log.test.ts b/src/audit-log/tests/audit-log.test.ts index 91b58ebb..fab3fc62 100644 --- a/src/audit-log/tests/audit-log.test.ts +++ b/src/audit-log/tests/audit-log.test.ts @@ -2,22 +2,19 @@ import { AuditLogService } from '../audit-log.service'; import { Repository } from 'typeorm'; import { AuditLog } from '../audit-log.entity'; import { ConfigService } from '@nestjs/config'; - describe('AuditLogService', () => { - let service: AuditLogService; - let repo: Repository; - let configService: ConfigService; - - beforeEach(() => { - // Mock repository and config service - repo = {} as Repository; - configService = { - get: jest.fn().mockReturnValue(365), - } as any; - service = new AuditLogService(repo as any, configService); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); + let service: AuditLogService; + let repo: Repository; + let configService: ConfigService; + beforeEach(() => { + // Mock repository and config service + repo = {} as Repository; + configService = { + get: jest.fn().mockReturnValue(365), + } as unknown; + service = new AuditLogService(repo as unknown, configService); + }); + it('should be defined', () => { + expect(service).toBeDefined(); + }); }); diff --git a/src/auth/auth.controller.ts b/src/auth/auth.controller.ts index aa5aa5fa..5f49dd05 100644 --- a/src/auth/auth.controller.ts +++ b/src/auth/auth.controller.ts @@ -3,90 +3,95 @@ import { ApiBearerAuth, ApiOperation, ApiTags } from '@nestjs/swagger'; import { Throttle } from '@nestjs/throttler'; import { Request } from 'express'; import { AuthService } from './auth.service'; -import { - RegisterDto, - LoginDto, - RefreshTokenDto, - ForgotPasswordDto, - ResetPasswordDto, - ChangePasswordDto, - VerifyEmailDto, -} from './dto/auth.dto'; +import { RegisterDto, LoginDto, RefreshTokenDto, ForgotPasswordDto, ResetPasswordDto, ChangePasswordDto, VerifyEmailDto, } from './dto/auth.dto'; import { JwtAuthGuard } from './guards/jwt-auth.guard'; import { CurrentUser } from './decorators/current-user.decorator'; - @ApiTags('auth') @Controller('auth') export class AuthController { - constructor(private readonly authService: AuthService) {} - - @Post('register') - @Throttle({ default: { limit: 3, ttl: 3600000 } }) // 3 requests per hour - @ApiOperation({ summary: 'Register a new user' }) - async register(@Body() registerDto: RegisterDto, @Req() req: Request): Promise { - const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; - const userAgent = req.headers['user-agent'] || 'unknown'; - return this.authService.register(registerDto, ipAddress, userAgent); - } - - @Post('login') - @Throttle({ default: { limit: 5, ttl: 900000 } }) // 5 requests per 15 minutes - @ApiOperation({ summary: 'Login user and get tokens' }) - async login(@Body() loginDto: LoginDto, @Req() req: Request): Promise { - const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; - const userAgent = req.headers['user-agent'] || 'unknown'; - return this.authService.login(loginDto, ipAddress, userAgent); - } - - @Post('refresh') - @Throttle({ default: { limit: 20, ttl: 60000 } }) // 20 requests per minute - @ApiOperation({ summary: 'Refresh access token using refresh token' }) - async refresh(@Body() refreshTokenDto: RefreshTokenDto): Promise { - return this.authService.refreshToken(refreshTokenDto.refreshToken); - } - - @Post('logout') - @UseGuards(JwtAuthGuard) - @ApiBearerAuth() - @ApiOperation({ summary: 'Logout user (invalidate refresh token)' }) - async logout(@CurrentUser() user: any, @Req() req: Request): Promise { - const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; - const userAgent = req.headers['user-agent'] || 'unknown'; - return this.authService.logout(user.userId, user.sessionId, ipAddress, userAgent); - } - - @Post('forgot-password') - @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour - @ApiOperation({ summary: 'Request a password reset link' }) - async forgotPassword(@Body() forgotPasswordDto: ForgotPasswordDto): Promise { - return this.authService.forgotPassword(forgotPasswordDto.email); - } - - @Post('reset-password') - @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour - @ApiOperation({ summary: 'Reset password using token' }) - async resetPassword(@Body() resetPasswordDto: ResetPasswordDto): Promise { - return this.authService.resetPassword(resetPasswordDto); - } - - @Post('change-password') - @UseGuards(JwtAuthGuard) - @ApiBearerAuth() - @ApiOperation({ summary: 'Change password for authenticated user' }) - async changePassword( - @CurrentUser() user: any, - @Body() changePasswordDto: ChangePasswordDto, - @Req() req: Request, - ): Promise { - const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; - const userAgent = req.headers['user-agent'] || 'unknown'; - return this.authService.changePassword(user.userId, changePasswordDto, ipAddress, userAgent); - } - - @Post('verify-email') - @Throttle({ default: { limit: 10, ttl: 3600000 } }) // 10 requests per hour - @ApiOperation({ summary: 'Verify email using token' }) - async verifyEmail(@Body() verifyEmailDto: VerifyEmailDto): Promise { - return this.authService.verifyEmail(verifyEmailDto.token); - } + constructor(private readonly authService: AuthService) { } + @Post('register') + @Throttle({ default: { limit: 3, ttl: 3600000 } }) // 3 requests per hour + @ApiOperation({ summary: 'Register a new user' }) + async register( + @Body() + registerDto: RegisterDto, + @Req() + req: Request): Promise { + const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; + const userAgent = req.headers['user-agent'] || 'unknown'; + return this.authService.register(registerDto, ipAddress, userAgent); + } + @Post('login') + @Throttle({ default: { limit: 5, ttl: 900000 } }) // 5 requests per 15 minutes + @ApiOperation({ summary: 'Login user and get tokens' }) + async login( + @Body() + loginDto: LoginDto, + @Req() + req: Request): Promise { + const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; + const userAgent = req.headers['user-agent'] || 'unknown'; + return this.authService.login(loginDto, ipAddress, userAgent); + } + @Post('refresh') + @Throttle({ default: { limit: 20, ttl: 60000 } }) // 20 requests per minute + @ApiOperation({ summary: 'Refresh access token using refresh token' }) + async refresh( + @Body() + refreshTokenDto: RefreshTokenDto): Promise { + return this.authService.refreshToken(refreshTokenDto.refreshToken); + } + @Post('logout') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: 'Logout user (invalidate refresh token)' }) + async logout( + @CurrentUser() + user: unknown, + @Req() + req: Request): Promise { + const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; + const userAgent = req.headers['user-agent'] || 'unknown'; + return this.authService.logout(user.userId, user.sessionId, ipAddress, userAgent); + } + @Post('forgot-password') + @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour + @ApiOperation({ summary: 'Request a password reset link' }) + async forgotPassword( + @Body() + forgotPasswordDto: ForgotPasswordDto): Promise { + return this.authService.forgotPassword(forgotPasswordDto.email); + } + @Post('reset-password') + @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour + @ApiOperation({ summary: 'Reset password using token' }) + async resetPassword( + @Body() + resetPasswordDto: ResetPasswordDto): Promise { + return this.authService.resetPassword(resetPasswordDto); + } + @Post('change-password') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: 'Change password for authenticated user' }) + async changePassword( + @CurrentUser() + user: unknown, + @Body() + changePasswordDto: ChangePasswordDto, + @Req() + req: Request): Promise { + const ipAddress = req.ip || req.socket.remoteAddress || 'unknown'; + const userAgent = req.headers['user-agent'] || 'unknown'; + return this.authService.changePassword(user.userId, changePasswordDto, ipAddress, userAgent); + } + @Post('verify-email') + @Throttle({ default: { limit: 10, ttl: 3600000 } }) // 10 requests per hour + @ApiOperation({ summary: 'Verify email using token' }) + async verifyEmail( + @Body() + verifyEmailDto: VerifyEmailDto): Promise { + return this.authService.verifyEmail(verifyEmailDto.token); + } } diff --git a/src/auth/auth.module.ts b/src/auth/auth.module.ts index e5d2edbf..bf17cd07 100644 --- a/src/auth/auth.module.ts +++ b/src/auth/auth.module.ts @@ -10,66 +10,68 @@ import { SessionModule } from '../session/session.module'; import { TransactionService } from '../common/database/transaction.service'; import { NotificationsModule } from '../notifications/notifications.module'; import { AuditLogModule } from '../audit-log/audit-log.module'; - function parseJwtSecrets(raw: string): Record { - try { - const parsed = JSON.parse(raw) as unknown; - if (parsed && typeof parsed === 'object') { - return parsed as Record; + try { + const parsed = JSON.parse(raw) as unknown; + if (parsed && typeof parsed === 'object') { + return parsed as Record; + } } - } catch { - // ignore - } - - return raw - .split(',') - .map((pair) => pair.trim()) - .filter(Boolean) - .reduce>((acc, pair) => { - const idx = pair.indexOf(':'); - if (idx <= 0) return acc; - const version = pair.slice(0, idx).trim(); - const secret = pair.slice(idx + 1).trim(); - if (!version || !secret) return acc; - acc[version] = secret; - return acc; + catch { + // ignore + } + return raw + .split(',') + .map((pair) => pair.trim()) + .filter(Boolean) + .reduce>((acc, pair) => { + const idx = pair.indexOf(':'); + if (idx <= 0) + return acc; + const version = pair.slice(0, idx).trim(); + const secret = pair.slice(idx + 1).trim(); + if (!version || !secret) + return acc; + acc[version] = secret; + return acc; }, {}); } - function getCurrentJwtAccessSecret(configService: ConfigService): string { - const jwtSecretsRaw = configService.get('JWT_SECRETS'); - if (!jwtSecretsRaw) return configService.get('JWT_SECRET') ?? 'your-secret-key'; - - const currentVersion = configService.get('JWT_SECRET_CURRENT_VERSION'); - const secrets = parseJwtSecrets(jwtSecretsRaw); - const current = - (currentVersion && secrets[currentVersion]) || configService.get('JWT_SECRET'); - return current || Object.values(secrets)[0] || 'your-secret-key'; + const jwtSecretsRaw = configService.get('JWT_SECRETS'); + if (!jwtSecretsRaw) + return configService.get('JWT_SECRET') ?? 'your-secret-key'; + const currentVersion = configService.get('JWT_SECRET_CURRENT_VERSION'); + const secrets = parseJwtSecrets(jwtSecretsRaw); + const current = (currentVersion && secrets[currentVersion]) || configService.get('JWT_SECRET'); + return current || Object.values(secrets)[0] || 'your-secret-key'; } - @Module({ - imports: [ - ConfigModule, - UsersModule, - SessionModule, - NotificationsModule, - AuditLogModule, - PassportModule, - JwtModule.registerAsync({ - imports: [ConfigModule], - inject: [ConfigService], - useFactory: async ( - configService: ConfigService, - ): Promise<{ secret: string; signOptions: { expiresIn: number } }> => ({ - secret: getCurrentJwtAccessSecret(configService), - signOptions: { - expiresIn: parseInt(configService.get('JWT_EXPIRES_IN') ?? '900', 10), // Convert to seconds (number) - }, - }), - }), - ], - controllers: [AuthController], - providers: [AuthService, JwtStrategy, TransactionService], - exports: [AuthService], + imports: [ + ConfigModule, + UsersModule, + SessionModule, + NotificationsModule, + AuditLogModule, + PassportModule, + JwtModule.registerAsync({ + imports: [ConfigModule], + inject: [ConfigService], + useFactory: async (configService: ConfigService): Promise<{ + secret: string; + signOptions: { + expiresIn: number; + }; + }> => ({ + secret: getCurrentJwtAccessSecret(configService), + signOptions: { + expiresIn: parseInt(configService.get('JWT_EXPIRES_IN') ?? '900', 10), // Convert to seconds (number) + }, + }), + }), + ], + controllers: [AuthController], + providers: [AuthService, JwtStrategy, TransactionService], + exports: [AuthService], }) -export class AuthModule {} +export class AuthModule { +} diff --git a/src/auth/auth.service.spec.ts b/src/auth/auth.service.spec.ts index 1e52b11a..ad49978b 100644 --- a/src/auth/auth.service.spec.ts +++ b/src/auth/auth.service.spec.ts @@ -10,519 +10,375 @@ import { NotificationsService } from '../notifications/notifications.service'; import { AuditLogService } from '../audit-log/audit-log.service'; import { UserRole } from '../users/entities/user.entity'; import { AuditAction, AuditSeverity } from '../audit-log/enums/audit-action.enum'; -import { - createMockRepository, - createMockConfigService, - createMockEventEmitter, -} from 'test/utils/mock-factories'; +import { createMockRepository, createMockConfigService, createMockEventEmitter, } from 'test/utils/mock-factories'; import { Repository } from 'typeorm'; - describe('AuthService', () => { - // ───────────────────────────────────────────────────────────────────────── - // DECLARATIONS - // ───────────────────────────────────────────────────────────────────────── - - let service: AuthService; - let mockUsersService: jest.Mocked; - let mockJwtService: jest.Mocked; - let mockConfigService: jest.Mocked; - let mockSessionService: jest.Mocked; - let mockTransactionService: jest.Mocked; - let mockNotificationsService: jest.Mocked; - let mockAuditLogService: jest.Mocked; - - // ───────────────────────────────────────────────────────────────────────── - // SETUP & TEARDOWN - // ───────────────────────────────────────────────────────────────────────── - - beforeEach(async () => { - // Initialize all dependency mocks - mockUsersService = { - create: jest.fn(), - findByEmail: jest.fn(), - findOne: jest.fn(), - updateEmailVerificationToken: jest.fn(), - updateRefreshToken: jest.fn(), - updateLastLogin: jest.fn(), - updatePasswordResetToken: jest.fn(), - } as jest.Mocked; - - mockJwtService = { - sign: jest.fn(), - verify: jest.fn(), - } as jest.Mocked; - - mockConfigService = createMockConfigService({ - JWT_ACCESS_SECRET: 'access-secret', - JWT_REFRESH_SECRET: 'refresh-secret', - JWT_ACCESS_EXPIRES_IN: '15m', - JWT_REFRESH_EXPIRES_IN: '7d', + // ───────────────────────────────────────────────────────────────────────── + // DECLARATIONS + // ───────────────────────────────────────────────────────────────────────── + let service: AuthService; + let mockUsersService: jest.Mocked; + let mockJwtService: jest.Mocked; + let mockConfigService: jest.Mocked; + let mockSessionService: jest.Mocked; + let mockTransactionService: jest.Mocked; + let mockNotificationsService: jest.Mocked; + let mockAuditLogService: jest.Mocked; + // ───────────────────────────────────────────────────────────────────────── + // SETUP & TEARDOWN + // ───────────────────────────────────────────────────────────────────────── + beforeEach(async () => { + // Initialize all dependency mocks + mockUsersService = { + create: jest.fn(), + findByEmail: jest.fn(), + findOne: jest.fn(), + updateEmailVerificationToken: jest.fn(), + updateRefreshToken: jest.fn(), + updateLastLogin: jest.fn(), + updatePasswordResetToken: jest.fn(), + } as jest.Mocked; + mockJwtService = { + sign: jest.fn(), + verify: jest.fn(), + } as jest.Mocked; + mockConfigService = createMockConfigService({ + JWT_ACCESS_SECRET: 'access-secret', + JWT_REFRESH_SECRET: 'refresh-secret', + JWT_ACCESS_EXPIRES_IN: '15m', + JWT_REFRESH_EXPIRES_IN: '7d', + }); + mockSessionService = { + createSession: jest.fn(), + getSession: jest.fn(), + removeSession: jest.fn(), + touchSession: jest.fn(), + withLock: jest.fn(), + } as jest.Mocked; + mockTransactionService = { + runInTransaction: jest.fn(), + } as jest.Mocked; + mockNotificationsService = { + sendVerificationEmail: jest.fn(), + } as jest.Mocked; + mockAuditLogService = { + logAuth: jest.fn(), + } as jest.Mocked; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + AuthService, + { + provide: UsersService, + useValue: mockUsersService, + }, + { + provide: JwtService, + useValue: mockJwtService, + }, + { + provide: ConfigService, + useValue: mockConfigService, + }, + { + provide: SessionService, + useValue: mockSessionService, + }, + { + provide: TransactionService, + useValue: mockTransactionService, + }, + { + provide: NotificationsService, + useValue: mockNotificationsService, + }, + { + provide: AuditLogService, + useValue: mockAuditLogService, + }, + ], + }).compile(); + service = module.get(AuthService); }); - - mockSessionService = { - createSession: jest.fn(), - getSession: jest.fn(), - removeSession: jest.fn(), - touchSession: jest.fn(), - withLock: jest.fn(), - } as jest.Mocked; - - mockTransactionService = { - runInTransaction: jest.fn(), - } as jest.Mocked; - - mockNotificationsService = { - sendVerificationEmail: jest.fn(), - } as jest.Mocked; - - mockAuditLogService = { - logAuth: jest.fn(), - } as jest.Mocked; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - AuthService, - { - provide: UsersService, - useValue: mockUsersService, - }, - { - provide: JwtService, - useValue: mockJwtService, - }, - { - provide: ConfigService, - useValue: mockConfigService, - }, - { - provide: SessionService, - useValue: mockSessionService, - }, - { - provide: TransactionService, - useValue: mockTransactionService, - }, - { - provide: NotificationsService, - useValue: mockNotificationsService, - }, - { - provide: AuditLogService, - useValue: mockAuditLogService, - }, - ], - }).compile(); - - service = module.get(AuthService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - // ───────────────────────────────────────────────────────────────────────── - // TEST SUITES - // ───────────────────────────────────────────────────────────────────────── - - describe('register', () => { - const registerDto = { - email: 'test@example.com', - password: 'Password123!', - firstName: 'John', - lastName: 'Doe', - }; - - const mockUser = { - id: 'user-1', - email: registerDto.email, - firstName: registerDto.firstName, - lastName: registerDto.lastName, - role: UserRole.STUDENT, - isEmailVerified: false, - }; - - beforeEach(() => { - mockTransactionService.runInTransaction.mockImplementation(async (fn) => fn()); - mockUsersService.create.mockResolvedValue(mockUser); - mockUsersService.updateEmailVerificationToken.mockResolvedValue(undefined); - mockUsersService.updateRefreshToken.mockResolvedValue(undefined); - mockSessionService.createSession.mockResolvedValue('session-1'); - mockJwtService.sign - .mockReturnValueOnce('access-token') - .mockReturnValueOnce('refresh-token'); - mockNotificationsService.sendVerificationEmail.mockResolvedValue(undefined); - mockAuditLogService.logAuth.mockResolvedValue(undefined); + afterEach(() => { + jest.clearAllMocks(); }); - - it('should register a new user successfully', async () => { - const result = await service.register(registerDto, '127.0.0.1', 'TestAgent'); - - expect(result).toEqual({ - user: { - id: mockUser.id, - email: mockUser.email, - firstName: mockUser.firstName, - lastName: mockUser.lastName, - role: mockUser.role, - isEmailVerified: mockUser.isEmailVerified, - }, - accessToken: 'access-token', - refreshToken: 'refresh-token', - message: 'Registration successful. Please check your email to verify your account.', - }); - - expect(mockUsersService.create).toHaveBeenCalledWith(registerDto); - expect(mockNotificationsService.sendVerificationEmail).toHaveBeenCalled(); - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.REGISTER, - mockUser.id, - mockUser.email, - '127.0.0.1', - 'TestAgent', - { sessionId: 'session-1' }, - ); + // ───────────────────────────────────────────────────────────────────────── + // TEST SUITES + // ───────────────────────────────────────────────────────────────────────── + describe('register', () => { + const registerDto = { + email: 'test@example.com', + password: 'Password123!', + firstName: 'John', + lastName: 'Doe', + }; + const mockUser = { + id: 'user-1', + email: registerDto.email, + firstName: registerDto.firstName, + lastName: registerDto.lastName, + role: UserRole.STUDENT, + isEmailVerified: false, + }; + beforeEach(() => { + mockTransactionService.runInTransaction.mockImplementation(async (fn) => fn()); + mockUsersService.create.mockResolvedValue(mockUser); + mockUsersService.updateEmailVerificationToken.mockResolvedValue(undefined); + mockUsersService.updateRefreshToken.mockResolvedValue(undefined); + mockSessionService.createSession.mockResolvedValue('session-1'); + mockJwtService.sign + .mockReturnValueOnce('access-token') + .mockReturnValueOnce('refresh-token'); + mockNotificationsService.sendVerificationEmail.mockResolvedValue(undefined); + mockAuditLogService.logAuth.mockResolvedValue(undefined); + }); + it('should register a new user successfully', async () => { + const result = await service.register(registerDto, '127.0.0.1', 'TestAgent'); + expect(result).toEqual({ + user: { + id: mockUser.id, + email: mockUser.email, + firstName: mockUser.firstName, + lastName: mockUser.lastName, + role: mockUser.role, + isEmailVerified: mockUser.isEmailVerified, + }, + accessToken: 'access-token', + refreshToken: 'refresh-token', + message: 'Registration successful. Please check your email to verify your account.', + }); + expect(mockUsersService.create).toHaveBeenCalledWith(registerDto); + expect(mockNotificationsService.sendVerificationEmail).toHaveBeenCalled(); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.REGISTER, mockUser.id, mockUser.email, '127.0.0.1', 'TestAgent', { sessionId: 'session-1' }); + }); + it('should handle registration without IP and user agent', async () => { + await service.register(registerDto); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.REGISTER, mockUser.id, mockUser.email, 'unknown', 'unknown', { sessionId: 'session-1' }); + }); + it('should run registration in a transaction', async () => { + await service.register(registerDto); + expect(mockTransactionService.runInTransaction).toHaveBeenCalled(); + }); }); - - it('should handle registration without IP and user agent', async () => { - await service.register(registerDto); - - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.REGISTER, - mockUser.id, - mockUser.email, - 'unknown', - 'unknown', - { sessionId: 'session-1' }, - ); + describe('login', () => { + const loginDto = { + email: 'test@example.com', + password: 'Password123!', + }; + const mockUser = { + id: 'user-1', + email: loginDto.email, + password: '$2a$10$hashedpassword', + firstName: 'John', + lastName: 'Doe', + role: UserRole.STUDENT, + isEmailVerified: true, + status: 'ACTIVE', + }; + beforeEach(() => { + mockUsersService.findByEmail.mockResolvedValue(mockUser); + mockUsersService.updateLastLogin.mockResolvedValue(undefined); + mockUsersService.updateRefreshToken.mockResolvedValue(undefined); + mockSessionService.createSession.mockResolvedValue('session-1'); + mockJwtService.sign + .mockReturnValueOnce('access-token') + .mockReturnValueOnce('refresh-token'); + mockAuditLogService.logAuth.mockResolvedValue(undefined); + }); + it('should login user successfully with valid credentials', async () => { + // Mock bcrypt.compare to return true + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); + const result = await service.login(loginDto, '127.0.0.1', 'TestAgent'); + expect(result).toEqual({ + user: { + id: mockUser.id, + email: mockUser.email, + firstName: mockUser.firstName, + lastName: mockUser.lastName, + role: mockUser.role, + isEmailVerified: mockUser.isEmailVerified, + }, + accessToken: 'access-token', + refreshToken: 'refresh-token', + }); + expect(mockUsersService.findByEmail).toHaveBeenCalledWith(loginDto.email); + expect(mockUsersService.updateLastLogin).toHaveBeenCalledWith(mockUser.id); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGIN, mockUser.id, mockUser.email, '127.0.0.1', 'TestAgent', { sessionId: 'session-1' }); + }); + it('should throw UnauthorizedException when user not found', async () => { + mockUsersService.findByEmail.mockResolvedValue(null); + await expect(service.login(loginDto)).rejects.toThrow(UnauthorizedException); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGIN_FAILED, null, loginDto.email, 'unknown', 'unknown', { reason: 'User not found' }, AuditSeverity.WARNING); + }); + it('should throw UnauthorizedException when password is invalid', async () => { + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(false); + await expect(service.login(loginDto)).rejects.toThrow(UnauthorizedException); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGIN_FAILED, mockUser.id, mockUser.email, 'unknown', 'unknown', { reason: 'Invalid password' }, AuditSeverity.WARNING); + }); + it('should handle login without IP and user agent', async () => { + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); + await service.login(loginDto); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGIN, mockUser.id, mockUser.email, 'unknown', 'unknown', { sessionId: 'session-1' }); + }); }); - - it('should run registration in a transaction', async () => { - await service.register(registerDto); - - expect(mockTransactionService.runInTransaction).toHaveBeenCalled(); + describe('refreshToken', () => { + const refreshToken = 'valid-refresh-token'; + const mockUser = { + id: 'user-1', + email: 'test@example.com', + refreshToken: '$2a$10$hashedrefreshtoken', + }; + beforeEach(() => { + mockJwtService.verify.mockReturnValue({ + sub: mockUser.id, + email: mockUser.email, + sid: 'session-1', + }); + mockSessionService.withLock.mockImplementation(async (key, fn) => fn()); + mockUsersService.findOne.mockResolvedValue(mockUser); + mockSessionService.getSession.mockResolvedValue({ id: 'session-1' }); + mockSessionService.touchSession.mockResolvedValue(undefined); + mockJwtService.sign + .mockReturnValueOnce('new-access-token') + .mockReturnValueOnce('new-refresh-token'); + mockUsersService.updateRefreshToken.mockResolvedValue(undefined); + }); + it('should refresh tokens successfully', async () => { + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); + const result = await service.refreshToken(refreshToken); + expect(result).toEqual({ + accessToken: 'new-access-token', + refreshToken: 'new-refresh-token', + }); + expect(mockJwtService.verify).toHaveBeenCalledWith(refreshToken, { + secret: 'refresh-secret', + }); + expect(mockSessionService.touchSession).toHaveBeenCalledWith('session-1', { + lastRefreshAt: expect.any(Number), + }); + }); + it('should create new session if current session not found', async () => { + mockSessionService.getSession.mockResolvedValue(null); + mockSessionService.createSession.mockResolvedValue('new-session-1'); + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); + await service.refreshToken(refreshToken); + expect(mockSessionService.createSession).toHaveBeenCalledWith(mockUser.id, { + type: 'auth-refresh', + }); + }); + it('should throw UnauthorizedException when refresh token is invalid', async () => { + mockJwtService.verify.mockImplementation(() => { + throw new Error('Invalid token'); + }); + await expect(service.refreshToken('invalid-token')).rejects.toThrow(UnauthorizedException); + }); + it('should throw UnauthorizedException when user not found', async () => { + mockUsersService.findOne.mockResolvedValue(null); + await expect(service.refreshToken(refreshToken)).rejects.toThrow(UnauthorizedException); + }); + it('should throw UnauthorizedException when stored refresh token is invalid', async () => { + const bcrypt = require('bcryptjs'); + jest.spyOn(bcrypt, 'compare').mockResolvedValue(false); + await expect(service.refreshToken(refreshToken)).rejects.toThrow(UnauthorizedException); + }); }); - }); - - describe('login', () => { - const loginDto = { - email: 'test@example.com', - password: 'Password123!', - }; - - const mockUser = { - id: 'user-1', - email: loginDto.email, - password: '$2a$10$hashedpassword', - firstName: 'John', - lastName: 'Doe', - role: UserRole.STUDENT, - isEmailVerified: true, - status: 'ACTIVE', - }; - - beforeEach(() => { - mockUsersService.findByEmail.mockResolvedValue(mockUser); - mockUsersService.updateLastLogin.mockResolvedValue(undefined); - mockUsersService.updateRefreshToken.mockResolvedValue(undefined); - mockSessionService.createSession.mockResolvedValue('session-1'); - mockJwtService.sign - .mockReturnValueOnce('access-token') - .mockReturnValueOnce('refresh-token'); - mockAuditLogService.logAuth.mockResolvedValue(undefined); + describe('logout', () => { + const userId = 'user-1'; + const sessionId = 'session-1'; + const mockUser = { + id: userId, + email: 'test@example.com', + }; + beforeEach(() => { + mockUsersService.findOne.mockResolvedValue(mockUser); + mockSessionService.withLock.mockImplementation(async (key, fn) => fn()); + mockSessionService.removeSession.mockResolvedValue(undefined); + mockUsersService.updateRefreshToken.mockResolvedValue(undefined); + mockAuditLogService.logAuth.mockResolvedValue(undefined); + }); + it('should logout user successfully', async () => { + const result = await service.logout(userId, sessionId, '127.0.0.1', 'TestAgent'); + expect(result).toEqual({ message: 'Logout successful' }); + expect(mockSessionService.removeSession).toHaveBeenCalledWith(sessionId); + expect(mockUsersService.updateRefreshToken).toHaveBeenCalledWith(userId, null); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGOUT, userId, mockUser.email, '127.0.0.1', 'TestAgent', { sessionId }); + }); + it('should handle logout without session ID', async () => { + await service.logout(userId); + expect(mockSessionService.removeSession).not.toHaveBeenCalled(); + expect(mockUsersService.updateRefreshToken).toHaveBeenCalledWith(userId, null); + }); + it('should handle logout without IP and user agent', async () => { + await service.logout(userId, sessionId); + expect(mockAuditLogService.logAuth).toHaveBeenCalledWith(AuditAction.LOGOUT, userId, mockUser.email, 'unknown', 'unknown', { sessionId }); + }); }); - - it('should login user successfully with valid credentials', async () => { - // Mock bcrypt.compare to return true - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); - - const result = await service.login(loginDto, '127.0.0.1', 'TestAgent'); - - expect(result).toEqual({ - user: { - id: mockUser.id, - email: mockUser.email, - firstName: mockUser.firstName, - lastName: mockUser.lastName, - role: mockUser.role, - isEmailVerified: mockUser.isEmailVerified, - }, - accessToken: 'access-token', - refreshToken: 'refresh-token', - }); - - expect(mockUsersService.findByEmail).toHaveBeenCalledWith(loginDto.email); - expect(mockUsersService.updateLastLogin).toHaveBeenCalledWith(mockUser.id); - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGIN, - mockUser.id, - mockUser.email, - '127.0.0.1', - 'TestAgent', - { sessionId: 'session-1' }, - ); + describe('forgotPassword', () => { + const email = 'test@example.com'; + const mockUser = { id: 'user-1', email }; + beforeEach(() => { + mockUsersService.findByEmail.mockResolvedValue(mockUser); + mockUsersService.updatePasswordResetToken.mockResolvedValue(undefined); + }); + it('should initiate password reset for existing user', async () => { + const result = await service.forgotPassword(email); + expect(result).toEqual({ + message: 'If the email exists, a password reset link has been sent.', + }); + expect(mockUsersService.findByEmail).toHaveBeenCalledWith(email); + expect(mockUsersService.updatePasswordResetToken).toHaveBeenCalled(); + }); + it('should not reveal if user does not exist', async () => { + mockUsersService.findByEmail.mockResolvedValue(null); + const result = await service.forgotPassword('nonexistent@example.com'); + expect(result).toEqual({ + message: 'If the email exists, a password reset link has been sent.', + }); + expect(mockUsersService.updatePasswordResetToken).not.toHaveBeenCalled(); + }); }); - - it('should throw UnauthorizedException when user not found', async () => { - mockUsersService.findByEmail.mockResolvedValue(null); - - await expect(service.login(loginDto)).rejects.toThrow(UnauthorizedException); - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGIN_FAILED, - null, - loginDto.email, - 'unknown', - 'unknown', - { reason: 'User not found' }, - AuditSeverity.WARNING, - ); + describe('generateTokens', () => { + const mockUser = { + id: 'user-1', + email: 'test@example.com', + role: UserRole.STUDENT, + }; + const sessionId = 'session-1'; + beforeEach(() => { + mockJwtService.sign + .mockReturnValueOnce('access-token') + .mockReturnValueOnce('refresh-token'); + }); + it('should generate access and refresh tokens', async () => { + const result = await (service as unknown).generateTokens(mockUser, sessionId); + expect(result).toEqual({ + accessToken: 'access-token', + refreshToken: 'refresh-token', + }); + expect(mockJwtService.sign).toHaveBeenCalledTimes(2); + expect(mockJwtService.sign).toHaveBeenNthCalledWith(1, { + sub: mockUser.id, + email: mockUser.email, + role: mockUser.role, + sid: sessionId, + }, { + secret: 'access-secret', + expiresIn: '15m', + }); + }); }); - - it('should throw UnauthorizedException when password is invalid', async () => { - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(false); - - await expect(service.login(loginDto)).rejects.toThrow(UnauthorizedException); - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGIN_FAILED, - mockUser.id, - mockUser.email, - 'unknown', - 'unknown', - { reason: 'Invalid password' }, - AuditSeverity.WARNING, - ); + describe('generateRandomToken', () => { + it('should generate a random token', () => { + const token1 = (service as unknown).generateRandomToken(); + const token2 = (service as unknown).generateRandomToken(); + expect(typeof token1).toBe('string'); + expect(token1.length).toBeGreaterThan(0); + expect(token1).not.toBe(token2); // Should be different each time + }); }); - - it('should handle login without IP and user agent', async () => { - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); - - await service.login(loginDto); - - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGIN, - mockUser.id, - mockUser.email, - 'unknown', - 'unknown', - { sessionId: 'session-1' }, - ); - }); - }); - - describe('refreshToken', () => { - const refreshToken = 'valid-refresh-token'; - const mockUser = { - id: 'user-1', - email: 'test@example.com', - refreshToken: '$2a$10$hashedrefreshtoken', - }; - - beforeEach(() => { - mockJwtService.verify.mockReturnValue({ - sub: mockUser.id, - email: mockUser.email, - sid: 'session-1', - }); - mockSessionService.withLock.mockImplementation(async (key, fn) => fn()); - mockUsersService.findOne.mockResolvedValue(mockUser); - mockSessionService.getSession.mockResolvedValue({ id: 'session-1' }); - mockSessionService.touchSession.mockResolvedValue(undefined); - mockJwtService.sign - .mockReturnValueOnce('new-access-token') - .mockReturnValueOnce('new-refresh-token'); - mockUsersService.updateRefreshToken.mockResolvedValue(undefined); - }); - - it('should refresh tokens successfully', async () => { - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); - - const result = await service.refreshToken(refreshToken); - - expect(result).toEqual({ - accessToken: 'new-access-token', - refreshToken: 'new-refresh-token', - }); - - expect(mockJwtService.verify).toHaveBeenCalledWith(refreshToken, { - secret: 'refresh-secret', - }); - expect(mockSessionService.touchSession).toHaveBeenCalledWith('session-1', { - lastRefreshAt: expect.any(Number), - }); - }); - - it('should create new session if current session not found', async () => { - mockSessionService.getSession.mockResolvedValue(null); - mockSessionService.createSession.mockResolvedValue('new-session-1'); - - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(true); - - await service.refreshToken(refreshToken); - - expect(mockSessionService.createSession).toHaveBeenCalledWith(mockUser.id, { - type: 'auth-refresh', - }); - }); - - it('should throw UnauthorizedException when refresh token is invalid', async () => { - mockJwtService.verify.mockImplementation(() => { - throw new Error('Invalid token'); - }); - - await expect(service.refreshToken('invalid-token')).rejects.toThrow( - UnauthorizedException, - ); - }); - - it('should throw UnauthorizedException when user not found', async () => { - mockUsersService.findOne.mockResolvedValue(null); - - await expect(service.refreshToken(refreshToken)).rejects.toThrow(UnauthorizedException); - }); - - it('should throw UnauthorizedException when stored refresh token is invalid', async () => { - const bcrypt = require('bcryptjs'); - jest.spyOn(bcrypt, 'compare').mockResolvedValue(false); - - await expect(service.refreshToken(refreshToken)).rejects.toThrow(UnauthorizedException); - }); - }); - - describe('logout', () => { - const userId = 'user-1'; - const sessionId = 'session-1'; - const mockUser = { - id: userId, - email: 'test@example.com', - }; - - beforeEach(() => { - mockUsersService.findOne.mockResolvedValue(mockUser); - mockSessionService.withLock.mockImplementation(async (key, fn) => fn()); - mockSessionService.removeSession.mockResolvedValue(undefined); - mockUsersService.updateRefreshToken.mockResolvedValue(undefined); - mockAuditLogService.logAuth.mockResolvedValue(undefined); - }); - - it('should logout user successfully', async () => { - const result = await service.logout(userId, sessionId, '127.0.0.1', 'TestAgent'); - - expect(result).toEqual({ message: 'Logout successful' }); - expect(mockSessionService.removeSession).toHaveBeenCalledWith(sessionId); - expect(mockUsersService.updateRefreshToken).toHaveBeenCalledWith(userId, null); - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGOUT, - userId, - mockUser.email, - '127.0.0.1', - 'TestAgent', - { sessionId }, - ); - }); - - it('should handle logout without session ID', async () => { - await service.logout(userId); - - expect(mockSessionService.removeSession).not.toHaveBeenCalled(); - expect(mockUsersService.updateRefreshToken).toHaveBeenCalledWith(userId, null); - }); - - it('should handle logout without IP and user agent', async () => { - await service.logout(userId, sessionId); - - expect(mockAuditLogService.logAuth).toHaveBeenCalledWith( - AuditAction.LOGOUT, - userId, - mockUser.email, - 'unknown', - 'unknown', - { sessionId }, - ); - }); - }); - - describe('forgotPassword', () => { - const email = 'test@example.com'; - const mockUser = { id: 'user-1', email }; - - beforeEach(() => { - mockUsersService.findByEmail.mockResolvedValue(mockUser); - mockUsersService.updatePasswordResetToken.mockResolvedValue(undefined); - }); - - it('should initiate password reset for existing user', async () => { - const result = await service.forgotPassword(email); - - expect(result).toEqual({ - message: 'If the email exists, a password reset link has been sent.', - }); - expect(mockUsersService.findByEmail).toHaveBeenCalledWith(email); - expect(mockUsersService.updatePasswordResetToken).toHaveBeenCalled(); - }); - - it('should not reveal if user does not exist', async () => { - mockUsersService.findByEmail.mockResolvedValue(null); - - const result = await service.forgotPassword('nonexistent@example.com'); - - expect(result).toEqual({ - message: 'If the email exists, a password reset link has been sent.', - }); - expect(mockUsersService.updatePasswordResetToken).not.toHaveBeenCalled(); - }); - }); - - describe('generateTokens', () => { - const mockUser = { - id: 'user-1', - email: 'test@example.com', - role: UserRole.STUDENT, - }; - const sessionId = 'session-1'; - - beforeEach(() => { - mockJwtService.sign - .mockReturnValueOnce('access-token') - .mockReturnValueOnce('refresh-token'); - }); - - it('should generate access and refresh tokens', async () => { - const result = await (service as any).generateTokens(mockUser, sessionId); - - expect(result).toEqual({ - accessToken: 'access-token', - refreshToken: 'refresh-token', - }); - - expect(mockJwtService.sign).toHaveBeenCalledTimes(2); - expect(mockJwtService.sign).toHaveBeenNthCalledWith( - 1, - { - sub: mockUser.id, - email: mockUser.email, - role: mockUser.role, - sid: sessionId, - }, - { - secret: 'access-secret', - expiresIn: '15m', - }, - ); - }); - }); - - describe('generateRandomToken', () => { - it('should generate a random token', () => { - const token1 = (service as any).generateRandomToken(); - const token2 = (service as any).generateRandomToken(); - - expect(typeof token1).toBe('string'); - expect(token1.length).toBeGreaterThan(0); - expect(token1).not.toBe(token2); // Should be different each time - }); - }); -}); \ No newline at end of file +}); diff --git a/src/auth/auth.service.ts b/src/auth/auth.service.ts index f2671f14..b5ba8232 100644 --- a/src/auth/auth.service.ts +++ b/src/auth/auth.service.ts @@ -8,451 +8,303 @@ import { randomBytes } from 'crypto'; import { SessionService } from '../session/session.service'; import { TransactionService } from '../common/database/transaction.service'; import { UserRole } from '../users/entities/user.entity'; -import { - ensureValidCredentials, - ensureUserIsActive, - ensureValidUserToken, -} from '../common/utils/user.utils'; +import { ensureValidCredentials, ensureUserIsActive, ensureValidUserToken, } from '../common/utils/user.utils'; import { NotificationsService } from '../notifications/notifications.service'; import { AuditLogService } from '../audit-log/audit-log.service'; import { AuditAction, AuditSeverity } from '../audit-log/enums/audit-action.enum'; - interface JwtTokenPayload { - sub: string; - email: string; - role: UserRole; - sid: string; + sub: string; + email: string; + role: UserRole; + sid: string; } - interface AuthTokens { - accessToken: string; - refreshToken: string; + accessToken: string; + refreshToken: string; } - interface AuthUserResponse { - id: string; - email: string; - firstName: string; - lastName: string; - role: UserRole; - isEmailVerified: boolean; + id: string; + email: string; + firstName: string; + lastName: string; + role: UserRole; + isEmailVerified: boolean; } - interface RegisterResponse { - user: AuthUserResponse; - accessToken: string; - refreshToken: string; - message: string; + user: AuthUserResponse; + accessToken: string; + refreshToken: string; + message: string; } - interface LoginResponse { - user: AuthUserResponse; - accessToken: string; - refreshToken: string; + user: AuthUserResponse; + accessToken: string; + refreshToken: string; } - interface TokenUser { - id: string; - email: string; - role: UserRole; + id: string; + email: string; + role: UserRole; } - @Injectable() export class AuthService { - private readonly logger = new Logger(AuthService.name); - - constructor( - private readonly usersService: UsersService, - private readonly jwtService: JwtService, - private readonly configService: ConfigService, - private readonly sessionService: SessionService, - private readonly transactionService: TransactionService, - private readonly notificationsService: NotificationsService, - private readonly auditLogService: AuditLogService, - ) {} - - async register( - registerDto: RegisterDto, - ipAddress?: string, - userAgent?: string, - ): Promise { - return await this.transactionService.runInTransaction(async (_manager) => { - // Create user - const user = await this.usersService.create(registerDto); - - // Generate email verification token - const verificationToken = this.generateRandomToken(); - const verificationExpires = new Date(Date.now() + 24 * 60 * 60 * 1000); // 24 hours - - await this.usersService.updateEmailVerificationToken( - user.id, - verificationToken, - verificationExpires, - ); - - // Send verification email - await this.notificationsService.sendVerificationEmail(user.email, verificationToken); - - const sessionId = await this.sessionService.createSession(user.id, { type: 'auth-register' }); - const { accessToken, refreshToken } = await this.generateTokens(user, sessionId); - - // Save refresh token - const hashedRefreshToken = await bcrypt.hash(refreshToken, 10); - await this.usersService.updateRefreshToken(user.id, hashedRefreshToken); - - // Log registration - await this.auditLogService.logAuth( - AuditAction.REGISTER, - user.id, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { sessionId }, - ); - - return { - user: { - id: user.id, - email: user.email, - firstName: user.firstName, - lastName: user.lastName, - role: user.role, - isEmailVerified: user.isEmailVerified, - }, - accessToken, - refreshToken, - message: 'Registration successful. Please check your email to verify your account.', - }; - }); - } - - async login(loginDto: LoginDto, ipAddress?: string, userAgent?: string): Promise { - // Find user - const userOrNull = await this.usersService.findByEmail(loginDto.email); - - // Log failed login attempt if user not found - if (!userOrNull) { - await this.auditLogService.logAuth( - AuditAction.LOGIN_FAILED, - null, - loginDto.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { reason: 'User not found' }, - AuditSeverity.WARNING, - ); - throw new UnauthorizedException('Invalid credentials'); + private readonly logger = new Logger(AuthService.name); + constructor(private readonly usersService: UsersService, private readonly jwtService: JwtService, private readonly configService: ConfigService, private readonly sessionService: SessionService, private readonly transactionService: TransactionService, private readonly notificationsService: NotificationsService, private readonly auditLogService: AuditLogService) { } + async register(registerDto: RegisterDto, ipAddress?: string, userAgent?: string): Promise { + return await this.transactionService.runInTransaction(async (_manager) => { + // Create user + const user = await this.usersService.create(registerDto); + // Generate email verification token + const verificationToken = this.generateRandomToken(); + const verificationExpires = new Date(Date.now() + 24 * 60 * 60 * 1000); // 24 hours + await this.usersService.updateEmailVerificationToken(user.id, verificationToken, verificationExpires); + // Send verification email + await this.notificationsService.sendVerificationEmail(user.email, verificationToken); + const sessionId = await this.sessionService.createSession(user.id, { type: 'auth-register' }); + const { accessToken, refreshToken } = await this.generateTokens(user, sessionId); + // Save refresh token + const hashedRefreshToken = await bcrypt.hash(refreshToken, 10); + await this.usersService.updateRefreshToken(user.id, hashedRefreshToken); + // Log registration + await this.auditLogService.logAuth(AuditAction.REGISTER, user.id, user.email, ipAddress || 'unknown', userAgent || 'unknown', { sessionId }); + return { + user: { + id: user.id, + email: user.email, + firstName: user.firstName, + lastName: user.lastName, + role: user.role, + isEmailVerified: user.isEmailVerified, + }, + accessToken, + refreshToken, + message: 'Registration successful. Please check your email to verify your account.', + }; + }); } - - const user = ensureValidCredentials(userOrNull); - - // Verify password - const isPasswordValid = await bcrypt.compare(loginDto.password, user.password); - if (!isPasswordValid) { - await this.auditLogService.logAuth( - AuditAction.LOGIN_FAILED, - user.id, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { reason: 'Invalid password' }, - AuditSeverity.WARNING, - ); - throw new UnauthorizedException('Invalid credentials'); - } - - // Check if user is active - try { - ensureUserIsActive(user); - } catch (error) { - await this.auditLogService.logAuth( - AuditAction.LOGIN_FAILED, - user.id, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { reason: 'User account inactive' }, - AuditSeverity.WARNING, - ); - throw error; - } - - // Update last login - await this.usersService.updateLastLogin(user.id); - - const sessionId = await this.sessionService.createSession(user.id, { type: 'auth-login' }); - const { accessToken, refreshToken } = await this.generateTokens(user, sessionId); - - // Save refresh token - const hashedRefreshToken = await bcrypt.hash(refreshToken, 10); - await this.usersService.updateRefreshToken(user.id, hashedRefreshToken); - - // Log successful login - await this.auditLogService.logAuth( - AuditAction.LOGIN, - user.id, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { sessionId }, - ); - - return { - user: { - id: user.id, - email: user.email, - firstName: user.firstName, - lastName: user.lastName, - role: user.role, - isEmailVerified: user.isEmailVerified, - }, - accessToken, - refreshToken, - }; - } - - async refreshToken(refreshToken: string): Promise { - try { - // Verify refresh token - const payload = this.jwtService.verify(refreshToken, { - secret: this.configService.get('JWT_REFRESH_SECRET') || 'refresh-secret-key', - }); - return this.sessionService.withLock(`refresh:${payload.sub}`, async () => { + async login(loginDto: LoginDto, ipAddress?: string, userAgent?: string): Promise { // Find user - const user = await this.usersService.findOne(payload.sub); - if (!user || !user.refreshToken) { - throw new UnauthorizedException('Invalid refresh token'); + const userOrNull = await this.usersService.findByEmail(loginDto.email); + // Log failed login attempt if user not found + if (!userOrNull) { + await this.auditLogService.logAuth(AuditAction.LOGIN_FAILED, null, loginDto.email, ipAddress || 'unknown', userAgent || 'unknown', { reason: 'User not found' }, AuditSeverity.WARNING); + throw new UnauthorizedException('Invalid credentials'); } - - // Verify stored refresh token - const isRefreshTokenValid = await bcrypt.compare(refreshToken, user.refreshToken); - if (!isRefreshTokenValid) { - throw new UnauthorizedException('Invalid refresh token'); + const user = ensureValidCredentials(userOrNull); + // Verify password + const isPasswordValid = await bcrypt.compare(loginDto.password, user.password); + if (!isPasswordValid) { + await this.auditLogService.logAuth(AuditAction.LOGIN_FAILED, user.id, user.email, ipAddress || 'unknown', userAgent || 'unknown', { reason: 'Invalid password' }, AuditSeverity.WARNING); + throw new UnauthorizedException('Invalid credentials'); } - - let sessionId = payload.sid as string | undefined; - if (sessionId) { - const session = await this.sessionService.getSession(sessionId); - if (!session) { - sessionId = await this.sessionService.createSession(user.id, { type: 'auth-refresh' }); - } else { - await this.sessionService.touchSession(sessionId, { - lastRefreshAt: Date.now(), - }); - } - } else { - sessionId = await this.sessionService.createSession(user.id, { type: 'auth-refresh' }); + // Check if user is active + try { + ensureUserIsActive(user); } - - // Generate new tokens - const tokens = await this.generateTokens(user, sessionId); - - // Update refresh token - const hashedRefreshToken = await bcrypt.hash(tokens.refreshToken, 10); + catch (error) { + await this.auditLogService.logAuth(AuditAction.LOGIN_FAILED, user.id, user.email, ipAddress || 'unknown', userAgent || 'unknown', { reason: 'User account inactive' }, AuditSeverity.WARNING); + throw error; + } + // Update last login + await this.usersService.updateLastLogin(user.id); + const sessionId = await this.sessionService.createSession(user.id, { type: 'auth-login' }); + const { accessToken, refreshToken } = await this.generateTokens(user, sessionId); + // Save refresh token + const hashedRefreshToken = await bcrypt.hash(refreshToken, 10); await this.usersService.updateRefreshToken(user.id, hashedRefreshToken); - - return tokens; - }); - } catch { - throw new UnauthorizedException('Invalid refresh token'); + // Log successful login + await this.auditLogService.logAuth(AuditAction.LOGIN, user.id, user.email, ipAddress || 'unknown', userAgent || 'unknown', { sessionId }); + return { + user: { + id: user.id, + email: user.email, + firstName: user.firstName, + lastName: user.lastName, + role: user.role, + isEmailVerified: user.isEmailVerified, + }, + accessToken, + refreshToken, + }; + } + async refreshToken(refreshToken: string): Promise { + try { + // Verify refresh token + const payload = this.jwtService.verify(refreshToken, { + secret: this.configService.get('JWT_REFRESH_SECRET') || 'refresh-secret-key', + }); + return this.sessionService.withLock(`refresh:${payload.sub}`, async () => { + // Find user + const user = await this.usersService.findOne(payload.sub); + if (!user || !user.refreshToken) { + throw new UnauthorizedException('Invalid refresh token'); + } + // Verify stored refresh token + const isRefreshTokenValid = await bcrypt.compare(refreshToken, user.refreshToken); + if (!isRefreshTokenValid) { + throw new UnauthorizedException('Invalid refresh token'); + } + let sessionId = payload.sid as string | undefined; + if (sessionId) { + const session = await this.sessionService.getSession(sessionId); + if (!session) { + sessionId = await this.sessionService.createSession(user.id, { type: 'auth-refresh' }); + } + else { + await this.sessionService.touchSession(sessionId, { + lastRefreshAt: Date.now(), + }); + } + } + else { + sessionId = await this.sessionService.createSession(user.id, { type: 'auth-refresh' }); + } + // Generate new tokens + const tokens = await this.generateTokens(user, sessionId); + // Update refresh token + const hashedRefreshToken = await bcrypt.hash(tokens.refreshToken, 10); + await this.usersService.updateRefreshToken(user.id, hashedRefreshToken); + return tokens; + }); + } + catch { + throw new UnauthorizedException('Invalid refresh token'); + } } - } - - async logout( - userId: string, - sessionId?: string, - ipAddress?: string, - userAgent?: string, - ): Promise<{ message: string }> { - const user = await this.usersService.findOne(userId); - - await this.sessionService.withLock(`logout:${userId}`, async () => { - if (sessionId) { - await this.sessionService.removeSession(sessionId); - } - await this.usersService.updateRefreshToken(userId, null); - }); - - // Log logout - await this.auditLogService.logAuth( - AuditAction.LOGOUT, - userId, - user?.email || null, - ipAddress || 'unknown', - userAgent || 'unknown', - { sessionId }, - ); - - return { message: 'Logout successful' }; - } - - async forgotPassword(email: string): Promise<{ message: string }> { - const user = await this.usersService.findByEmail(email); - if (!user) { - // Don't reveal if user exists - return { message: 'If the email exists, a password reset link has been sent.' }; + async logout(userId: string, sessionId?: string, ipAddress?: string, userAgent?: string): Promise<{ + message: string; + }> { + const user = await this.usersService.findOne(userId); + await this.sessionService.withLock(`logout:${userId}`, async () => { + if (sessionId) { + await this.sessionService.removeSession(sessionId); + } + await this.usersService.updateRefreshToken(userId, null); + }); + // Log logout + await this.auditLogService.logAuth(AuditAction.LOGOUT, userId, user?.email || null, ipAddress || 'unknown', userAgent || 'unknown', { sessionId }); + return { message: 'Logout successful' }; } - - // Generate reset token - const resetToken = this.generateRandomToken(); - const resetExpires = new Date(Date.now() + 60 * 60 * 1000); // 1 hour - - await this.usersService.updatePasswordResetToken(user.id, resetToken, resetExpires); - - // Send password reset email - await this.notificationsService.sendPasswordResetEmail(user.email, resetToken); - - return { message: 'If the email exists, a password reset link has been sent.' }; - } - - async resetPassword(resetPasswordDto: ResetPasswordDto): Promise<{ message: string }> { - // Find user by reset token - const userOrNull = await this.usersService.findByPasswordResetToken(resetPasswordDto.token); - const user = ensureValidUserToken( - userOrNull, - 'passwordResetToken', - 'passwordResetExpires', - 'Invalid or expired reset token', - ); - - // Update password - await this.usersService.update(user.id, { password: resetPasswordDto.newPassword }); - - // Clear reset token - await this.usersService.updatePasswordResetToken(user.id, null, null); - - return { message: 'Password has been reset successfully' }; - } - - async changePassword( - userId: string, - changePasswordDto: ChangePasswordDto, - ipAddress?: string, - userAgent?: string, - ): Promise<{ message: string }> { - const user = await this.usersService.findOne(userId); - - // Verify current password - const isPasswordValid = await bcrypt.compare(changePasswordDto.currentPassword, user.password); - if (!isPasswordValid) { - await this.auditLogService.logAuth( - AuditAction.PASSWORD_CHANGE, - userId, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { success: false, reason: 'Current password incorrect' }, - AuditSeverity.WARNING, - ); - throw new BadRequestException('Current password is incorrect'); + async forgotPassword(email: string): Promise<{ + message: string; + }> { + const user = await this.usersService.findByEmail(email); + if (!user) { + // Don't reveal if user exists + return { message: 'If the email exists, a password reset link has been sent.' }; + } + // Generate reset token + const resetToken = this.generateRandomToken(); + const resetExpires = new Date(Date.now() + 60 * 60 * 1000); // 1 hour + await this.usersService.updatePasswordResetToken(user.id, resetToken, resetExpires); + // Send password reset email + await this.notificationsService.sendPasswordResetEmail(user.email, resetToken); + return { message: 'If the email exists, a password reset link has been sent.' }; } - - // Update password - await this.usersService.update(userId, { password: changePasswordDto.newPassword }); - - // Log password change - await this.auditLogService.logAuth( - AuditAction.PASSWORD_CHANGE, - userId, - user.email, - ipAddress || 'unknown', - userAgent || 'unknown', - { success: true }, - ); - - return { message: 'Password changed successfully' }; - } - - async verifyEmail(token: string): Promise<{ message: string }> { - // Find user by verification token - const userOrNull = await this.usersService.findByEmailVerificationToken(token); - const user = ensureValidUserToken( - userOrNull, - 'emailVerificationToken', - 'emailVerificationExpires', - 'Invalid or expired verification token', - ); - - // Update user as verified - await this.usersService.update(user.id, { isEmailVerified: true }); - - // Clear verification token - await this.usersService.updateEmailVerificationToken(user.id, null, null); - - return { message: 'Email verified successfully' }; - } - - private async generateTokens(user: TokenUser, sessionId: string): Promise { - const payload: JwtTokenPayload = { - sub: user.id, - email: user.email, - role: user.role, - sid: sessionId, - }; - - const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); - - const [accessToken, refreshToken] = await Promise.all([ - this.jwtService.signAsync(payload, { - secret: currentSecret, - expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), // 900s = 15m - header: currentVersion ? { kid: currentVersion } : undefined, - }), - this.jwtService.signAsync(payload, { - secret: this.configService.get('JWT_REFRESH_SECRET') || 'refresh-secret-key', - expiresIn: parseInt( - this.configService.get('JWT_REFRESH_EXPIRES_IN') || '604800', - 10, - ), // 604800s = 7d - }), - ]); - - return { accessToken, refreshToken }; - } - - private getCurrentJwtAccessSecret(): { currentVersion: string | null; currentSecret: string } { - const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); - const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; - - if (!jwtSecretsRaw) { - return { - currentVersion, - currentSecret: this.configService.get('JWT_SECRET') || 'your-secret-key', - }; + async resetPassword(resetPasswordDto: ResetPasswordDto): Promise<{ + message: string; + }> { + // Find user by reset token + const userOrNull = await this.usersService.findByPasswordResetToken(resetPasswordDto.token); + const user = ensureValidUserToken(userOrNull, 'passwordResetToken', 'passwordResetExpires', 'Invalid or expired reset token'); + // Update password + await this.usersService.update(user.id, { password: resetPasswordDto.newPassword }); + // Clear reset token + await this.usersService.updatePasswordResetToken(user.id, null, null); + return { message: 'Password has been reset successfully' }; + } + async changePassword(userId: string, changePasswordDto: ChangePasswordDto, ipAddress?: string, userAgent?: string): Promise<{ + message: string; + }> { + const user = await this.usersService.findOne(userId); + // Verify current password + const isPasswordValid = await bcrypt.compare(changePasswordDto.currentPassword, user.password); + if (!isPasswordValid) { + await this.auditLogService.logAuth(AuditAction.PASSWORD_CHANGE, userId, user.email, ipAddress || 'unknown', userAgent || 'unknown', { success: false, reason: 'Current password incorrect' }, AuditSeverity.WARNING); + throw new BadRequestException('Current password is incorrect'); + } + // Update password + await this.usersService.update(userId, { password: changePasswordDto.newPassword }); + // Log password change + await this.auditLogService.logAuth(AuditAction.PASSWORD_CHANGE, userId, user.email, ipAddress || 'unknown', userAgent || 'unknown', { success: true }); + return { message: 'Password changed successfully' }; + } + async verifyEmail(token: string): Promise<{ + message: string; + }> { + // Find user by verification token + const userOrNull = await this.usersService.findByEmailVerificationToken(token); + const user = ensureValidUserToken(userOrNull, 'emailVerificationToken', 'emailVerificationExpires', 'Invalid or expired verification token'); + // Update user as verified + await this.usersService.update(user.id, { isEmailVerified: true }); + // Clear verification token + await this.usersService.updateEmailVerificationToken(user.id, null, null); + return { message: 'Email verified successfully' }; + } + private async generateTokens(user: TokenUser, sessionId: string): Promise { + const payload: JwtTokenPayload = { + sub: user.id, + email: user.email, + role: user.role, + sid: sessionId, + }; + const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); + const [accessToken, refreshToken] = await Promise.all([ + this.jwtService.signAsync(payload, { + secret: currentSecret, + expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), // 900s = 15m + header: currentVersion ? { kid: currentVersion } : undefined, + }), + this.jwtService.signAsync(payload, { + secret: this.configService.get('JWT_REFRESH_SECRET') || 'refresh-secret-key', + expiresIn: parseInt(this.configService.get('JWT_REFRESH_EXPIRES_IN') || '604800', 10), // 604800s = 7d + }), + ]); + return { accessToken, refreshToken }; + } + private getCurrentJwtAccessSecret(): { + currentVersion: string | null; + currentSecret: string; + } { + const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); + const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; + if (!jwtSecretsRaw) { + return { + currentVersion, + currentSecret: this.configService.get('JWT_SECRET') || 'your-secret-key', + }; + } + const secrets = this.parseJwtSecrets(jwtSecretsRaw); + const currentSecret = (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); + return { currentVersion, currentSecret: currentSecret || 'your-secret-key' }; + } + private parseJwtSecrets(raw: string): Record { + try { + const parsed = JSON.parse(raw) as unknown; + if (parsed && typeof parsed === 'object') { + return parsed as Record; + } + } + catch { + // ignore + } + return raw + .split(',') + .map((pair) => pair.trim()) + .filter(Boolean) + .reduce>((acc, pair) => { + const idx = pair.indexOf(':'); + if (idx <= 0) + return acc; + const version = pair.slice(0, idx).trim(); + const secret = pair.slice(idx + 1).trim(); + if (!version || !secret) + return acc; + acc[version] = secret; + return acc; + }, {}); } - - const secrets = this.parseJwtSecrets(jwtSecretsRaw); - const currentSecret = - (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); - return { currentVersion, currentSecret: currentSecret || 'your-secret-key' }; - } - - private parseJwtSecrets(raw: string): Record { - try { - const parsed = JSON.parse(raw) as unknown; - if (parsed && typeof parsed === 'object') { - return parsed as Record; - } - } catch { - // ignore + private generateRandomToken(): string { + return randomBytes(32).toString('hex'); } - - return raw - .split(',') - .map((pair) => pair.trim()) - .filter(Boolean) - .reduce>((acc, pair) => { - const idx = pair.indexOf(':'); - if (idx <= 0) return acc; - const version = pair.slice(0, idx).trim(); - const secret = pair.slice(idx + 1).trim(); - if (!version || !secret) return acc; - acc[version] = secret; - return acc; - }, {}); - } - - private generateRandomToken(): string { - return randomBytes(32).toString('hex'); - } } diff --git a/src/auth/decorators/current-user.decorator.ts b/src/auth/decorators/current-user.decorator.ts index 4e610b76..fab8f8ed 100644 --- a/src/auth/decorators/current-user.decorator.ts +++ b/src/auth/decorators/current-user.decorator.ts @@ -1,8 +1,6 @@ import { createParamDecorator, ExecutionContext } from '@nestjs/common'; - export const CurrentUser = createParamDecorator((data: string, ctx: ExecutionContext) => { - const request = ctx.switchToHttp().getRequest(); - const user = request.user; - - return data ? user?.[data] : user; + const request = ctx.switchToHttp().getRequest(); + const user = request.user; + return data ? user?.[data] : user; }); diff --git a/src/auth/decorators/roles.decorator.ts b/src/auth/decorators/roles.decorator.ts index 3288d7e2..850c8575 100644 --- a/src/auth/decorators/roles.decorator.ts +++ b/src/auth/decorators/roles.decorator.ts @@ -1,5 +1,4 @@ import { SetMetadata } from '@nestjs/common'; import { UserRole } from '../../users/entities/user.entity'; - export const ROLES_KEY = 'roles'; export const Roles = (...roles: UserRole[]) => SetMetadata(ROLES_KEY, roles); diff --git a/src/auth/dto/auth.dto.ts b/src/auth/dto/auth.dto.ts index 0db261a6..53ef5a8c 100644 --- a/src/auth/dto/auth.dto.ts +++ b/src/auth/dto/auth.dto.ts @@ -2,87 +2,73 @@ import { IsEmail, IsString, IsEnum, IsOptional, IsNotEmpty } from 'class-validat import { ApiProperty } from '@nestjs/swagger'; import { UserRole } from '../../users/entities/user.entity'; import { IsStrongPassword } from '../../common/validators/password.validator'; - export class RegisterDto { - @ApiProperty({ example: 'john.doe@example.com' }) - @IsEmail({}, { message: 'Must be a valid email address' }) - @IsNotEmpty({ message: 'Email is required' }) - email: string; - - @ApiProperty({ example: 'StrongPass123!' }) - @IsString({ message: 'Password must be a string' }) - @IsStrongPassword({ message: 'Password must be stronger' }) - password: string; - - @ApiProperty({ example: 'John' }) - @IsString({ message: 'First name must be a string' }) - @IsNotEmpty({ message: 'First name is required' }) - firstName: string; - - @ApiProperty({ example: 'Doe' }) - @IsString({ message: 'Last name must be a string' }) - @IsNotEmpty({ message: 'Last name is required' }) - lastName: string; - - @ApiProperty({ enum: UserRole, required: false, default: UserRole.STUDENT }) - @IsOptional() - @IsEnum(UserRole, { message: 'Role must be a valid enum value' }) - role?: UserRole; + @ApiProperty({ example: 'john.doe@example.com' }) + @IsEmail({}, { message: 'Must be a valid email address' }) + @IsNotEmpty({ message: 'Email is required' }) + email: string; + @ApiProperty({ example: 'StrongPass123!' }) + @IsString({ message: 'Password must be a string' }) + @IsStrongPassword({ message: 'Password must be stronger' }) + password: string; + @ApiProperty({ example: 'John' }) + @IsString({ message: 'First name must be a string' }) + @IsNotEmpty({ message: 'First name is required' }) + firstName: string; + @ApiProperty({ example: 'Doe' }) + @IsString({ message: 'Last name must be a string' }) + @IsNotEmpty({ message: 'Last name is required' }) + lastName: string; + @ApiProperty({ enum: UserRole, required: false, default: UserRole.STUDENT }) + @IsOptional() + @IsEnum(UserRole, { message: 'Role must be a valid enum value' }) + role?: UserRole; } - export class LoginDto { - @ApiProperty({ example: 'john.doe@example.com' }) - @IsEmail({}, { message: 'Must be a valid email address' }) - @IsNotEmpty({ message: 'Email is required' }) - email: string; - - @ApiProperty({ example: 'StrongPass123!' }) - @IsString() - @IsNotEmpty({ message: 'Password is required' }) - password: string; + @ApiProperty({ example: 'john.doe@example.com' }) + @IsEmail({}, { message: 'Must be a valid email address' }) + @IsNotEmpty({ message: 'Email is required' }) + email: string; + @ApiProperty({ example: 'StrongPass123!' }) + @IsString() + @IsNotEmpty({ message: 'Password is required' }) + password: string; } - export class RefreshTokenDto { - @ApiProperty() - @IsString() - @IsNotEmpty({ message: 'Refresh token is required' }) - refreshToken: string; + @ApiProperty() + @IsString() + @IsNotEmpty({ message: 'Refresh token is required' }) + refreshToken: string; } - export class ForgotPasswordDto { - @ApiProperty({ example: 'john.doe@example.com' }) - @IsEmail({}, { message: 'Must be a valid email address' }) - @IsNotEmpty({ message: 'Email is required' }) - email: string; + @ApiProperty({ example: 'john.doe@example.com' }) + @IsEmail({}, { message: 'Must be a valid email address' }) + @IsNotEmpty({ message: 'Email is required' }) + email: string; } - export class ResetPasswordDto { - @ApiProperty() - @IsString() - @IsNotEmpty({ message: 'Token is required' }) - token: string; - - @ApiProperty({ example: 'NewStrongPass123!' }) - @IsString() - @IsStrongPassword() - newPassword: string; + @ApiProperty() + @IsString() + @IsNotEmpty({ message: 'Token is required' }) + token: string; + @ApiProperty({ example: 'NewStrongPass123!' }) + @IsString() + @IsStrongPassword() + newPassword: string; } - export class ChangePasswordDto { - @ApiProperty({ example: 'OldPass123!' }) - @IsString() - @IsNotEmpty({ message: 'Current password is required' }) - currentPassword: string; - - @ApiProperty({ example: 'NewPass123!' }) - @IsString() - @IsStrongPassword() - newPassword: string; + @ApiProperty({ example: 'OldPass123!' }) + @IsString() + @IsNotEmpty({ message: 'Current password is required' }) + currentPassword: string; + @ApiProperty({ example: 'NewPass123!' }) + @IsString() + @IsStrongPassword() + newPassword: string; } - export class VerifyEmailDto { - @ApiProperty() - @IsString() - @IsNotEmpty({ message: 'Token is required' }) - token: string; + @ApiProperty() + @IsString() + @IsNotEmpty({ message: 'Token is required' }) + token: string; } diff --git a/src/auth/guards/jwt-auth.guard.ts b/src/auth/guards/jwt-auth.guard.ts index 91447a67..f50f2a8f 100644 --- a/src/auth/guards/jwt-auth.guard.ts +++ b/src/auth/guards/jwt-auth.guard.ts @@ -1,6 +1,6 @@ import { Injectable } from '@nestjs/common'; import { AuthGuard } from '@nestjs/passport'; import { AUTH_STRATEGY } from '../../common/constants/auth.constants'; - @Injectable() -export class JwtAuthGuard extends AuthGuard(AUTH_STRATEGY.JWT) {} +export class JwtAuthGuard extends AuthGuard(AUTH_STRATEGY.JWT) { +} diff --git a/src/auth/guards/roles.guard.ts b/src/auth/guards/roles.guard.ts index bf7aeac7..52197c4d 100644 --- a/src/auth/guards/roles.guard.ts +++ b/src/auth/guards/roles.guard.ts @@ -2,22 +2,18 @@ import { Injectable, CanActivate, ExecutionContext } from '@nestjs/common'; import { Reflector } from '@nestjs/core'; import { ROLES_KEY } from '../decorators/roles.decorator'; import { UserRole } from '../../users/entities/user.entity'; - @Injectable() export class RolesGuard implements CanActivate { - constructor(private reflector: Reflector) {} - - canActivate(context: ExecutionContext): boolean { - const requiredRoles = this.reflector.getAllAndOverride(ROLES_KEY, [ - context.getHandler(), - context.getClass(), - ]); - - if (!requiredRoles) { - return true; + constructor(private reflector: Reflector) { } + canActivate(context: ExecutionContext): boolean { + const requiredRoles = this.reflector.getAllAndOverride(ROLES_KEY, [ + context.getHandler(), + context.getClass(), + ]); + if (!requiredRoles) { + return true; + } + const { user } = context.switchToHttp().getRequest(); + return requiredRoles.some((role) => user?.role === role); } - - const { user } = context.switchToHttp().getRequest(); - return requiredRoles.some((role) => user?.role === role); - } } diff --git a/src/auth/guards/ws-jwt-auth.guard.ts b/src/auth/guards/ws-jwt-auth.guard.ts index 67c7920a..348f1d6d 100644 --- a/src/auth/guards/ws-jwt-auth.guard.ts +++ b/src/auth/guards/ws-jwt-auth.guard.ts @@ -3,107 +3,93 @@ import { JwtService } from '@nestjs/jwt'; import { ConfigService } from '@nestjs/config'; import { Socket } from 'socket.io'; import * as jwt from 'jsonwebtoken'; - @Injectable() export class WsJwtAuthGuard implements CanActivate { - constructor( - private readonly jwtService: JwtService, - private readonly configService: ConfigService, - ) {} - - async canActivate(context: ExecutionContext): Promise { - const client: Socket = context.switchToWs().getClient(); - const token = - client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; - - if (!token) { - client.disconnect(true); - return false; + constructor(private readonly jwtService: JwtService, private readonly configService: ConfigService) { } + async canActivate(context: ExecutionContext): Promise { + const client: Socket = context.switchToWs().getClient(); + const token = client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; + if (!token) { + client.disconnect(true); + return false; + } + try { + const payload = await this.verifyAccessToken(token); + (client as unknown).user = payload; // attach user context + return true; + } + catch { + client.disconnect(true); + return false; + } } - - try { - const payload = await this.verifyAccessToken(token); - (client as any).user = payload; // attach user context - return true; - } catch { - client.disconnect(true); - return false; + private async verifyAccessToken(token: string): Promise { + const { secrets } = this.getJwtAccessSecrets(); + const kid = this.getTokenKid(token); + if (kid && secrets[kid]) { + return this.jwtService.verifyAsync(token, { secret: secrets[kid] }); + } + const secretList = Object.values(secrets); + if (secretList.length === 0) { + return this.jwtService.verifyAsync(token, { + secret: this.configService.get('JWT_SECRET') || 'your-secret-key', + }); + } + let lastError: unknown; + for (const secret of secretList) { + try { + return await this.jwtService.verifyAsync(token, { secret }); + } + catch (err) { + lastError = err; + } + } + throw lastError; } - } - - private async verifyAccessToken(token: string): Promise { - const { secrets } = this.getJwtAccessSecrets(); - const kid = this.getTokenKid(token); - - if (kid && secrets[kid]) { - return this.jwtService.verifyAsync(token, { secret: secrets[kid] }); + private getTokenKid(token: string): string | null { + const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; + const kid = decoded && typeof decoded === 'object' ? (decoded.header as unknown)?.kid : undefined; + return typeof kid === 'string' ? kid : null; } - - const secretList = Object.values(secrets); - if (secretList.length === 0) { - return this.jwtService.verifyAsync(token, { - secret: this.configService.get('JWT_SECRET') || 'your-secret-key', - }); + private getJwtAccessSecrets(): { + currentVersion: string | null; + secrets: Record; + } { + const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); + const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; + if (!jwtSecretsRaw) { + const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; + return { + currentVersion, + secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, + }; + } + return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; } - - let lastError: unknown; - for (const secret of secretList) { - try { - return await this.jwtService.verifyAsync(token, { secret }); - } catch (err) { - lastError = err; - } + private parseJwtSecrets(raw: string): Record { + try { + const parsed = JSON.parse(raw) as unknown; + if (parsed && typeof parsed === 'object') { + return parsed as Record; + } + } + catch { + // ignore + } + return raw + .split(',') + .map((pair) => pair.trim()) + .filter(Boolean) + .reduce>((acc, pair) => { + const idx = pair.indexOf(':'); + if (idx <= 0) + return acc; + const version = pair.slice(0, idx).trim(); + const secret = pair.slice(idx + 1).trim(); + if (!version || !secret) + return acc; + acc[version] = secret; + return acc; + }, {}); } - - throw lastError; - } - - private getTokenKid(token: string): string | null { - const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; - const kid = decoded && typeof decoded === 'object' ? (decoded.header as any)?.kid : undefined; - return typeof kid === 'string' ? kid : null; - } - - private getJwtAccessSecrets(): { - currentVersion: string | null; - secrets: Record; - } { - const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); - const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; - - if (!jwtSecretsRaw) { - const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; - return { - currentVersion, - secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, - }; - } - - return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; - } - - private parseJwtSecrets(raw: string): Record { - try { - const parsed = JSON.parse(raw) as unknown; - if (parsed && typeof parsed === 'object') { - return parsed as Record; - } - } catch { - // ignore - } - - return raw - .split(',') - .map((pair) => pair.trim()) - .filter(Boolean) - .reduce>((acc, pair) => { - const idx = pair.indexOf(':'); - if (idx <= 0) return acc; - const version = pair.slice(0, idx).trim(); - const secret = pair.slice(idx + 1).trim(); - if (!version || !secret) return acc; - acc[version] = secret; - return acc; - }, {}); - } } diff --git a/src/auth/jwt.strategy.ts b/src/auth/jwt.strategy.ts index a21a8386..e8dce9b8 100644 --- a/src/auth/jwt.strategy.ts +++ b/src/auth/jwt.strategy.ts @@ -6,159 +6,138 @@ import { JwtService } from '@nestjs/jwt'; import type { Request } from 'express'; import * as jwt from 'jsonwebtoken'; import { UserRole } from '../users/entities/user.entity'; - interface JwtPayload { - sub: string; - email: string; - role: UserRole; - sid: string; + sub: string; + email: string; + role: UserRole; + sid: string; } - interface ValidatedUser { - sub: string; - email: string; - role: UserRole; - sid: string; + sub: string; + email: string; + role: UserRole; + sid: string; } - @Injectable() export class JwtStrategy extends PassportStrategy(Strategy) { - constructor( - private readonly configService: ConfigService, - private readonly jwtService: JwtService, - ) { - super({ - jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(), - ignoreExpiration: false, - passReqToCallback: true, - secretOrKeyProvider: ( - req: Request, - rawJwtToken: string, - done: (err: any, secret?: string) => void, - ) => { - const { secrets } = this.getJwtAccessSecrets(); - const decoded = jwt.decode(rawJwtToken, { complete: true }) as jwt.Jwt | null; - const kid = - decoded && typeof decoded === 'object' ? (decoded.header as any)?.kid : undefined; - - if (kid && secrets[kid]) { - (req as any).jwtAccessSecretVersionUsed = kid; - return done(null, secrets[kid]); - } - - const entries = Object.entries(secrets); - if (entries.length === 0) { - (req as any).jwtAccessSecretVersionUsed = null; - return done(null, this.configService.get('JWT_SECRET') || 'your-secret-key'); + constructor(private readonly configService: ConfigService, private readonly jwtService: JwtService) { + super({ + jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(), + ignoreExpiration: false, + passReqToCallback: true, + secretOrKeyProvider: (req: Request, rawJwtToken: string, done: (err: unknown, secret?: string) => void) => { + const { secrets } = this.getJwtAccessSecrets(); + const decoded = jwt.decode(rawJwtToken, { complete: true }) as jwt.Jwt | null; + const kid = decoded && typeof decoded === 'object' ? (decoded.header as unknown)?.kid : undefined; + if (kid && secrets[kid]) { + (req as unknown).jwtAccessSecretVersionUsed = kid; + return done(null, secrets[kid]); + } + const entries = Object.entries(secrets); + if (entries.length === 0) { + (req as unknown).jwtAccessSecretVersionUsed = null; + return done(null, this.configService.get('JWT_SECRET') || 'your-secret-key'); + } + for (const [version, secret] of entries) { + try { + jwt.verify(rawJwtToken, secret); + (req as unknown).jwtAccessSecretVersionUsed = version; + return done(null, secret); + } + catch { + // try next + } + } + (req as unknown).jwtAccessSecretVersionUsed = null; + return done(null, entries[0][1]); + }, + }); + } + async validate(req: Request, payload: JwtPayload): Promise { + const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); + const token = this.extractBearerToken(req); + const tokenKid = token ? this.getTokenKid(token) : null; + const usedVersion = (req as unknown).jwtAccessSecretVersionUsed ?? tokenKid; + const shouldReissue = !!currentVersion && !!usedVersion && usedVersion !== currentVersion; + if (shouldReissue && (req as unknown)?.res) { + const newAccessToken = await this.jwtService.signAsync({ + sub: payload.sub, + email: payload.email, + role: payload.role, + sid: payload.sid, + }, { + secret: currentSecret, + expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), + header: { kid: currentVersion }, + }); + (req as unknown).res.setHeader('x-access-token', newAccessToken); } - - for (const [version, secret] of entries) { - try { - jwt.verify(rawJwtToken, secret); - (req as any).jwtAccessSecretVersionUsed = version; - return done(null, secret); - } catch { - // try next - } + return { sub: payload.sub, email: payload.email, role: payload.role, sid: payload.sid }; + } + private extractBearerToken(req: Request): string | null { + const header = req.headers.authorization; + if (!header) + return null; + const [type, token] = header.split(' '); + if (!token || type.toLowerCase() !== 'bearer') + return null; + return token; + } + private getTokenKid(token: string): string | null { + const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; + const kid = decoded && typeof decoded === 'object' ? (decoded.header as unknown)?.kid : undefined; + return typeof kid === 'string' ? kid : null; + } + private getJwtAccessSecrets(): { + currentVersion: string | null; + secrets: Record; + } { + const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); + const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; + if (!jwtSecretsRaw) { + const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; + return { + currentVersion, + secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, + }; } - - (req as any).jwtAccessSecretVersionUsed = null; - return done(null, entries[0][1]); - }, - }); - } - - async validate(req: Request, payload: JwtPayload): Promise { - const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); - const token = this.extractBearerToken(req); - const tokenKid = token ? this.getTokenKid(token) : null; - const usedVersion = (req as any).jwtAccessSecretVersionUsed ?? tokenKid; - const shouldReissue = !!currentVersion && !!usedVersion && usedVersion !== currentVersion; - - if (shouldReissue && (req as any)?.res) { - const newAccessToken = await this.jwtService.signAsync( - { - sub: payload.sub, - email: payload.email, - role: payload.role, - sid: payload.sid, - }, - { - secret: currentSecret, - expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), - header: { kid: currentVersion }, - }, - ); - (req as any).res.setHeader('x-access-token', newAccessToken); + return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; } - - return { sub: payload.sub, email: payload.email, role: payload.role, sid: payload.sid }; - } - - private extractBearerToken(req: Request): string | null { - const header = req.headers.authorization; - if (!header) return null; - const [type, token] = header.split(' '); - if (!token || type.toLowerCase() !== 'bearer') return null; - return token; - } - - private getTokenKid(token: string): string | null { - const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; - const kid = decoded && typeof decoded === 'object' ? (decoded.header as any)?.kid : undefined; - return typeof kid === 'string' ? kid : null; - } - - private getJwtAccessSecrets(): { - currentVersion: string | null; - secrets: Record; - } { - const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); - const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; - - if (!jwtSecretsRaw) { - const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; - return { - currentVersion, - secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, - }; + private getCurrentJwtAccessSecret(): { + currentVersion: string | null; + currentSecret: string; + } { + const { currentVersion, secrets } = this.getJwtAccessSecrets(); + const currentSecret = (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); + return { + currentVersion, + currentSecret: currentSecret || Object.values(secrets)[0] || 'your-secret-key', + }; } - - return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; - } - - private getCurrentJwtAccessSecret(): { currentVersion: string | null; currentSecret: string } { - const { currentVersion, secrets } = this.getJwtAccessSecrets(); - const currentSecret = - (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); - return { - currentVersion, - currentSecret: currentSecret || Object.values(secrets)[0] || 'your-secret-key', - }; - } - - private parseJwtSecrets(raw: string): Record { - try { - const parsed = JSON.parse(raw) as unknown; - if (parsed && typeof parsed === 'object') { - return parsed as Record; - } - } catch { - // ignore + private parseJwtSecrets(raw: string): Record { + try { + const parsed = JSON.parse(raw) as unknown; + if (parsed && typeof parsed === 'object') { + return parsed as Record; + } + } + catch { + // ignore + } + return raw + .split(',') + .map((pair) => pair.trim()) + .filter(Boolean) + .reduce>((acc, pair) => { + const idx = pair.indexOf(':'); + if (idx <= 0) + return acc; + const version = pair.slice(0, idx).trim(); + const secret = pair.slice(idx + 1).trim(); + if (!version || !secret) + return acc; + acc[version] = secret; + return acc; + }, {}); } - - return raw - .split(',') - .map((pair) => pair.trim()) - .filter(Boolean) - .reduce>((acc, pair) => { - const idx = pair.indexOf(':'); - if (idx <= 0) return acc; - const version = pair.slice(0, idx).trim(); - const secret = pair.slice(idx + 1).trim(); - if (!version || !secret) return acc; - acc[version] = secret; - return acc; - }, {}); - } } diff --git a/src/auth/strategies/jwt.strategy.ts b/src/auth/strategies/jwt.strategy.ts index a85beae1..97172c4c 100644 --- a/src/auth/strategies/jwt.strategy.ts +++ b/src/auth/strategies/jwt.strategy.ts @@ -5,163 +5,142 @@ import { ConfigService } from '@nestjs/config'; import { JwtService } from '@nestjs/jwt'; import type { Request } from 'express'; import * as jwt from 'jsonwebtoken'; - export interface JwtPayload { - sub: string; - email: string; - role?: string; - roles?: string[]; - sid?: string; + sub: string; + email: string; + role?: string; + roles?: string[]; + sid?: string; } - @Injectable() export class JwtStrategy extends PassportStrategy(Strategy) { - constructor( - private readonly configService: ConfigService, - private readonly jwtService: JwtService, - ) { - super({ - jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(), - ignoreExpiration: false, - passReqToCallback: true, - secretOrKeyProvider: ( - req: Request, - rawJwtToken: string, - done: (err: any, secret?: string) => void, - ) => { - const { secrets } = this.getJwtAccessSecrets(); - const decoded = jwt.decode(rawJwtToken, { complete: true }) as jwt.Jwt | null; - const kid = - decoded && typeof decoded === 'object' ? (decoded.header as any)?.kid : undefined; - - if (kid && secrets[kid]) { - (req as any).jwtAccessSecretVersionUsed = kid; - return done(null, secrets[kid]); - } - - // No/unknown kid: probe each secret until one verifies the token signature. - const entries = Object.entries(secrets); - if (entries.length === 0) { - (req as any).jwtAccessSecretVersionUsed = null; - return done(null, this.configService.get('JWT_SECRET') || 'your-secret-key'); + constructor(private readonly configService: ConfigService, private readonly jwtService: JwtService) { + super({ + jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(), + ignoreExpiration: false, + passReqToCallback: true, + secretOrKeyProvider: (req: Request, rawJwtToken: string, done: (err: unknown, secret?: string) => void) => { + const { secrets } = this.getJwtAccessSecrets(); + const decoded = jwt.decode(rawJwtToken, { complete: true }) as jwt.Jwt | null; + const kid = decoded && typeof decoded === 'object' ? (decoded.header as unknown)?.kid : undefined; + if (kid && secrets[kid]) { + (req as unknown).jwtAccessSecretVersionUsed = kid; + return done(null, secrets[kid]); + } + // No/unknown kid: probe each secret until one verifies the token signature. + const entries = Object.entries(secrets); + if (entries.length === 0) { + (req as unknown).jwtAccessSecretVersionUsed = null; + return done(null, this.configService.get('JWT_SECRET') || 'your-secret-key'); + } + for (const [version, secret] of entries) { + try { + jwt.verify(rawJwtToken, secret); + (req as unknown).jwtAccessSecretVersionUsed = version; + return done(null, secret); + } + catch { + // try next + } + } + // Fall back to first secret so passport-jwt returns a consistent 401 on bad signature. + (req as unknown).jwtAccessSecretVersionUsed = null; + return done(null, entries[0][1]); + }, + }); + } + async validate(req: Request, payload: JwtPayload) { + const roles = payload.roles || (payload.role ? [payload.role] : []); + const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); + const token = this.extractBearerToken(req); + const tokenKid = token ? this.getTokenKid(token) : null; + const usedVersion = (req as unknown).jwtAccessSecretVersionUsed ?? tokenKid; + const shouldReissue = !!currentVersion && !!usedVersion && usedVersion !== currentVersion; + if (shouldReissue && (req as unknown)?.res) { + const newAccessToken = await this.jwtService.signAsync({ + sub: payload.sub, + email: payload.email, + role: roles[0], + roles, + sid: payload.sid, + }, { + secret: currentSecret, + expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), + header: { kid: currentVersion }, + }); + (req as unknown).res.setHeader('x-access-token', newAccessToken); } - - for (const [version, secret] of entries) { - try { - jwt.verify(rawJwtToken, secret); - (req as any).jwtAccessSecretVersionUsed = version; - return done(null, secret); - } catch { - // try next - } + return { + userId: payload.sub, + email: payload.email, + roles, + sessionId: payload.sid, + }; + } + private extractBearerToken(req: Request): string | null { + const header = req.headers.authorization; + if (!header) + return null; + const [type, token] = header.split(' '); + if (!token || type.toLowerCase() !== 'bearer') + return null; + return token; + } + private getTokenKid(token: string): string | null { + const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; + const kid = decoded && typeof decoded === 'object' ? (decoded.header as unknown)?.kid : undefined; + return typeof kid === 'string' ? kid : null; + } + private getJwtAccessSecrets(): { + currentVersion: string | null; + secrets: Record; + } { + const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); + const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; + if (!jwtSecretsRaw) { + const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; + return { + currentVersion, + secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, + }; } - - // Fall back to first secret so passport-jwt returns a consistent 401 on bad signature. - (req as any).jwtAccessSecretVersionUsed = null; - return done(null, entries[0][1]); - }, - }); - } - - async validate(req: Request, payload: JwtPayload) { - const roles = payload.roles || (payload.role ? [payload.role] : []); - - const { currentVersion, currentSecret } = this.getCurrentJwtAccessSecret(); - const token = this.extractBearerToken(req); - const tokenKid = token ? this.getTokenKid(token) : null; - const usedVersion = (req as any).jwtAccessSecretVersionUsed ?? tokenKid; - const shouldReissue = !!currentVersion && !!usedVersion && usedVersion !== currentVersion; - - if (shouldReissue && (req as any)?.res) { - const newAccessToken = await this.jwtService.signAsync( - { - sub: payload.sub, - email: payload.email, - role: roles[0], - roles, - sid: payload.sid, - }, - { - secret: currentSecret, - expiresIn: parseInt(this.configService.get('JWT_EXPIRES_IN') || '900', 10), - header: { kid: currentVersion }, - }, - ); - (req as any).res.setHeader('x-access-token', newAccessToken); + return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; } - - return { - userId: payload.sub, - email: payload.email, - roles, - sessionId: payload.sid, - }; - } - - private extractBearerToken(req: Request): string | null { - const header = req.headers.authorization; - if (!header) return null; - const [type, token] = header.split(' '); - if (!token || type.toLowerCase() !== 'bearer') return null; - return token; - } - - private getTokenKid(token: string): string | null { - const decoded = jwt.decode(token, { complete: true }) as jwt.Jwt | null; - const kid = decoded && typeof decoded === 'object' ? (decoded.header as any)?.kid : undefined; - return typeof kid === 'string' ? kid : null; - } - - private getJwtAccessSecrets(): { - currentVersion: string | null; - secrets: Record; - } { - const jwtSecretsRaw = this.configService.get('JWT_SECRETS'); - const currentVersion = this.configService.get('JWT_SECRET_CURRENT_VERSION') || null; - - if (!jwtSecretsRaw) { - const secret = this.configService.get('JWT_SECRET') || 'your-secret-key'; - return { - currentVersion, - secrets: currentVersion ? { [currentVersion]: secret } : { default: secret }, - }; + private getCurrentJwtAccessSecret(): { + currentVersion: string | null; + currentSecret: string; + } { + const { currentVersion, secrets } = this.getJwtAccessSecrets(); + const currentSecret = (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); + return { + currentVersion, + currentSecret: currentSecret || Object.values(secrets)[0] || 'your-secret-key', + }; } - - return { currentVersion, secrets: this.parseJwtSecrets(jwtSecretsRaw) }; - } - - private getCurrentJwtAccessSecret(): { currentVersion: string | null; currentSecret: string } { - const { currentVersion, secrets } = this.getJwtAccessSecrets(); - const currentSecret = - (currentVersion && secrets[currentVersion]) || this.configService.get('JWT_SECRET'); - return { - currentVersion, - currentSecret: currentSecret || Object.values(secrets)[0] || 'your-secret-key', - }; - } - - private parseJwtSecrets(raw: string): Record { - try { - const parsed = JSON.parse(raw) as unknown; - if (parsed && typeof parsed === 'object') { - return parsed as Record; - } - } catch { - // ignore + private parseJwtSecrets(raw: string): Record { + try { + const parsed = JSON.parse(raw) as unknown; + if (parsed && typeof parsed === 'object') { + return parsed as Record; + } + } + catch { + // ignore + } + return raw + .split(',') + .map((pair) => pair.trim()) + .filter(Boolean) + .reduce>((acc, pair) => { + const idx = pair.indexOf(':'); + if (idx <= 0) + return acc; + const version = pair.slice(0, idx).trim(); + const secret = pair.slice(idx + 1).trim(); + if (!version || !secret) + return acc; + acc[version] = secret; + return acc; + }, {}); } - - return raw - .split(',') - .map((pair) => pair.trim()) - .filter(Boolean) - .reduce>((acc, pair) => { - const idx = pair.indexOf(':'); - if (idx <= 0) return acc; - const version = pair.slice(0, idx).trim(); - const secret = pair.slice(idx + 1).trim(); - if (!version || !secret) return acc; - acc[version] = secret; - return acc; - }, {}); - } } diff --git a/src/backup/backup.controller.ts b/src/backup/backup.controller.ts index 31be6689..40a955c0 100644 --- a/src/backup/backup.controller.ts +++ b/src/backup/backup.controller.ts @@ -1,14 +1,4 @@ -import { - Controller, - Post, - Get, - Param, - Body, - ParseUUIDPipe, - HttpCode, - HttpStatus, - UseGuards, -} from '@nestjs/common'; +import { Controller, Post, Get, Param, Body, ParseUUIDPipe, HttpCode, HttpStatus, UseGuards, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiBearerAuth, ApiResponse } from '@nestjs/swagger'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RecoveryTestingService } from './testing/recovery-testing.service'; @@ -17,51 +7,51 @@ import { BackupMonitoringService } from './monitoring/backup-monitoring.service' import { RestoreBackupDto } from './dto/restore-backup.dto'; import { TriggerRecoveryTestDto } from './dto/trigger-recovery-test.dto'; import { RecoveryTestResponseDto } from './dto/recovery-test-response.dto'; - @ApiTags('backup') @ApiBearerAuth() @UseGuards(JwtAuthGuard) @Controller('backup') export class BackupController { - constructor( - private readonly recoveryTestingService: RecoveryTestingService, - private readonly disasterRecoveryService: DisasterRecoveryService, - private readonly backupMonitoringService: BackupMonitoringService, - ) {} - - @Post('restore') - @ApiOperation({ summary: 'Restore from backup' }) - @ApiResponse({ status: HttpStatus.ACCEPTED, description: 'Restore initiated' }) - @HttpCode(HttpStatus.ACCEPTED) - async restoreBackup(@Body() dto: RestoreBackupDto): Promise<{ message: string }> { - await this.disasterRecoveryService.executeRestore(dto.backupRecordId); - return { message: 'Restore initiated' }; - } - - @Post('test') - @ApiOperation({ summary: 'Trigger recovery test' }) - @ApiResponse({ status: HttpStatus.OK, description: 'Recovery test triggered' }) - async triggerRecoveryTest(@Body() dto: TriggerRecoveryTestDto): Promise { - return this.recoveryTestingService.createRecoveryTest(dto.backupRecordId); - } - - @Get('test/:testId') - @ApiOperation({ summary: 'Get recovery test results' }) - @ApiResponse({ - status: HttpStatus.OK, - description: 'Recovery test results', - type: RecoveryTestResponseDto, - }) - async getRecoveryTest( - @Param('testId', ParseUUIDPipe) testId: string, - ): Promise { - return this.recoveryTestingService.getTestResults(testId); - } - - @Get('health') - @ApiOperation({ summary: 'Get backup system health' }) - @ApiResponse({ status: HttpStatus.OK, description: 'Backup health status' }) - async getBackupHealth(): Promise<{ healthy: boolean; issues: string[] }> { - return this.backupMonitoringService.checkBackupHealth(); - } + constructor(private readonly recoveryTestingService: RecoveryTestingService, private readonly disasterRecoveryService: DisasterRecoveryService, private readonly backupMonitoringService: BackupMonitoringService) { } + @Post('restore') + @ApiOperation({ summary: 'Restore from backup' }) + @ApiResponse({ status: HttpStatus.ACCEPTED, description: 'Restore initiated' }) + @HttpCode(HttpStatus.ACCEPTED) + async restoreBackup( + @Body() + dto: RestoreBackupDto): Promise<{ + message: string; + }> { + await this.disasterRecoveryService.executeRestore(dto.backupRecordId); + return { message: 'Restore initiated' }; + } + @Post('test') + @ApiOperation({ summary: 'Trigger recovery test' }) + @ApiResponse({ status: HttpStatus.OK, description: 'Recovery test triggered' }) + async triggerRecoveryTest( + @Body() + dto: TriggerRecoveryTestDto): Promise { + return this.recoveryTestingService.createRecoveryTest(dto.backupRecordId); + } + @Get('test/:testId') + @ApiOperation({ summary: 'Get recovery test results' }) + @ApiResponse({ + status: HttpStatus.OK, + description: 'Recovery test results', + type: RecoveryTestResponseDto, + }) + async getRecoveryTest( + @Param('testId', ParseUUIDPipe) + testId: string): Promise { + return this.recoveryTestingService.getTestResults(testId); + } + @Get('health') + @ApiOperation({ summary: 'Get backup system health' }) + @ApiResponse({ status: HttpStatus.OK, description: 'Backup health status' }) + async getBackupHealth(): Promise<{ + healthy: boolean; + issues: string[]; + }> { + return this.backupMonitoringService.checkBackupHealth(); + } } diff --git a/src/backup/backup.module.ts b/src/backup/backup.module.ts index 0d8ee817..bfd9254f 100644 --- a/src/backup/backup.module.ts +++ b/src/backup/backup.module.ts @@ -4,48 +4,43 @@ import { BullModule } from '@nestjs/bull'; import { ConfigModule } from '@nestjs/config'; import { ScheduleModule } from '@nestjs/schedule'; import { QUEUE_NAMES } from '../common/constants/queue.constants'; - // Entities import { BackupRecord } from './entities/backup-record.entity'; import { RecoveryTest } from './entities/recovery-test.entity'; - // Services import { BackupService } from './backup.service'; import { DisasterRecoveryService } from './disaster-recovery/disaster-recovery.service'; import { DataIntegrityService } from './integrity/data-integrity.service'; import { RecoveryTestingService } from './testing/recovery-testing.service'; import { BackupMonitoringService } from './monitoring/backup-monitoring.service'; - // Controller import { BackupController } from './backup.controller'; - // Processor import { BackupQueueProcessor } from './processing/backup-queue.processor'; - // External modules import { MediaModule } from '../media/media.module'; import { MonitoringModule } from '../monitoring/monitoring.module'; - @Module({ - imports: [ - ConfigModule, - ScheduleModule.forRoot(), - TypeOrmModule.forFeature([BackupRecord, RecoveryTest]), - BullModule.registerQueue({ - name: QUEUE_NAMES.BACKUP_PROCESSING, - }), - MediaModule, // For FileStorageService - MonitoringModule, // For AlertingService and MetricsCollectionService - ], - controllers: [BackupController], - providers: [ - BackupService, - DisasterRecoveryService, - DataIntegrityService, - RecoveryTestingService, - BackupMonitoringService, - BackupQueueProcessor, - ], - exports: [BackupService, DisasterRecoveryService], + imports: [ + ConfigModule, + ScheduleModule.forRoot(), + TypeOrmModule.forFeature([BackupRecord, RecoveryTest]), + BullModule.registerQueue({ + name: QUEUE_NAMES.BACKUP_PROCESSING, + }), + MediaModule, // For FileStorageService + MonitoringModule, // For AlertingService and MetricsCollectionService + ], + controllers: [BackupController], + providers: [ + BackupService, + DisasterRecoveryService, + DataIntegrityService, + RecoveryTestingService, + BackupMonitoringService, + BackupQueueProcessor, + ], + exports: [BackupService, DisasterRecoveryService], }) -export class BackupModule {} +export class BackupModule { +} diff --git a/src/backup/backup.service.spec.ts b/src/backup/backup.service.spec.ts index 79caeb4a..ba2c1bf0 100644 --- a/src/backup/backup.service.spec.ts +++ b/src/backup/backup.service.spec.ts @@ -5,538 +5,370 @@ import { BackupService } from './backup.service'; import { BackupRecord } from './entities/backup-record.entity'; import { BackupStatus, BackupType, Region } from './enums/backup-status.enum'; import { QUEUE_NAMES, JOB_NAMES } from '../common/constants/queue.constants'; -import { - createMockRepository, - createMockConfigService, - createMockQueue, -} from 'test/utils/mock-factories'; +import { createMockRepository, createMockConfigService, createMockQueue, } from 'test/utils/mock-factories'; import { Repository } from 'typeorm'; - describe('BackupService', () => { - // ───────────────────────────────────────────────────────────────────────── - // DECLARATIONS - // ───────────────────────────────────────────────────────────────────────── - - let service: BackupService; - let mockBackupRepo: jest.Mocked>; - let mockBackupQueue: jest.Mocked; - let mockConfigService: jest.Mocked; - let mockAlertingService: jest.Mocked; - let mockMetricsService: jest.Mocked; - let mockScheduledTaskMonitoringService: jest.Mocked; - - // ───────────────────────────────────────────────────────────────────────── - // SETUP & TEARDOWN - // ───────────────────────────────────────────────────────────────────────── - - beforeEach(async () => { - // Initialize all dependency mocks - mockBackupRepo = createMockRepository(); - mockBackupQueue = createMockQueue(); - mockConfigService = createMockConfigService({ - BACKUP_RETENTION_DAYS: 30, - BACKUP_SCHEDULED_TASK_RETRY_LIMIT: 2, - BACKUP_SCHEDULED_TASK_RETRY_DELAY_MS: 10000, - BACKUP_SCHEDULED_TASK_TIMEOUT_MS: 30 * 60 * 1000, - BACKUP_PRIMARY_REGION: Region.US_EAST_1, - DB_DATABASE: 'teachlink', + // ───────────────────────────────────────────────────────────────────────── + // DECLARATIONS + // ───────────────────────────────────────────────────────────────────────── + let service: BackupService; + let mockBackupRepo: jest.Mocked>; + let mockBackupQueue: jest.Mocked; + let mockConfigService: jest.Mocked; + let mockAlertingService: jest.Mocked; + let mockMetricsService: jest.Mocked; + let mockScheduledTaskMonitoringService: jest.Mocked; + // ───────────────────────────────────────────────────────────────────────── + // SETUP & TEARDOWN + // ───────────────────────────────────────────────────────────────────────── + beforeEach(async () => { + // Initialize all dependency mocks + mockBackupRepo = createMockRepository(); + mockBackupQueue = createMockQueue(); + mockConfigService = createMockConfigService({ + BACKUP_RETENTION_DAYS: 30, + BACKUP_SCHEDULED_TASK_RETRY_LIMIT: 2, + BACKUP_SCHEDULED_TASK_RETRY_DELAY_MS: 10000, + BACKUP_SCHEDULED_TASK_TIMEOUT_MS: 30 * 60 * 1000, + BACKUP_PRIMARY_REGION: Region.US_EAST_1, + DB_DATABASE: 'teachlink', + }); + mockAlertingService = { + sendAlert: jest.fn(), + }; + mockMetricsService = { + recordMetric: jest.fn(), + }; + mockScheduledTaskMonitoringService = { + registerTask: jest.fn(), + startExecution: jest.fn().mockReturnValue('execution-1'), + markSuccess: jest.fn(), + markFailure: jest.fn(), + recordRetry: jest.fn(), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + BackupService, + { + provide: getRepositoryToken(BackupRecord), + useValue: mockBackupRepo, + }, + { + provide: getQueueToken(QUEUE_NAMES.BACKUP_PROCESSING), + useValue: mockBackupQueue, + }, + { + provide: 'ConfigService', + useValue: mockConfigService, + }, + { + provide: 'AlertingService', + useValue: mockAlertingService, + }, + { + provide: 'MetricsCollectionService', + useValue: mockMetricsService, + }, + { + provide: 'ScheduledTaskMonitoringService', + useValue: mockScheduledTaskMonitoringService, + }, + ], + }).compile(); + service = module.get(BackupService); }); - mockAlertingService = { - sendAlert: jest.fn(), - }; - mockMetricsService = { - recordMetric: jest.fn(), - }; - mockScheduledTaskMonitoringService = { - registerTask: jest.fn(), - startExecution: jest.fn().mockReturnValue('execution-1'), - markSuccess: jest.fn(), - markFailure: jest.fn(), - recordRetry: jest.fn(), - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - BackupService, - { - provide: getRepositoryToken(BackupRecord), - useValue: mockBackupRepo, - }, - { - provide: getQueueToken(QUEUE_NAMES.BACKUP_PROCESSING), - useValue: mockBackupQueue, - }, - { - provide: 'ConfigService', - useValue: mockConfigService, - }, - { - provide: 'AlertingService', - useValue: mockAlertingService, - }, - { - provide: 'MetricsCollectionService', - useValue: mockMetricsService, - }, - { - provide: 'ScheduledTaskMonitoringService', - useValue: mockScheduledTaskMonitoringService, - }, - ], - }).compile(); - - service = module.get(BackupService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - // ───────────────────────────────────────────────────────────────────────── - // TEST SUITES - // ───────────────────────────────────────────────────────────────────────── - - describe('constructor', () => { - it('should initialize with default configuration values', () => { - const defaultConfigService = createMockConfigService({}); - const defaultService = new (BackupService as any)( - mockBackupRepo, - mockBackupQueue, - defaultConfigService, - mockAlertingService, - mockMetricsService, - mockScheduledTaskMonitoringService, - ); - - expect((defaultService as any).retentionDays).toBe(30); - expect((defaultService as any).scheduledTaskRetryLimit).toBe(2); - expect((defaultService as any).scheduledTaskRetryDelayMs).toBe(10000); - expect((defaultService as any).scheduledTaskTimeoutMs).toBe(30 * 60 * 1000); + afterEach(() => { + jest.clearAllMocks(); }); - - it('should register scheduled tasks', () => { - expect(mockScheduledTaskMonitoringService.registerTask).toHaveBeenCalledWith( - 'weekly-database-backup', - { - expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, - timeoutMs: 30 * 60 * 1000, - maxRetries: 2, - }, - ); - expect(mockScheduledTaskMonitoringService.registerTask).toHaveBeenCalledWith( - 'cleanup-expired-backups', - { - expectedIntervalMs: 24 * 60 * 60 * 1000, - timeoutMs: 30 * 60 * 1000, - maxRetries: 2, - }, - ); + // ───────────────────────────────────────────────────────────────────────── + // TEST SUITES + // ───────────────────────────────────────────────────────────────────────── + describe('constructor', () => { + it('should initialize with default configuration values', () => { + const defaultConfigService = createMockConfigService({}); + const defaultService = new (BackupService as unknown)(mockBackupRepo, mockBackupQueue, defaultConfigService, mockAlertingService, mockMetricsService, mockScheduledTaskMonitoringService); + expect((defaultService as unknown).retentionDays).toBe(30); + expect((defaultService as unknown).scheduledTaskRetryLimit).toBe(2); + expect((defaultService as unknown).scheduledTaskRetryDelayMs).toBe(10000); + expect((defaultService as unknown).scheduledTaskTimeoutMs).toBe(30 * 60 * 1000); + }); + it('should register scheduled tasks', () => { + expect(mockScheduledTaskMonitoringService.registerTask).toHaveBeenCalledWith('weekly-database-backup', { + expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, + timeoutMs: 30 * 60 * 1000, + maxRetries: 2, + }); + expect(mockScheduledTaskMonitoringService.registerTask).toHaveBeenCalledWith('cleanup-expired-backups', { + expectedIntervalMs: 24 * 60 * 60 * 1000, + timeoutMs: 30 * 60 * 1000, + maxRetries: 2, + }); + }); }); - }); - - describe('handleScheduledBackup', () => { - const mockBackupRecord = { - id: 'backup-1', - backupType: BackupType.FULL, - status: BackupStatus.PENDING, - region: Region.US_EAST_1, - databaseName: 'teachlink', - storageKey: '', - expiresAt: new Date(), - metadata: { startTime: new Date() }, - }; - - beforeEach(() => { - mockScheduledTaskMonitoringService.startExecution.mockReturnValue('execution-1'); - mockBackupRepo.create.mockReturnValue(mockBackupRecord as BackupRecord); - mockBackupRepo.save.mockResolvedValue(mockBackupRecord as BackupRecord); - mockBackupQueue.add.mockResolvedValue({} as any); - mockScheduledTaskMonitoringService.markSuccess.mockResolvedValue(); + describe('handleScheduledBackup', () => { + const mockBackupRecord = { + id: 'backup-1', + backupType: BackupType.FULL, + status: BackupStatus.PENDING, + region: Region.US_EAST_1, + databaseName: 'teachlink', + storageKey: '', + expiresAt: new Date(), + metadata: { startTime: new Date() }, + }; + beforeEach(() => { + mockScheduledTaskMonitoringService.startExecution.mockReturnValue('execution-1'); + mockBackupRepo.create.mockReturnValue(mockBackupRecord as BackupRecord); + mockBackupRepo.save.mockResolvedValue(mockBackupRecord as BackupRecord); + mockBackupQueue.add.mockResolvedValue({} as unknown); + mockScheduledTaskMonitoringService.markSuccess.mockResolvedValue(); + }); + it('should create and queue a scheduled backup', async () => { + await service.handleScheduledBackup(); + expect(mockScheduledTaskMonitoringService.startExecution).toHaveBeenCalledWith('weekly-database-backup', { + expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, + timeoutMs: 30 * 60 * 1000, + maxRetries: 2, + }, { source: 'BackupService' }); + expect(mockBackupRepo.create).toHaveBeenCalledWith(expect.objectContaining({ + backupType: BackupType.FULL, + status: BackupStatus.PENDING, + region: Region.US_EAST_1, + databaseName: 'teachlink', + storageKey: '', + metadata: expect.objectContaining({ + startTime: expect.any(Date), + }), + })); + expect(mockBackupQueue.add).toHaveBeenCalledWith(JOB_NAMES.CREATE_BACKUP, { + backupRecordId: 'backup-1', + backupType: BackupType.FULL, + region: Region.US_EAST_1, + databaseName: 'teachlink', + }, { + attempts: 3, + backoff: { + type: 'exponential', + delay: 10000, + }, + timeout: 3600000, + }); + expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith('execution-1', { + attempt: 1, + maxAttempts: 3, + retriesUsed: 0, + }); + }); + it('should handle backup creation failure and retry', async () => { + mockBackupRepo.save.mockRejectedValueOnce(new Error('Database error')); + mockBackupRepo.save.mockResolvedValueOnce(mockBackupRecord as BackupRecord); + await service.handleScheduledBackup(); + expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith('weekly-database-backup', 1, 2, 'Database error'); + expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith('execution-1', { + attempt: 2, + maxAttempts: 3, + retriesUsed: 1, + }); + }); + it('should handle complete failure after retries', async () => { + const error = new Error('Persistent error'); + mockBackupRepo.save.mockRejectedValue(error); + await service.handleScheduledBackup(); + expect(mockScheduledTaskMonitoringService.markFailure).toHaveBeenCalledWith('execution-1', 'Persistent error', { + attempt: 3, + maxAttempts: 3, + retriesUsed: 2, + }); + expect(mockAlertingService.sendAlert).toHaveBeenCalledWith('BACKUP_SCHEDULED_FAILED', 'Scheduled task weekly-database-backup failed after 3 attempt(s): Persistent error', 'CRITICAL'); + }); }); - - it('should create and queue a scheduled backup', async () => { - await service.handleScheduledBackup(); - - expect(mockScheduledTaskMonitoringService.startExecution).toHaveBeenCalledWith( - 'weekly-database-backup', - { - expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, - timeoutMs: 30 * 60 * 1000, - maxRetries: 2, - }, - { source: 'BackupService' }, - ); - - expect(mockBackupRepo.create).toHaveBeenCalledWith( - expect.objectContaining({ - backupType: BackupType.FULL, - status: BackupStatus.PENDING, - region: Region.US_EAST_1, - databaseName: 'teachlink', - storageKey: '', - metadata: expect.objectContaining({ - startTime: expect.any(Date), - }), - }), - ); - - expect(mockBackupQueue.add).toHaveBeenCalledWith( - JOB_NAMES.CREATE_BACKUP, - { - backupRecordId: 'backup-1', - backupType: BackupType.FULL, - region: Region.US_EAST_1, - databaseName: 'teachlink', - }, - { - attempts: 3, - backoff: { - type: 'exponential', - delay: 10000, - }, - timeout: 3600000, - }, - ); - - expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith( - 'execution-1', - { - attempt: 1, - maxAttempts: 3, - retriesUsed: 0, - }, - ); + describe('handleBackupCleanup', () => { + const mockExpiredBackups = [ + { id: 'backup-1', createdAt: new Date('2024-01-01') }, + { id: 'backup-2', createdAt: new Date('2024-01-02') }, + ]; + beforeEach(() => { + mockScheduledTaskMonitoringService.startExecution.mockReturnValue('execution-1'); + mockBackupRepo.find.mockResolvedValue(mockExpiredBackups as BackupRecord[]); + mockBackupQueue.add.mockResolvedValue({} as unknown); + mockScheduledTaskMonitoringService.markSuccess.mockResolvedValue(); + }); + it('should find and queue deletion of expired backups', async () => { + await service.handleBackupCleanup(); + expect(mockBackupRepo.find).toHaveBeenCalledWith({ + where: { + createdAt: expect.any(Object), // LessThan operator + status: BackupStatus.COMPLETED, + }, + }); + expect(mockBackupQueue.add).toHaveBeenCalledTimes(2); + expect(mockBackupQueue.add).toHaveBeenCalledWith(JOB_NAMES.DELETE_BACKUP, { backupRecordId: 'backup-1' }, { + attempts: 3, + backoff: { type: 'exponential', delay: 5000 }, + }); + expect(mockBackupQueue.add).toHaveBeenCalledWith(JOB_NAMES.DELETE_BACKUP, { backupRecordId: 'backup-2' }, { + attempts: 3, + backoff: { type: 'exponential', delay: 5000 }, + }); + expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith('execution-1', { + attempt: 1, + maxAttempts: 3, + retriesUsed: 0, + }); + }); + it('should handle cleanup failure and retry', async () => { + mockBackupRepo.find.mockRejectedValueOnce(new Error('Query error')); + mockBackupRepo.find.mockResolvedValueOnce([]); + await service.handleBackupCleanup(); + expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith('cleanup-expired-backups', 1, 2, 'Query error'); + }); }); - - it('should handle backup creation failure and retry', async () => { - mockBackupRepo.save.mockRejectedValueOnce(new Error('Database error')); - mockBackupRepo.save.mockResolvedValueOnce(mockBackupRecord as BackupRecord); - - await service.handleScheduledBackup(); - - expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith( - 'weekly-database-backup', - 1, - 2, - 'Database error', - ); - expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith( - 'execution-1', - { - attempt: 2, - maxAttempts: 3, - retriesUsed: 1, - }, - ); + describe('getLatestBackup', () => { + const mockBackup = { + id: 'backup-1', + status: BackupStatus.COMPLETED, + integrityVerified: true, + completedAt: new Date('2024-01-15'), + }; + beforeEach(() => { + mockBackupRepo.findOne.mockResolvedValue(mockBackup as BackupRecord); + }); + it('should return latest completed and verified backup', async () => { + const result = await service.getLatestBackup(); + expect(result).toEqual(mockBackup); + expect(mockBackupRepo.findOne).toHaveBeenCalledWith({ + where: { + status: BackupStatus.COMPLETED, + integrityVerified: true, + }, + order: { completedAt: 'DESC' }, + }); + }); + it('should filter by region when specified', async () => { + await service.getLatestBackup(Region.EU_WEST_1); + expect(mockBackupRepo.findOne).toHaveBeenCalledWith({ + where: { + status: BackupStatus.COMPLETED, + integrityVerified: true, + region: Region.EU_WEST_1, + }, + order: { completedAt: 'DESC' }, + }); + }); + it('should return null when no backup found', async () => { + mockBackupRepo.findOne.mockResolvedValue(null); + const result = await service.getLatestBackup(); + expect(result).toBeNull(); + }); }); - - it('should handle complete failure after retries', async () => { - const error = new Error('Persistent error'); - mockBackupRepo.save.mockRejectedValue(error); - - await service.handleScheduledBackup(); - - expect(mockScheduledTaskMonitoringService.markFailure).toHaveBeenCalledWith( - 'execution-1', - 'Persistent error', - { - attempt: 3, - maxAttempts: 3, - retriesUsed: 2, - }, - ); - - expect(mockAlertingService.sendAlert).toHaveBeenCalledWith( - 'BACKUP_SCHEDULED_FAILED', - 'Scheduled task weekly-database-backup failed after 3 attempt(s): Persistent error', - 'CRITICAL', - ); + describe('updateBackupStatus', () => { + const backupId = 'backup-1'; + const updates = { errorMessage: 'Test error' }; + beforeEach(() => { + mockBackupRepo.update.mockResolvedValue({ affected: 1, raw: {}, generatedMaps: [] }); + }); + it('should update backup status to completed and send success alert', async () => { + await service.updateBackupStatus(backupId, BackupStatus.COMPLETED, updates); + expect(mockBackupRepo.update).toHaveBeenCalledWith(backupId, { + status: BackupStatus.COMPLETED, + ...updates, + updatedAt: expect.any(Date), + }); + expect(mockAlertingService.sendAlert).toHaveBeenCalledWith('BACKUP_COMPLETED', `Backup ${backupId} completed successfully`, 'INFO'); + }); + it('should update backup status to failed and send critical alert', async () => { + await service.updateBackupStatus(backupId, BackupStatus.FAILED, updates); + expect(mockAlertingService.sendAlert).toHaveBeenCalledWith('BACKUP_FAILED', `Backup ${backupId} failed: Test error`, 'CRITICAL'); + }); + it('should update status without sending alert for other statuses', async () => { + await service.updateBackupStatus(backupId, BackupStatus.IN_PROGRESS); + expect(mockAlertingService.sendAlert).not.toHaveBeenCalled(); + }); }); - }); - - describe('handleBackupCleanup', () => { - const mockExpiredBackups = [ - { id: 'backup-1', createdAt: new Date('2024-01-01') }, - { id: 'backup-2', createdAt: new Date('2024-01-02') }, - ]; - - beforeEach(() => { - mockScheduledTaskMonitoringService.startExecution.mockReturnValue('execution-1'); - mockBackupRepo.find.mockResolvedValue(mockExpiredBackups as BackupRecord[]); - mockBackupQueue.add.mockResolvedValue({} as any); - mockScheduledTaskMonitoringService.markSuccess.mockResolvedValue(); + describe('toResponseDto', () => { + const mockBackup = { + id: 'backup-1', + backupType: BackupType.FULL, + status: BackupStatus.COMPLETED, + region: Region.US_EAST_1, + databaseName: 'teachlink', + backupSizeBytes: 1024000, + integrityVerified: true, + completedAt: new Date('2024-01-15T10:00:00Z'), + expiresAt: new Date('2024-02-15T10:00:00Z'), + createdAt: new Date('2024-01-15T09:00:00Z'), + metadata: { version: '1.0' }, + storageKey: 'backup-key', + updatedAt: new Date(), + }; + it('should convert backup record to response DTO', () => { + const result = service.toResponseDto(mockBackup as BackupRecord); + expect(result).toEqual({ + id: 'backup-1', + backupType: BackupType.FULL, + status: BackupStatus.COMPLETED, + region: Region.US_EAST_1, + databaseName: 'teachlink', + backupSizeBytes: 1024000, + integrityVerified: true, + completedAt: new Date('2024-01-15T10:00:00Z'), + expiresAt: new Date('2024-02-15T10:00:00Z'), + createdAt: new Date('2024-01-15T09:00:00Z'), + metadata: { version: '1.0' }, + }); + }); }); - - it('should find and queue deletion of expired backups', async () => { - await service.handleBackupCleanup(); - - expect(mockBackupRepo.find).toHaveBeenCalledWith({ - where: { - createdAt: expect.any(Object), // LessThan operator - status: BackupStatus.COMPLETED, - }, - }); - - expect(mockBackupQueue.add).toHaveBeenCalledTimes(2); - expect(mockBackupQueue.add).toHaveBeenCalledWith( - JOB_NAMES.DELETE_BACKUP, - { backupRecordId: 'backup-1' }, - { - attempts: 3, - backoff: { type: 'exponential', delay: 5000 }, - }, - ); - expect(mockBackupQueue.add).toHaveBeenCalledWith( - JOB_NAMES.DELETE_BACKUP, - { backupRecordId: 'backup-2' }, - { - attempts: 3, - backoff: { type: 'exponential', delay: 5000 }, - }, - ); - - expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith( - 'execution-1', - { - attempt: 1, - maxAttempts: 3, - retriesUsed: 0, - }, - ); + describe('private methods', () => { + describe('executeMonitoredScheduledTask', () => { + const taskConfig = { + expectedIntervalMs: 24 * 60 * 60 * 1000, + timeoutMs: 30 * 60 * 1000, + maxRetries: 2, + }; + it('should execute task successfully on first attempt', async () => { + const taskRunner = jest.fn().mockResolvedValue(undefined); + await (service as unknown).executeMonitoredScheduledTask('test-task', taskConfig, taskRunner); + expect(taskRunner).toHaveBeenCalledTimes(1); + expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith('execution-1', { + attempt: 1, + maxAttempts: 3, + retriesUsed: 0, + }); + }); + it('should retry on failure and succeed', async () => { + const taskRunner = jest + .fn() + .mockRejectedValueOnce(new Error('First attempt failed')) + .mockResolvedValueOnce(undefined); + await (service as unknown).executeMonitoredScheduledTask('test-task', taskConfig, taskRunner); + expect(taskRunner).toHaveBeenCalledTimes(2); + expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith('test-task', 1, 2, 'First attempt failed'); + expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith('execution-1', { + attempt: 2, + maxAttempts: 3, + retriesUsed: 1, + }); + }); + it('should fail after all retries exhausted', async () => { + const error = new Error('Persistent failure'); + const taskRunner = jest.fn().mockRejectedValue(error); + await (service as unknown).executeMonitoredScheduledTask('test-task', taskConfig, taskRunner); + expect(taskRunner).toHaveBeenCalledTimes(3); + expect(mockScheduledTaskMonitoringService.markFailure).toHaveBeenCalledWith('execution-1', 'Persistent failure', { + attempt: 3, + maxAttempts: 3, + retriesUsed: 2, + }); + expect(mockAlertingService.sendAlert).toHaveBeenCalledWith('BACKUP_SCHEDULED_FAILED', 'Scheduled task test-task failed after 3 attempt(s): Persistent failure', 'CRITICAL'); + }); + }); + describe('delay', () => { + it('should delay execution for specified milliseconds', async () => { + const startTime = Date.now(); + await (service as unknown).delay(100); + const endTime = Date.now(); + expect(endTime - startTime).toBeGreaterThanOrEqual(95); // Allow some tolerance + }); + }); }); - - it('should handle cleanup failure and retry', async () => { - mockBackupRepo.find.mockRejectedValueOnce(new Error('Query error')); - mockBackupRepo.find.mockResolvedValueOnce([]); - - await service.handleBackupCleanup(); - - expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith( - 'cleanup-expired-backups', - 1, - 2, - 'Query error', - ); - }); - }); - - describe('getLatestBackup', () => { - const mockBackup = { - id: 'backup-1', - status: BackupStatus.COMPLETED, - integrityVerified: true, - completedAt: new Date('2024-01-15'), - }; - - beforeEach(() => { - mockBackupRepo.findOne.mockResolvedValue(mockBackup as BackupRecord); - }); - - it('should return latest completed and verified backup', async () => { - const result = await service.getLatestBackup(); - - expect(result).toEqual(mockBackup); - expect(mockBackupRepo.findOne).toHaveBeenCalledWith({ - where: { - status: BackupStatus.COMPLETED, - integrityVerified: true, - }, - order: { completedAt: 'DESC' }, - }); - }); - - it('should filter by region when specified', async () => { - await service.getLatestBackup(Region.EU_WEST_1); - - expect(mockBackupRepo.findOne).toHaveBeenCalledWith({ - where: { - status: BackupStatus.COMPLETED, - integrityVerified: true, - region: Region.EU_WEST_1, - }, - order: { completedAt: 'DESC' }, - }); - }); - - it('should return null when no backup found', async () => { - mockBackupRepo.findOne.mockResolvedValue(null); - - const result = await service.getLatestBackup(); - - expect(result).toBeNull(); - }); - }); - - describe('updateBackupStatus', () => { - const backupId = 'backup-1'; - const updates = { errorMessage: 'Test error' }; - - beforeEach(() => { - mockBackupRepo.update.mockResolvedValue({ affected: 1, raw: {}, generatedMaps: [] }); - }); - - it('should update backup status to completed and send success alert', async () => { - await service.updateBackupStatus(backupId, BackupStatus.COMPLETED, updates); - - expect(mockBackupRepo.update).toHaveBeenCalledWith(backupId, { - status: BackupStatus.COMPLETED, - ...updates, - updatedAt: expect.any(Date), - }); - - expect(mockAlertingService.sendAlert).toHaveBeenCalledWith( - 'BACKUP_COMPLETED', - `Backup ${backupId} completed successfully`, - 'INFO', - ); - }); - - it('should update backup status to failed and send critical alert', async () => { - await service.updateBackupStatus(backupId, BackupStatus.FAILED, updates); - - expect(mockAlertingService.sendAlert).toHaveBeenCalledWith( - 'BACKUP_FAILED', - `Backup ${backupId} failed: Test error`, - 'CRITICAL', - ); - }); - - it('should update status without sending alert for other statuses', async () => { - await service.updateBackupStatus(backupId, BackupStatus.IN_PROGRESS); - - expect(mockAlertingService.sendAlert).not.toHaveBeenCalled(); - }); - }); - - describe('toResponseDto', () => { - const mockBackup = { - id: 'backup-1', - backupType: BackupType.FULL, - status: BackupStatus.COMPLETED, - region: Region.US_EAST_1, - databaseName: 'teachlink', - backupSizeBytes: 1024000, - integrityVerified: true, - completedAt: new Date('2024-01-15T10:00:00Z'), - expiresAt: new Date('2024-02-15T10:00:00Z'), - createdAt: new Date('2024-01-15T09:00:00Z'), - metadata: { version: '1.0' }, - storageKey: 'backup-key', - updatedAt: new Date(), - }; - - it('should convert backup record to response DTO', () => { - const result = service.toResponseDto(mockBackup as BackupRecord); - - expect(result).toEqual({ - id: 'backup-1', - backupType: BackupType.FULL, - status: BackupStatus.COMPLETED, - region: Region.US_EAST_1, - databaseName: 'teachlink', - backupSizeBytes: 1024000, - integrityVerified: true, - completedAt: new Date('2024-01-15T10:00:00Z'), - expiresAt: new Date('2024-02-15T10:00:00Z'), - createdAt: new Date('2024-01-15T09:00:00Z'), - metadata: { version: '1.0' }, - }); - }); - }); - - describe('private methods', () => { - describe('executeMonitoredScheduledTask', () => { - const taskConfig = { - expectedIntervalMs: 24 * 60 * 60 * 1000, - timeoutMs: 30 * 60 * 1000, - maxRetries: 2, - }; - - it('should execute task successfully on first attempt', async () => { - const taskRunner = jest.fn().mockResolvedValue(undefined); - - await (service as any).executeMonitoredScheduledTask( - 'test-task', - taskConfig, - taskRunner, - ); - - expect(taskRunner).toHaveBeenCalledTimes(1); - expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith( - 'execution-1', - { - attempt: 1, - maxAttempts: 3, - retriesUsed: 0, - }, - ); - }); - - it('should retry on failure and succeed', async () => { - const taskRunner = jest - .fn() - .mockRejectedValueOnce(new Error('First attempt failed')) - .mockResolvedValueOnce(undefined); - - await (service as any).executeMonitoredScheduledTask( - 'test-task', - taskConfig, - taskRunner, - ); - - expect(taskRunner).toHaveBeenCalledTimes(2); - expect(mockScheduledTaskMonitoringService.recordRetry).toHaveBeenCalledWith( - 'test-task', - 1, - 2, - 'First attempt failed', - ); - expect(mockScheduledTaskMonitoringService.markSuccess).toHaveBeenCalledWith( - 'execution-1', - { - attempt: 2, - maxAttempts: 3, - retriesUsed: 1, - }, - ); - }); - - it('should fail after all retries exhausted', async () => { - const error = new Error('Persistent failure'); - const taskRunner = jest.fn().mockRejectedValue(error); - - await (service as any).executeMonitoredScheduledTask( - 'test-task', - taskConfig, - taskRunner, - ); - - expect(taskRunner).toHaveBeenCalledTimes(3); - expect(mockScheduledTaskMonitoringService.markFailure).toHaveBeenCalledWith( - 'execution-1', - 'Persistent failure', - { - attempt: 3, - maxAttempts: 3, - retriesUsed: 2, - }, - ); - expect(mockAlertingService.sendAlert).toHaveBeenCalledWith( - 'BACKUP_SCHEDULED_FAILED', - 'Scheduled task test-task failed after 3 attempt(s): Persistent failure', - 'CRITICAL', - ); - }); - }); - - describe('delay', () => { - it('should delay execution for specified milliseconds', async () => { - const startTime = Date.now(); - await (service as any).delay(100); - const endTime = Date.now(); - - expect(endTime - startTime).toBeGreaterThanOrEqual(95); // Allow some tolerance - }); - }); - }); -}); \ No newline at end of file +}); diff --git a/src/backup/backup.service.ts b/src/backup/backup.service.ts index 93d3ab9b..59f8ec85 100644 --- a/src/backup/backup.service.ts +++ b/src/backup/backup.service.ts @@ -14,275 +14,187 @@ import { BackupResponseDto } from './dto/backup-response.dto'; import { BackupJobData } from './interfaces/backup.interfaces'; import { AlertingService } from '../monitoring/alerting/alerting.service'; import { MetricsCollectionService } from '../monitoring/metrics/metrics-collection.service'; -import { - ScheduledTaskConfig, - ScheduledTaskMonitoringService, -} from '../monitoring/scheduled-task-monitoring.service'; - +import { ScheduledTaskConfig, ScheduledTaskMonitoringService, } from '../monitoring/scheduled-task-monitoring.service'; @Injectable() export class BackupService { - private readonly logger = new Logger(BackupService.name); - private readonly retentionDays: number; - private readonly scheduledTaskRetryLimit: number; - private readonly scheduledTaskRetryDelayMs: number; - private readonly scheduledTaskTimeoutMs: number; - - constructor( + private readonly logger = new Logger(BackupService.name); + private readonly retentionDays: number; + private readonly scheduledTaskRetryLimit: number; + private readonly scheduledTaskRetryDelayMs: number; + private readonly scheduledTaskTimeoutMs: number; + constructor( @InjectRepository(BackupRecord) - private readonly backupRepository: Repository, + private readonly backupRepository: Repository, @InjectQueue(QUEUE_NAMES.BACKUP_PROCESSING) - private readonly backupQueue: Queue, - private readonly configService: ConfigService, - private readonly alertingService: AlertingService, - private readonly metricsService: MetricsCollectionService, - private readonly scheduledTaskMonitoringService: ScheduledTaskMonitoringService, - ) { - this.retentionDays = this.configService.get('BACKUP_RETENTION_DAYS', 30); - this.scheduledTaskRetryLimit = this.configService.get( - 'BACKUP_SCHEDULED_TASK_RETRY_LIMIT', - 2, - ); - this.scheduledTaskRetryDelayMs = this.configService.get( - 'BACKUP_SCHEDULED_TASK_RETRY_DELAY_MS', - 10000, - ); - this.scheduledTaskTimeoutMs = this.configService.get( - 'BACKUP_SCHEDULED_TASK_TIMEOUT_MS', - 30 * 60 * 1000, - ); - - this.scheduledTaskMonitoringService.registerTask('weekly-database-backup', { - expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, - timeoutMs: this.scheduledTaskTimeoutMs, - maxRetries: this.scheduledTaskRetryLimit, - }); - this.scheduledTaskMonitoringService.registerTask('cleanup-expired-backups', { - expectedIntervalMs: 24 * 60 * 60 * 1000, - timeoutMs: this.scheduledTaskTimeoutMs, - maxRetries: this.scheduledTaskRetryLimit, - }); - } - - /** - * Scheduled weekly backup (every Sunday at 2 AM UTC) - */ - @Cron('0 2 * * 0', { - name: 'weekly-database-backup', - timeZone: 'UTC', - }) - async handleScheduledBackup(): Promise { - await this.executeMonitoredScheduledTask( - 'weekly-database-backup', - { - expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, - timeoutMs: this.scheduledTaskTimeoutMs, - maxRetries: this.scheduledTaskRetryLimit, - }, - async () => { - this.logger.log('Starting scheduled weekly backup'); - - const region = - (this.configService.get('BACKUP_PRIMARY_REGION') as Region) || Region.US_EAST_1; - const databaseName = this.configService.get('DB_DATABASE', 'teachlink'); - - const expiresAt = new Date(); - expiresAt.setDate(expiresAt.getDate() + this.retentionDays); - - const backupRecord = this.backupRepository.create({ - backupType: BackupType.FULL, - status: BackupStatus.PENDING, - region, - databaseName, - storageKey: '', - expiresAt, - metadata: { - startTime: new Date(), - }, + private readonly backupQueue: Queue, private readonly configService: ConfigService, private readonly alertingService: AlertingService, private readonly metricsService: MetricsCollectionService, private readonly scheduledTaskMonitoringService: ScheduledTaskMonitoringService) { + this.retentionDays = this.configService.get('BACKUP_RETENTION_DAYS', 30); + this.scheduledTaskRetryLimit = this.configService.get('BACKUP_SCHEDULED_TASK_RETRY_LIMIT', 2); + this.scheduledTaskRetryDelayMs = this.configService.get('BACKUP_SCHEDULED_TASK_RETRY_DELAY_MS', 10000); + this.scheduledTaskTimeoutMs = this.configService.get('BACKUP_SCHEDULED_TASK_TIMEOUT_MS', 30 * 60 * 1000); + this.scheduledTaskMonitoringService.registerTask('weekly-database-backup', { + expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, + timeoutMs: this.scheduledTaskTimeoutMs, + maxRetries: this.scheduledTaskRetryLimit, }); - - await this.backupRepository.save(backupRecord); - - // Queue backup job - await this.backupQueue.add( - JOB_NAMES.CREATE_BACKUP, - { - backupRecordId: backupRecord.id, - backupType: BackupType.FULL, - region, - databaseName, - } as BackupJobData, - { - attempts: 3, - backoff: { - type: 'exponential', - delay: 10000, - }, - timeout: 3600000, // 1 hour timeout - }, - ); - - this.logger.log(`Scheduled backup ${backupRecord.id} queued`); - }, - ); - } - - /** - * Cleanup expired backups (daily at 3 AM UTC) - */ - @Cron('0 3 * * *', { - name: 'cleanup-expired-backups', - timeZone: 'UTC', - }) - async handleBackupCleanup(): Promise { - await this.executeMonitoredScheduledTask( - 'cleanup-expired-backups', - { - expectedIntervalMs: 24 * 60 * 60 * 1000, - timeoutMs: this.scheduledTaskTimeoutMs, - maxRetries: this.scheduledTaskRetryLimit, - }, - async () => { - this.logger.log('Starting backup cleanup job'); - - const expirationDate = new Date(); - expirationDate.setDate(expirationDate.getDate() - this.retentionDays); - - const expiredBackups = await this.backupRepository.find({ - where: { - createdAt: LessThan(expirationDate), - status: BackupStatus.COMPLETED, - }, + this.scheduledTaskMonitoringService.registerTask('cleanup-expired-backups', { + expectedIntervalMs: 24 * 60 * 60 * 1000, + timeoutMs: this.scheduledTaskTimeoutMs, + maxRetries: this.scheduledTaskRetryLimit, }); - - this.logger.log(`Found ${expiredBackups.length} expired backups to cleanup`); - - for (const backup of expiredBackups) { - await this.backupQueue.add( - JOB_NAMES.DELETE_BACKUP, - { backupRecordId: backup.id }, - { - attempts: 3, - backoff: { type: 'exponential', delay: 5000 }, - }, - ); - } - }, - ); - } - - private async executeMonitoredScheduledTask( - taskName: string, - config: ScheduledTaskConfig, - taskRunner: () => Promise, - ): Promise { - const executionId = this.scheduledTaskMonitoringService.startExecution(taskName, config, { - source: BackupService.name, - }); - - const maxAttempts = (config.maxRetries || 0) + 1; - - for (let attempt = 1; attempt <= maxAttempts; attempt += 1) { - try { - await taskRunner(); - this.scheduledTaskMonitoringService.markSuccess(executionId, { - attempt, - maxAttempts, - retriesUsed: attempt - 1, + } + /** + * Scheduled weekly backup (every Sunday at 2 AM UTC) + */ + @Cron('0 2 * * 0', { + name: 'weekly-database-backup', + timeZone: 'UTC', + }) + async handleScheduledBackup(): Promise { + await this.executeMonitoredScheduledTask('weekly-database-backup', { + expectedIntervalMs: 7 * 24 * 60 * 60 * 1000, + timeoutMs: this.scheduledTaskTimeoutMs, + maxRetries: this.scheduledTaskRetryLimit, + }, async () => { + this.logger.log('Starting scheduled weekly backup'); + const region = (this.configService.get('BACKUP_PRIMARY_REGION') as Region) || Region.US_EAST_1; + const databaseName = this.configService.get('DB_DATABASE', 'teachlink'); + const expiresAt = new Date(); + expiresAt.setDate(expiresAt.getDate() + this.retentionDays); + const backupRecord = this.backupRepository.create({ + backupType: BackupType.FULL, + status: BackupStatus.PENDING, + region, + databaseName, + storageKey: '', + expiresAt, + metadata: { + startTime: new Date(), + }, + }); + await this.backupRepository.save(backupRecord); + // Queue backup job + await this.backupQueue.add(JOB_NAMES.CREATE_BACKUP, { + backupRecordId: backupRecord.id, + backupType: BackupType.FULL, + region, + databaseName, + } as BackupJobData, { + attempts: 3, + backoff: { + type: 'exponential', + delay: 10000, + }, + timeout: 3600000, // 1 hour timeout + }); + this.logger.log(`Scheduled backup ${backupRecord.id} queued`); + }); + } + /** + * Cleanup expired backups (daily at 3 AM UTC) + */ + @Cron('0 3 * * *', { + name: 'cleanup-expired-backups', + timeZone: 'UTC', + }) + async handleBackupCleanup(): Promise { + await this.executeMonitoredScheduledTask('cleanup-expired-backups', { + expectedIntervalMs: 24 * 60 * 60 * 1000, + timeoutMs: this.scheduledTaskTimeoutMs, + maxRetries: this.scheduledTaskRetryLimit, + }, async () => { + this.logger.log('Starting backup cleanup job'); + const expirationDate = new Date(); + expirationDate.setDate(expirationDate.getDate() - this.retentionDays); + const expiredBackups = await this.backupRepository.find({ + where: { + createdAt: LessThan(expirationDate), + status: BackupStatus.COMPLETED, + }, + }); + this.logger.log(`Found ${expiredBackups.length} expired backups to cleanup`); + for (const backup of expiredBackups) { + await this.backupQueue.add(JOB_NAMES.DELETE_BACKUP, { backupRecordId: backup.id }, { + attempts: 3, + backoff: { type: 'exponential', delay: 5000 }, + }); + } + }); + } + private async executeMonitoredScheduledTask(taskName: string, config: ScheduledTaskConfig, taskRunner: () => Promise): Promise { + const executionId = this.scheduledTaskMonitoringService.startExecution(taskName, config, { + source: BackupService.name, }); - return; - } catch (error) { - const errorMessage = error instanceof Error ? error.message : String(error); - const shouldRetry = attempt < maxAttempts; - - this.logger.error( - `Scheduled task ${taskName} failed on attempt ${attempt}/${maxAttempts}: ${errorMessage}`, - error instanceof Error ? error.stack : undefined, - ); - - if (shouldRetry) { - this.scheduledTaskMonitoringService.recordRetry( - taskName, - attempt, - maxAttempts - 1, - errorMessage, - ); - await this.delay(this.scheduledTaskRetryDelayMs); - continue; + const maxAttempts = (config.maxRetries || 0) + 1; + for (let attempt = 1; attempt <= maxAttempts; attempt += 1) { + try { + await taskRunner(); + this.scheduledTaskMonitoringService.markSuccess(executionId, { + attempt, + maxAttempts, + retriesUsed: attempt - 1, + }); + return; + } + catch (error) { + const errorMessage = error instanceof Error ? error.message : String(error); + const shouldRetry = attempt < maxAttempts; + this.logger.error(`Scheduled task ${taskName} failed on attempt ${attempt}/${maxAttempts}: ${errorMessage}`, error instanceof Error ? error.stack : undefined); + if (shouldRetry) { + this.scheduledTaskMonitoringService.recordRetry(taskName, attempt, maxAttempts - 1, errorMessage); + await this.delay(this.scheduledTaskRetryDelayMs); + continue; + } + this.scheduledTaskMonitoringService.markFailure(executionId, errorMessage, { + attempt, + maxAttempts, + retriesUsed: attempt - 1, + }); + this.alertingService.sendAlert('BACKUP_SCHEDULED_FAILED', `Scheduled task ${taskName} failed after ${maxAttempts} attempt(s): ${errorMessage}`, 'CRITICAL'); + } } - - this.scheduledTaskMonitoringService.markFailure(executionId, errorMessage, { - attempt, - maxAttempts, - retriesUsed: attempt - 1, + } + private async delay(ms: number): Promise { + await new Promise((resolve) => setTimeout(resolve, ms)); + } + async getLatestBackup(region?: Region): Promise { + const where: unknown = { + status: BackupStatus.COMPLETED, + integrityVerified: true, + }; + if (region) { + where.region = region; + } + return this.backupRepository.findOne({ + where, + order: { completedAt: 'DESC' }, }); - - this.alertingService.sendAlert( - 'BACKUP_SCHEDULED_FAILED', - `Scheduled task ${taskName} failed after ${maxAttempts} attempt(s): ${errorMessage}`, - 'CRITICAL', - ); - } } - } - - private async delay(ms: number): Promise { - await new Promise((resolve) => setTimeout(resolve, ms)); - } - - async getLatestBackup(region?: Region): Promise { - const where: any = { - status: BackupStatus.COMPLETED, - integrityVerified: true, - }; - if (region) { - where.region = region; + async updateBackupStatus(backupId: string, status: BackupStatus, updates: Partial = {}): Promise { + await this.backupRepository.update(backupId, { + status, + ...updates, + updatedAt: new Date(), + }); + if (status === BackupStatus.COMPLETED) { + this.alertingService.sendAlert('BACKUP_COMPLETED', `Backup ${backupId} completed successfully`, 'INFO'); + } + else if (status === BackupStatus.FAILED) { + this.alertingService.sendAlert('BACKUP_FAILED', `Backup ${backupId} failed: ${updates.errorMessage}`, 'CRITICAL'); + } } - - return this.backupRepository.findOne({ - where, - order: { completedAt: 'DESC' }, - }); - } - - async updateBackupStatus( - backupId: string, - status: BackupStatus, - updates: Partial = {}, - ): Promise { - await this.backupRepository.update(backupId, { - status, - ...updates, - updatedAt: new Date(), - }); - - if (status === BackupStatus.COMPLETED) { - this.alertingService.sendAlert( - 'BACKUP_COMPLETED', - `Backup ${backupId} completed successfully`, - 'INFO', - ); - } else if (status === BackupStatus.FAILED) { - this.alertingService.sendAlert( - 'BACKUP_FAILED', - `Backup ${backupId} failed: ${updates.errorMessage}`, - 'CRITICAL', - ); + toResponseDto(backup: BackupRecord): BackupResponseDto { + return { + id: backup.id, + backupType: backup.backupType, + status: backup.status, + region: backup.region, + databaseName: backup.databaseName, + backupSizeBytes: backup.backupSizeBytes, + integrityVerified: backup.integrityVerified, + completedAt: backup.completedAt, + expiresAt: backup.expiresAt, + createdAt: backup.createdAt, + metadata: backup.metadata, + }; } - } - - toResponseDto(backup: BackupRecord): BackupResponseDto { - return { - id: backup.id, - backupType: backup.backupType, - status: backup.status, - region: backup.region, - databaseName: backup.databaseName, - backupSizeBytes: backup.backupSizeBytes, - integrityVerified: backup.integrityVerified, - completedAt: backup.completedAt, - expiresAt: backup.expiresAt, - createdAt: backup.createdAt, - metadata: backup.metadata, - }; - } } diff --git a/src/backup/disaster-recovery/disaster-recovery.service.ts b/src/backup/disaster-recovery/disaster-recovery.service.ts index e9a76013..2c24cdc5 100644 --- a/src/backup/disaster-recovery/disaster-recovery.service.ts +++ b/src/backup/disaster-recovery/disaster-recovery.service.ts @@ -7,99 +7,66 @@ import { KMSClient, DecryptCommand } from '@aws-sdk/client-kms'; import { exec } from 'child_process'; import { promisify } from 'util'; import * as fs from 'fs'; - const execAsync = promisify(exec); const RTO_THRESHOLD_SECONDS = 900; // 15 minutes - @Injectable() export class DisasterRecoveryService { - private readonly logger = new Logger(DisasterRecoveryService.name); - private readonly kmsClient: KMSClient; - - constructor( - private readonly backupService: BackupService, - private readonly alertingService: AlertingService, - private readonly fileStorageService: FileStorageService, - private readonly configService: ConfigService, - ) { - const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); - this.kmsClient = new KMSClient({ region: awsRegion }); - } - - async executeRestore(backupId: string): Promise { - this.logger.log(`Starting disaster recovery restore for backup: ${backupId}`); - - const rtoStartTime = Date.now(); - - try { - // Step 1: Get latest verified backup - const backup = await this.backupService.getLatestBackup(); - if (!backup) { - throw new Error('No verified backup available for restore'); - } - - // Step 2: Download from secondary region - this.logger.log('Downloading backup from secondary region'); - const backupData = await this.fileStorageService.downloadFile( - backup.replicatedStorageKey || backup.encryptedStorageKey, - ); - - // Step 3: Decrypt with AWS KMS - this.logger.log('Decrypting backup'); - const decryptCommand = new DecryptCommand({ - CiphertextBlob: backupData, - }); - const decryptResponse = await this.kmsClient.send(decryptCommand); - const decryptedData = Buffer.from(decryptResponse.Plaintext); - - // Save to temp file - const tempFile = `/tmp/disaster-recovery-${backupId}.sql`; - await fs.promises.writeFile(tempFile, decryptedData); - - // Step 4: Execute pg_restore to primary database - this.logger.log('Restoring to primary database'); - const databaseName = this.configService.get('DB_DATABASE', 'teachlink'); - await this.restoreDatabase(databaseName, tempFile); - - // Cleanup - await fs.promises.unlink(tempFile); - - // Step 5: Check RTO - const rtoActual = Math.floor((Date.now() - rtoStartTime) / 1000); - this.logger.log(`Disaster recovery completed. RTO: ${rtoActual} seconds`); - - if (rtoActual > RTO_THRESHOLD_SECONDS) { - this.alertingService.sendAlert( - 'DISASTER_RECOVERY_RTO_EXCEEDED', - `Disaster recovery completed but RTO exceeded: ${rtoActual}s > ${RTO_THRESHOLD_SECONDS}s`, - 'CRITICAL', - ); - } else { - this.alertingService.sendAlert( - 'DISASTER_RECOVERY_SUCCESS', - `Disaster recovery completed successfully in ${rtoActual} seconds`, - 'INFO', - ); - } - } catch (error) { - this.logger.error('Disaster recovery failed:', error); - this.alertingService.sendAlert( - 'DISASTER_RECOVERY_FAILED', - `Disaster recovery failed: ${error.message}`, - 'CRITICAL', - ); - throw error; + private readonly logger = new Logger(DisasterRecoveryService.name); + private readonly kmsClient: KMSClient; + constructor(private readonly backupService: BackupService, private readonly alertingService: AlertingService, private readonly fileStorageService: FileStorageService, private readonly configService: ConfigService) { + const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); + this.kmsClient = new KMSClient({ region: awsRegion }); + } + async executeRestore(backupId: string): Promise { + this.logger.log(`Starting disaster recovery restore for backup: ${backupId}`); + const rtoStartTime = Date.now(); + try { + // Step 1: Get latest verified backup + const backup = await this.backupService.getLatestBackup(); + if (!backup) { + throw new Error('No verified backup available for restore'); + } + // Step 2: Download from secondary region + this.logger.log('Downloading backup from secondary region'); + const backupData = await this.fileStorageService.downloadFile(backup.replicatedStorageKey || backup.encryptedStorageKey); + // Step 3: Decrypt with AWS KMS + this.logger.log('Decrypting backup'); + const decryptCommand = new DecryptCommand({ + CiphertextBlob: backupData, + }); + const decryptResponse = await this.kmsClient.send(decryptCommand); + const decryptedData = Buffer.from(decryptResponse.Plaintext); + // Save to temp file + const tempFile = `/tmp/disaster-recovery-${backupId}.sql`; + await fs.promises.writeFile(tempFile, decryptedData); + // Step 4: Execute pg_restore to primary database + this.logger.log('Restoring to primary database'); + const databaseName = this.configService.get('DB_DATABASE', 'teachlink'); + await this.restoreDatabase(databaseName, tempFile); + // Cleanup + await fs.promises.unlink(tempFile); + // Step 5: Check RTO + const rtoActual = Math.floor((Date.now() - rtoStartTime) / 1000); + this.logger.log(`Disaster recovery completed. RTO: ${rtoActual} seconds`); + if (rtoActual > RTO_THRESHOLD_SECONDS) { + this.alertingService.sendAlert('DISASTER_RECOVERY_RTO_EXCEEDED', `Disaster recovery completed but RTO exceeded: ${rtoActual}s > ${RTO_THRESHOLD_SECONDS}s`, 'CRITICAL'); + } + else { + this.alertingService.sendAlert('DISASTER_RECOVERY_SUCCESS', `Disaster recovery completed successfully in ${rtoActual} seconds`, 'INFO'); + } + } + catch (error) { + this.logger.error('Disaster recovery failed:', error); + this.alertingService.sendAlert('DISASTER_RECOVERY_FAILED', `Disaster recovery failed: ${error.message}`, 'CRITICAL'); + throw error; + } + } + private async restoreDatabase(dbName: string, backupFile: string): Promise { + const host = this.configService.get('DB_HOST', 'localhost'); + const port = this.configService.get('DB_PORT', '5432'); + const username = this.configService.get('DB_USERNAME', 'postgres'); + const password = this.configService.get('DB_PASSWORD', ''); + const restoreCommand = `PGPASSWORD="${password}" pg_restore -h ${host} -p ${port} -U ${username} -d ${dbName} -c ${backupFile}`; + await execAsync(restoreCommand); } - } - - private async restoreDatabase(dbName: string, backupFile: string): Promise { - const host = this.configService.get('DB_HOST', 'localhost'); - const port = this.configService.get('DB_PORT', '5432'); - const username = this.configService.get('DB_USERNAME', 'postgres'); - const password = this.configService.get('DB_PASSWORD', ''); - - const restoreCommand = `PGPASSWORD="${password}" pg_restore -h ${host} -p ${port} -U ${username} -d ${dbName} -c ${backupFile}`; - - await execAsync(restoreCommand); - } } diff --git a/src/backup/dto/backup-response.dto.ts b/src/backup/dto/backup-response.dto.ts index bee58496..dea4b6f8 100644 --- a/src/backup/dto/backup-response.dto.ts +++ b/src/backup/dto/backup-response.dto.ts @@ -2,72 +2,53 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { BackupStatus } from '../enums/backup-status.enum'; import { BackupType } from '../enums/backup-type.enum'; import { Region } from '../enums/region.enum'; -import { - IsNotEmpty, - IsUUID, - IsString, - IsOptional, - IsNumber, - IsBoolean, - IsDate, -} from 'class-validator'; - +import { IsNotEmpty, IsUUID, IsString, IsOptional, IsNumber, IsBoolean, IsDate, } from 'class-validator'; export class BackupResponseDto { - @ApiProperty() - @IsNotEmpty() - @IsUUID() - @IsString() - id: string; - - @ApiProperty({ enum: BackupType }) - @IsNotEmpty() - backupType: BackupType; - - @ApiProperty({ enum: BackupStatus }) - @IsNotEmpty() - status: BackupStatus; - - @ApiProperty({ enum: Region }) - @IsNotEmpty() - region: Region; - - @ApiProperty() - @IsNotEmpty() - @IsString() - databaseName: string; - - @ApiPropertyOptional() - @IsOptional() - @IsNumber() - backupSizeBytes?: number; - - @ApiPropertyOptional() - @IsOptional() - @IsBoolean() - integrityVerified?: boolean; - - @ApiPropertyOptional() - @IsOptional() - @IsDate() - completedAt?: Date; - - @ApiPropertyOptional() - @IsOptional() - @IsDate() - expiresAt?: Date; - - @ApiProperty() - @IsNotEmpty() - @IsDate() - createdAt: Date; - - @ApiPropertyOptional() - @IsOptional() - @IsString() - metadata?: { - pgVersion?: string; - tableCounts?: Record; - totalRows?: number; - dumpDuration?: number; - }; + @ApiProperty() + @IsNotEmpty() + @IsUUID() + @IsString() + id: string; + @ApiProperty({ enum: BackupType }) + @IsNotEmpty() + backupType: BackupType; + @ApiProperty({ enum: BackupStatus }) + @IsNotEmpty() + status: BackupStatus; + @ApiProperty({ enum: Region }) + @IsNotEmpty() + region: Region; + @ApiProperty() + @IsNotEmpty() + @IsString() + databaseName: string; + @ApiPropertyOptional() + @IsOptional() + @IsNumber() + backupSizeBytes?: number; + @ApiPropertyOptional() + @IsOptional() + @IsBoolean() + integrityVerified?: boolean; + @ApiPropertyOptional() + @IsOptional() + @IsDate() + completedAt?: Date; + @ApiPropertyOptional() + @IsOptional() + @IsDate() + expiresAt?: Date; + @ApiProperty() + @IsNotEmpty() + @IsDate() + createdAt: Date; + @ApiPropertyOptional() + @IsOptional() + @IsString() + metadata?: { + pgVersion?: string; + tableCounts?: Record; + totalRows?: number; + dumpDuration?: number; + }; } diff --git a/src/backup/dto/recovery-test-response.dto.ts b/src/backup/dto/recovery-test-response.dto.ts index 15c20604..3d4cfcfa 100644 --- a/src/backup/dto/recovery-test-response.dto.ts +++ b/src/backup/dto/recovery-test-response.dto.ts @@ -1,56 +1,48 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { RecoveryTestStatus } from '../enums/recovery-test-status.enum'; import { IsNotEmpty, IsUUID, IsString, IsOptional, IsNumber, IsDate } from 'class-validator'; - export class RecoveryTestResponseDto { - @ApiProperty() - @IsNotEmpty() - @IsUUID() - @IsString() - id: string; - - @ApiProperty() - @IsNotEmpty() - @IsUUID() - @IsString() - backupRecordId: string; - - @ApiProperty({ enum: RecoveryTestStatus }) - @IsNotEmpty() - status: RecoveryTestStatus; - - @ApiProperty() - @IsNotEmpty() - @IsString() - testDatabaseName: string; - - @ApiPropertyOptional() - @IsOptional() - @IsString() - validationResults?: { - tableCountMatch?: boolean; - rowCountMatch?: boolean; - checksumMatch?: boolean; - schemaValid?: boolean; - connectionSuccessful?: boolean; - queriesExecuted?: number; - errors?: string[]; - }; - - @ApiPropertyOptional() - @IsOptional() - @IsNumber() - performanceMetrics?: { - totalDuration?: number; - }; - - @ApiProperty() - @IsNotEmpty() - @IsDate() - createdAt: Date; - - @ApiPropertyOptional() - @IsOptional() - @IsDate() - testCompletedAt?: Date; + @ApiProperty() + @IsNotEmpty() + @IsUUID() + @IsString() + id: string; + @ApiProperty() + @IsNotEmpty() + @IsUUID() + @IsString() + backupRecordId: string; + @ApiProperty({ enum: RecoveryTestStatus }) + @IsNotEmpty() + status: RecoveryTestStatus; + @ApiProperty() + @IsNotEmpty() + @IsString() + testDatabaseName: string; + @ApiPropertyOptional() + @IsOptional() + @IsString() + validationResults?: { + tableCountMatch?: boolean; + rowCountMatch?: boolean; + checksumMatch?: boolean; + schemaValid?: boolean; + connectionSuccessful?: boolean; + queriesExecuted?: number; + errors?: string[]; + }; + @ApiPropertyOptional() + @IsOptional() + @IsNumber() + performanceMetrics?: { + totalDuration?: number; + }; + @ApiProperty() + @IsNotEmpty() + @IsDate() + createdAt: Date; + @ApiPropertyOptional() + @IsOptional() + @IsDate() + testCompletedAt?: Date; } diff --git a/src/backup/dto/restore-backup.dto.ts b/src/backup/dto/restore-backup.dto.ts index 989c087b..4e1211f6 100644 --- a/src/backup/dto/restore-backup.dto.ts +++ b/src/backup/dto/restore-backup.dto.ts @@ -1,10 +1,9 @@ import { IsUUID, IsNotEmpty, IsString } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class RestoreBackupDto { - @ApiProperty({ description: 'Backup record ID to restore from' }) - @IsUUID() - @IsNotEmpty() - @IsString() - backupRecordId: string; + @ApiProperty({ description: 'Backup record ID to restore from' }) + @IsUUID() + @IsNotEmpty() + @IsString() + backupRecordId: string; } diff --git a/src/backup/dto/trigger-recovery-test.dto.ts b/src/backup/dto/trigger-recovery-test.dto.ts index 050000f1..ba3ee0a2 100644 --- a/src/backup/dto/trigger-recovery-test.dto.ts +++ b/src/backup/dto/trigger-recovery-test.dto.ts @@ -1,10 +1,9 @@ import { IsUUID, IsNotEmpty, IsString } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class TriggerRecoveryTestDto { - @ApiProperty({ description: 'Backup record ID to test' }) - @IsUUID() - @IsNotEmpty() - @IsString() - backupRecordId: string; + @ApiProperty({ description: 'Backup record ID to test' }) + @IsUUID() + @IsNotEmpty() + @IsString() + backupRecordId: string; } diff --git a/src/backup/entities/backup-record.entity.ts b/src/backup/entities/backup-record.entity.ts index 845cf4ab..2af277d0 100644 --- a/src/backup/entities/backup-record.entity.ts +++ b/src/backup/entities/backup-record.entity.ts @@ -1,94 +1,64 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, Index, } from 'typeorm'; import { BackupStatus } from '../enums/backup-status.enum'; import { BackupType } from '../enums/backup-type.enum'; import { Region } from '../enums/region.enum'; - @Entity('backup_records') @Index(['status', 'createdAt']) @Index(['region']) @Index(['completedAt']) export class BackupRecord { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'enum', enum: BackupType, default: BackupType.FULL }) - backupType: BackupType; - - @Column({ type: 'enum', enum: BackupStatus, default: BackupStatus.PENDING }) - status: BackupStatus; - - @Column({ type: 'enum', enum: Region }) - region: Region; - - @Column() - databaseName: string; - - @Column() - storageKey: string; - - @Column({ nullable: true }) - encryptedStorageKey: string; - - @Column({ nullable: true }) - replicatedStorageKey: string; - - @Column({ nullable: true }) - kmsKeyId: string; - - @Column({ type: 'bigint', nullable: true }) - backupSizeBytes: number; - - @Column({ type: 'json', nullable: true }) - metadata: { - pgVersion?: string; - tableCounts?: Record; - totalRows?: number; - startTime?: Date; - endTime?: Date; - dumpDuration?: number; - uploadDuration?: number; - encryptionDuration?: number; - replicationDuration?: number; - }; - - @Column({ nullable: true }) - errorMessage: string; - - @Column({ default: 0 }) - retryCount: number; - - @Column({ nullable: true }) - checksumMd5: string; - - @Column({ nullable: true }) - checksumSha256: string; - - @Column({ default: false }) - integrityVerified: boolean; - - @Column({ nullable: true }) - verifiedAt: Date; - - @Column({ nullable: true }) - completedAt: Date; - - @Column({ nullable: true }) - expiresAt: Date; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'enum', enum: BackupType, default: BackupType.FULL }) + backupType: BackupType; + @Column({ type: 'enum', enum: BackupStatus, default: BackupStatus.PENDING }) + status: BackupStatus; + @Column({ type: 'enum', enum: Region }) + region: Region; + @Column() + databaseName: string; + @Column() + storageKey: string; + @Column({ nullable: true }) + encryptedStorageKey: string; + @Column({ nullable: true }) + replicatedStorageKey: string; + @Column({ nullable: true }) + kmsKeyId: string; + @Column({ type: 'bigint', nullable: true }) + backupSizeBytes: number; + @Column({ type: 'json', nullable: true }) + metadata: { + pgVersion?: string; + tableCounts?: Record; + totalRows?: number; + startTime?: Date; + endTime?: Date; + dumpDuration?: number; + uploadDuration?: number; + encryptionDuration?: number; + replicationDuration?: number; + }; + @Column({ nullable: true }) + errorMessage: string; + @Column({ default: 0 }) + retryCount: number; + @Column({ nullable: true }) + checksumMd5: string; + @Column({ nullable: true }) + checksumSha256: string; + @Column({ default: false }) + integrityVerified: boolean; + @Column({ nullable: true }) + verifiedAt: Date; + @Column({ nullable: true }) + completedAt: Date; + @Column({ nullable: true }) + expiresAt: Date; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/backup/entities/recovery-test.entity.ts b/src/backup/entities/recovery-test.entity.ts index ed564a9c..769bd46b 100644 --- a/src/backup/entities/recovery-test.entity.ts +++ b/src/backup/entities/recovery-test.entity.ts @@ -1,73 +1,52 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - CreateDateColumn, - UpdateDateColumn, - Index, - JoinColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, CreateDateColumn, UpdateDateColumn, Index, JoinColumn, } from 'typeorm'; import { RecoveryTestStatus } from '../enums/recovery-test-status.enum'; import { BackupRecord } from './backup-record.entity'; - @Entity('recovery_tests') @Index(['status', 'createdAt']) @Index(['backupRecordId']) @Index(['testCompletedAt']) export class RecoveryTest { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ name: 'backup_record_id' }) - backupRecordId: string; - - @ManyToOne(() => BackupRecord, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'backup_record_id' }) - backupRecord: BackupRecord; - - @Column({ - type: 'enum', - enum: RecoveryTestStatus, - default: RecoveryTestStatus.PENDING, - }) - status: RecoveryTestStatus; - - @Column() - testDatabaseName: string; - - @Column({ type: 'json', nullable: true }) - validationResults: { - tableCountMatch?: boolean; - rowCountMatch?: boolean; - checksumMatch?: boolean; - schemaValid?: boolean; - connectionSuccessful?: boolean; - queriesExecuted?: number; - errors?: string[]; - }; - - @Column({ type: 'json', nullable: true }) - performanceMetrics: { - downloadDuration?: number; - decryptionDuration?: number; - restoreDuration?: number; - validationDuration?: number; - totalDuration?: number; - }; - - @Column({ nullable: true }) - errorMessage: string; - - @Column({ default: 0 }) - retryCount: number; - - @Column({ nullable: true }) - testCompletedAt: Date; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ name: 'backup_record_id' }) + backupRecordId: string; + @ManyToOne(() => BackupRecord, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'backup_record_id' }) + backupRecord: BackupRecord; + @Column({ + type: 'enum', + enum: RecoveryTestStatus, + default: RecoveryTestStatus.PENDING, + }) + status: RecoveryTestStatus; + @Column() + testDatabaseName: string; + @Column({ type: 'json', nullable: true }) + validationResults: { + tableCountMatch?: boolean; + rowCountMatch?: boolean; + checksumMatch?: boolean; + schemaValid?: boolean; + connectionSuccessful?: boolean; + queriesExecuted?: number; + errors?: string[]; + }; + @Column({ type: 'json', nullable: true }) + performanceMetrics: { + downloadDuration?: number; + decryptionDuration?: number; + restoreDuration?: number; + validationDuration?: number; + totalDuration?: number; + }; + @Column({ nullable: true }) + errorMessage: string; + @Column({ default: 0 }) + retryCount: number; + @Column({ nullable: true }) + testCompletedAt: Date; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/backup/enums/backup-status.enum.ts b/src/backup/enums/backup-status.enum.ts index 193d09d2..f6b2a923 100644 --- a/src/backup/enums/backup-status.enum.ts +++ b/src/backup/enums/backup-status.enum.ts @@ -1,6 +1,6 @@ export enum BackupStatus { - PENDING = 'pending', - IN_PROGRESS = 'in_progress', - COMPLETED = 'completed', - FAILED = 'failed', + PENDING = 'pending', + IN_PROGRESS = 'in_progress', + COMPLETED = 'completed', + FAILED = 'failed' } diff --git a/src/backup/enums/backup-type.enum.ts b/src/backup/enums/backup-type.enum.ts index 6de7758d..b68038f1 100644 --- a/src/backup/enums/backup-type.enum.ts +++ b/src/backup/enums/backup-type.enum.ts @@ -1,3 +1,3 @@ export enum BackupType { - FULL = 'full', + FULL = 'full' } diff --git a/src/backup/enums/recovery-test-status.enum.ts b/src/backup/enums/recovery-test-status.enum.ts index 9f6f08e4..ad0cd1ae 100644 --- a/src/backup/enums/recovery-test-status.enum.ts +++ b/src/backup/enums/recovery-test-status.enum.ts @@ -1,6 +1,6 @@ export enum RecoveryTestStatus { - PENDING = 'pending', - RUNNING = 'running', - PASSED = 'passed', - FAILED = 'failed', + PENDING = 'pending', + RUNNING = 'running', + PASSED = 'passed', + FAILED = 'failed' } diff --git a/src/backup/enums/region.enum.ts b/src/backup/enums/region.enum.ts index 132a26e7..c1f51fe1 100644 --- a/src/backup/enums/region.enum.ts +++ b/src/backup/enums/region.enum.ts @@ -1,4 +1,4 @@ export enum Region { - US_EAST_1 = 'us-east-1', - US_WEST_2 = 'us-west-2', + US_EAST_1 = 'us-east-1', + US_WEST_2 = 'us-west-2' } diff --git a/src/backup/integrity/data-integrity.service.ts b/src/backup/integrity/data-integrity.service.ts index d45df828..f265c6d3 100644 --- a/src/backup/integrity/data-integrity.service.ts +++ b/src/backup/integrity/data-integrity.service.ts @@ -5,72 +5,58 @@ import { BackupRecord } from '../entities/backup-record.entity'; import { FileStorageService } from '../../media/storage/file-storage.service'; import * as crypto from 'crypto'; import * as fs from 'fs'; - @Injectable() export class DataIntegrityService { - private readonly logger = new Logger(DataIntegrityService.name); - - constructor( + private readonly logger = new Logger(DataIntegrityService.name); + constructor( @InjectRepository(BackupRecord) - private readonly backupRepository: Repository, - private readonly fileStorageService: FileStorageService, - ) {} - - async verifyBackupIntegrity(backupId: string): Promise { - this.logger.log(`Verifying backup integrity for: ${backupId}`); - - const backup = await this.backupRepository.findOne({ - where: { id: backupId }, - }); - - if (!backup) { - throw new NotFoundException(`Backup ${backupId} not found`); + private readonly backupRepository: Repository, private readonly fileStorageService: FileStorageService) { } + async verifyBackupIntegrity(backupId: string): Promise { + this.logger.log(`Verifying backup integrity for: ${backupId}`); + const backup = await this.backupRepository.findOne({ + where: { id: backupId }, + }); + if (!backup) { + throw new NotFoundException(`Backup ${backupId} not found`); + } + if (!backup.encryptedStorageKey) { + this.logger.error(`Backup ${backupId} has no encrypted storage key, cannot verify`); + return false; + } + try { + // Download backup from S3 + const backupData = await this.fileStorageService.downloadFile(backup.encryptedStorageKey); + // Save to temp file + const tempFile = `/tmp/verify-${backupId}.backup`; + await fs.promises.writeFile(tempFile, backupData); + // Calculate checksums + const checksums = await this.calculateChecksums(tempFile); + // Clean up temp file + await fs.promises.unlink(tempFile); + // Compare checksums + const md5Match = checksums.md5 === backup.checksumMd5; + const sha256Match = checksums.sha256 === backup.checksumSha256; + if (md5Match && sha256Match) { + this.logger.log(`Backup ${backupId} integrity verified successfully`); + return true; + } + else { + this.logger.error(`Backup ${backupId} integrity verification failed. MD5: ${md5Match}, SHA256: ${sha256Match}`); + return false; + } + } + catch (error) { + this.logger.error(`Error verifying backup ${backupId} integrity:`, error); + return false; + } } - - if (!backup.encryptedStorageKey) { - this.logger.error(`Backup ${backupId} has no encrypted storage key, cannot verify`); - return false; + async calculateChecksums(filePath: string): Promise<{ + md5: string; + sha256: string; + }> { + const fileBuffer = await fs.promises.readFile(filePath); + const md5 = crypto.createHash('md5').update(fileBuffer).digest('hex'); + const sha256 = crypto.createHash('sha256').update(fileBuffer).digest('hex'); + return { md5, sha256 }; } - - try { - // Download backup from S3 - const backupData = await this.fileStorageService.downloadFile(backup.encryptedStorageKey); - - // Save to temp file - const tempFile = `/tmp/verify-${backupId}.backup`; - await fs.promises.writeFile(tempFile, backupData); - - // Calculate checksums - const checksums = await this.calculateChecksums(tempFile); - - // Clean up temp file - await fs.promises.unlink(tempFile); - - // Compare checksums - const md5Match = checksums.md5 === backup.checksumMd5; - const sha256Match = checksums.sha256 === backup.checksumSha256; - - if (md5Match && sha256Match) { - this.logger.log(`Backup ${backupId} integrity verified successfully`); - return true; - } else { - this.logger.error( - `Backup ${backupId} integrity verification failed. MD5: ${md5Match}, SHA256: ${sha256Match}`, - ); - return false; - } - } catch (error) { - this.logger.error(`Error verifying backup ${backupId} integrity:`, error); - return false; - } - } - - async calculateChecksums(filePath: string): Promise<{ md5: string; sha256: string }> { - const fileBuffer = await fs.promises.readFile(filePath); - - const md5 = crypto.createHash('md5').update(fileBuffer).digest('hex'); - const sha256 = crypto.createHash('sha256').update(fileBuffer).digest('hex'); - - return { md5, sha256 }; - } } diff --git a/src/backup/interfaces/backup.interfaces.ts b/src/backup/interfaces/backup.interfaces.ts index 39c93fdb..a20bdd95 100644 --- a/src/backup/interfaces/backup.interfaces.ts +++ b/src/backup/interfaces/backup.interfaces.ts @@ -1,19 +1,16 @@ import { Region } from '../enums/region.enum'; - export interface BackupJobData { - backupRecordId: string; - backupType: string; - region: Region; - databaseName: string; + backupRecordId: string; + backupType: string; + region: Region; + databaseName: string; } - export interface VerificationJobData { - backupRecordId: string; - storageKey: string; + backupRecordId: string; + storageKey: string; } - export interface RecoveryTestJobData { - recoveryTestId: string; - backupRecordId: string; - testDatabaseName: string; + recoveryTestId: string; + backupRecordId: string; + testDatabaseName: string; } diff --git a/src/backup/monitoring/backup-monitoring.service.ts b/src/backup/monitoring/backup-monitoring.service.ts index c2359cf2..b5db1c65 100644 --- a/src/backup/monitoring/backup-monitoring.service.ts +++ b/src/backup/monitoring/backup-monitoring.service.ts @@ -8,103 +8,85 @@ import { RecoveryTestStatus } from '../enums/recovery-test-status.enum'; import { MetricsCollectionService } from '../../monitoring/metrics/metrics-collection.service'; import { AlertingService } from '../../monitoring/alerting/alerting.service'; import { Histogram, Counter } from 'prom-client'; - @Injectable() export class BackupMonitoringService { - private readonly logger = new Logger(BackupMonitoringService.name); - private backupDuration: Histogram; - private backupTotal: Counter; - - constructor( + private readonly logger = new Logger(BackupMonitoringService.name); + private backupDuration: Histogram; + private backupTotal: Counter; + constructor( @InjectRepository(BackupRecord) - private readonly backupRepository: Repository, + private readonly backupRepository: Repository, @InjectRepository(RecoveryTest) - private readonly recoveryTestRepository: Repository, - private readonly metricsService: MetricsCollectionService, - private readonly alertingService: AlertingService, - ) { - this.initializeMetrics(); - } - - private initializeMetrics(): void { - const registry = this.metricsService.getRegistry(); - - this.backupDuration = new Histogram({ - name: 'backup_duration_seconds', - help: 'Duration of backup operations in seconds', - labelNames: ['status'], - buckets: [60, 300, 600, 900, 1200], - registers: [registry], - }); - - this.backupTotal = new Counter({ - name: 'backup_total', - help: 'Total number of backups', - labelNames: ['status'], - registers: [registry], - }); - } - - async checkBackupHealth(): Promise<{ healthy: boolean; issues: string[] }> { - const issues: string[] = []; - - // Check last backup was successful - const lastBackup = await this.backupRepository.findOne({ - where: {}, - order: { createdAt: 'DESC' }, - }); - - if (!lastBackup) { - issues.push('No backups found'); - } else { - if (lastBackup.status === BackupStatus.FAILED) { - issues.push(`Last backup failed: ${lastBackup.errorMessage}`); - } - - // Check last backup was within 7 days - const sevenDaysAgo = new Date(); - sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7); - - if (lastBackup.createdAt < sevenDaysAgo) { - issues.push('Last backup is older than 7 days'); - } - - // Check replication - if (lastBackup.status === BackupStatus.COMPLETED && !lastBackup.replicatedStorageKey) { - issues.push('Last backup was not replicated to secondary region'); - } + private readonly recoveryTestRepository: Repository, private readonly metricsService: MetricsCollectionService, private readonly alertingService: AlertingService) { + this.initializeMetrics(); } - - // Check recent recovery tests - const lastTest = await this.recoveryTestRepository.findOne({ - where: {}, - order: { createdAt: 'DESC' }, - }); - - if (lastTest && lastTest.status === RecoveryTestStatus.FAILED) { - issues.push(`Last recovery test failed: ${lastTest.errorMessage}`); + private initializeMetrics(): void { + const registry = this.metricsService.getRegistry(); + this.backupDuration = new Histogram({ + name: 'backup_duration_seconds', + help: 'Duration of backup operations in seconds', + labelNames: ['status'], + buckets: [60, 300, 600, 900, 1200], + registers: [registry], + }); + this.backupTotal = new Counter({ + name: 'backup_total', + help: 'Total number of backups', + labelNames: ['status'], + registers: [registry], + }); } - - return { - healthy: issues.length === 0, - issues, - }; - } - - async recordBackupMetrics(backupId: string, duration: number): Promise { - const backup = await this.backupRepository.findOne({ - where: { id: backupId }, - }); - - if (!backup) { - return; + async checkBackupHealth(): Promise<{ + healthy: boolean; + issues: string[]; + }> { + const issues: string[] = []; + // Check last backup was successful + const lastBackup = await this.backupRepository.findOne({ + where: {}, + order: { createdAt: 'DESC' }, + }); + if (!lastBackup) { + issues.push('No backups found'); + } + else { + if (lastBackup.status === BackupStatus.FAILED) { + issues.push(`Last backup failed: ${lastBackup.errorMessage}`); + } + // Check last backup was within 7 days + const sevenDaysAgo = new Date(); + sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7); + if (lastBackup.createdAt < sevenDaysAgo) { + issues.push('Last backup is older than 7 days'); + } + // Check replication + if (lastBackup.status === BackupStatus.COMPLETED && !lastBackup.replicatedStorageKey) { + issues.push('Last backup was not replicated to secondary region'); + } + } + // Check recent recovery tests + const lastTest = await this.recoveryTestRepository.findOne({ + where: {}, + order: { createdAt: 'DESC' }, + }); + if (lastTest && lastTest.status === RecoveryTestStatus.FAILED) { + issues.push(`Last recovery test failed: ${lastTest.errorMessage}`); + } + return { + healthy: issues.length === 0, + issues, + }; + } + async recordBackupMetrics(backupId: string, duration: number): Promise { + const backup = await this.backupRepository.findOne({ + where: { id: backupId }, + }); + if (!backup) { + return; + } + const status = backup.status === BackupStatus.COMPLETED ? 'success' : 'failure'; + this.backupDuration.observe({ status }, duration / 1000); + this.backupTotal.inc({ status }); + this.logger.log(`Recorded backup metrics for ${backupId}: ${duration}ms, status: ${status}`); } - - const status = backup.status === BackupStatus.COMPLETED ? 'success' : 'failure'; - - this.backupDuration.observe({ status }, duration / 1000); - this.backupTotal.inc({ status }); - - this.logger.log(`Recorded backup metrics for ${backupId}: ${duration}ms, status: ${status}`); - } } diff --git a/src/backup/processing/backup-queue.processor.ts b/src/backup/processing/backup-queue.processor.ts index a77dc068..7bd6c25e 100644 --- a/src/backup/processing/backup-queue.processor.ts +++ b/src/backup/processing/backup-queue.processor.ts @@ -10,300 +10,220 @@ import { BackupStatus } from '../enums/backup-status.enum'; import { BackupService } from '../backup.service'; import { FileStorageService } from '../../media/storage/file-storage.service'; import { DataIntegrityService } from '../integrity/data-integrity.service'; -import { - BackupJobData, - VerificationJobData, - RecoveryTestJobData, -} from '../interfaces/backup.interfaces'; +import { BackupJobData, VerificationJobData, RecoveryTestJobData, } from '../interfaces/backup.interfaces'; import { exec } from 'child_process'; import { promisify } from 'util'; import * as fs from 'fs'; import * as path from 'path'; import { KMSClient, EncryptCommand } from '@aws-sdk/client-kms'; import { S3Client, CopyObjectCommand, DeleteObjectCommand } from '@aws-sdk/client-s3'; - const execAsync = promisify(exec); const MAX_RETRIES = 3; - @Processor(QUEUE_NAMES.BACKUP_PROCESSING) export class BackupQueueProcessor { - private readonly logger = new Logger(BackupQueueProcessor.name); - private readonly kmsClient: KMSClient; - private readonly s3Client: S3Client; - - constructor( + private readonly logger = new Logger(BackupQueueProcessor.name); + private readonly kmsClient: KMSClient; + private readonly s3Client: S3Client; + constructor( @InjectRepository(BackupRecord) - private readonly backupRepository: Repository, - private readonly backupService: BackupService, - private readonly fileStorageService: FileStorageService, - private readonly dataIntegrityService: DataIntegrityService, - private readonly configService: ConfigService, - ) { - const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); - - this.kmsClient = new KMSClient({ - region: awsRegion, - credentials: { - accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), - secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), - }, - }); - - this.s3Client = new S3Client({ - region: awsRegion, - credentials: { - accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), - secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), - }, - }); - } - - @Process(JOB_NAMES.CREATE_BACKUP) - async handleCreateBackup(job: Job) { - const { backupRecordId, databaseName } = job.data; - this.logger.log(`Processing backup creation for: ${backupRecordId}`); - - const backup = await this.backupRepository.findOne({ - where: { id: backupRecordId }, - }); - if (!backup) { - this.logger.warn(`Backup ${backupRecordId} not found, skipping`); - return; + private readonly backupRepository: Repository, private readonly backupService: BackupService, private readonly fileStorageService: FileStorageService, private readonly dataIntegrityService: DataIntegrityService, private readonly configService: ConfigService) { + const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); + this.kmsClient = new KMSClient({ + region: awsRegion, + credentials: { + accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), + secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), + }, + }); + this.s3Client = new S3Client({ + region: awsRegion, + credentials: { + accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), + secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), + }, + }); + } + @Process(JOB_NAMES.CREATE_BACKUP) + async handleCreateBackup(job: Job) { + const { backupRecordId, databaseName } = job.data; + this.logger.log(`Processing backup creation for: ${backupRecordId}`); + const backup = await this.backupRepository.findOne({ + where: { id: backupRecordId }, + }); + if (!backup) { + this.logger.warn(`Backup ${backupRecordId} not found, skipping`); + return; + } + try { + // Step 1: Create pg_dump + await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.IN_PROGRESS); + const dumpStartTime = Date.now(); + const tempFile = path.join('/tmp', `backup-${backupRecordId}.sql`); + const pgDumpCommand = this.buildPgDumpCommand(databaseName, tempFile); + await execAsync(pgDumpCommand); + const dumpDuration = Date.now() - dumpStartTime; + const stats = await fs.promises.stat(tempFile); + backup.backupSizeBytes = stats.size; + // Step 2: Upload to S3 + const uploadStartTime = Date.now(); + const storageKey = `backups/${backup.region}/${databaseName}/${backupRecordId}.sql`; + const fileBuffer = await fs.promises.readFile(tempFile); + await this.fileStorageService.uploadProcessedFile(fileBuffer, storageKey, 'application/sql'); + const uploadDuration = Date.now() - uploadStartTime; + backup.storageKey = storageKey; + // Step 3: Encrypt with AWS KMS + const encryptionStartTime = Date.now(); + const kmsKeyId = this.configService.get('AWS_KMS_KEY_ID'); + const encryptedKey = await this.encryptBackup(storageKey, kmsKeyId); + backup.encryptedStorageKey = encryptedKey; + backup.kmsKeyId = kmsKeyId; + const encryptionDuration = Date.now() - encryptionStartTime; + // Step 4: Replicate to secondary region + const replicationStartTime = Date.now(); + const secondaryRegion = this.configService.get('BACKUP_SECONDARY_REGION', 'us-west-2'); + const replicatedKey = await this.replicateToRegion(encryptedKey, secondaryRegion); + backup.replicatedStorageKey = replicatedKey; + const replicationDuration = Date.now() - replicationStartTime; + // Step 5: Calculate checksums + const checksums = await this.dataIntegrityService.calculateChecksums(tempFile); + backup.checksumMd5 = checksums.md5; + backup.checksumSha256 = checksums.sha256; + // Cleanup temp file + await fs.promises.unlink(tempFile); + // Update metadata + backup.metadata = { + ...backup.metadata, + endTime: new Date(), + dumpDuration, + uploadDuration, + encryptionDuration, + replicationDuration, + }; + await this.backupRepository.save(backup); + // Queue verification job + await (job.queue as unknown).add(JOB_NAMES.VERIFY_BACKUP, { backupRecordId, storageKey: encryptedKey }, { + attempts: 3, + backoff: { type: 'exponential', delay: 5000 }, + }); + this.logger.log(`Backup ${backupRecordId} completed successfully`); + } + catch (error) { + await this.handleProcessingError(backup, error, job.attemptsMade); + throw error; // Re-throw to trigger retry + } } - - try { - // Step 1: Create pg_dump - await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.IN_PROGRESS); - const dumpStartTime = Date.now(); - - const tempFile = path.join('/tmp', `backup-${backupRecordId}.sql`); - const pgDumpCommand = this.buildPgDumpCommand(databaseName, tempFile); - - await execAsync(pgDumpCommand); - const dumpDuration = Date.now() - dumpStartTime; - - const stats = await fs.promises.stat(tempFile); - backup.backupSizeBytes = stats.size; - - // Step 2: Upload to S3 - const uploadStartTime = Date.now(); - const storageKey = `backups/${backup.region}/${databaseName}/${backupRecordId}.sql`; - const fileBuffer = await fs.promises.readFile(tempFile); - - await this.fileStorageService.uploadProcessedFile(fileBuffer, storageKey, 'application/sql'); - - const uploadDuration = Date.now() - uploadStartTime; - backup.storageKey = storageKey; - - // Step 3: Encrypt with AWS KMS - const encryptionStartTime = Date.now(); - const kmsKeyId = this.configService.get('AWS_KMS_KEY_ID'); - const encryptedKey = await this.encryptBackup(storageKey, kmsKeyId); - backup.encryptedStorageKey = encryptedKey; - backup.kmsKeyId = kmsKeyId; - - const encryptionDuration = Date.now() - encryptionStartTime; - - // Step 4: Replicate to secondary region - const replicationStartTime = Date.now(); - const secondaryRegion = this.configService.get( - 'BACKUP_SECONDARY_REGION', - 'us-west-2', - ); - const replicatedKey = await this.replicateToRegion(encryptedKey, secondaryRegion); - backup.replicatedStorageKey = replicatedKey; - - const replicationDuration = Date.now() - replicationStartTime; - - // Step 5: Calculate checksums - const checksums = await this.dataIntegrityService.calculateChecksums(tempFile); - backup.checksumMd5 = checksums.md5; - backup.checksumSha256 = checksums.sha256; - - // Cleanup temp file - await fs.promises.unlink(tempFile); - - // Update metadata - backup.metadata = { - ...backup.metadata, - endTime: new Date(), - dumpDuration, - uploadDuration, - encryptionDuration, - replicationDuration, - }; - - await this.backupRepository.save(backup); - - // Queue verification job - await (job.queue as any).add( - JOB_NAMES.VERIFY_BACKUP, - { backupRecordId, storageKey: encryptedKey }, - { - attempts: 3, - backoff: { type: 'exponential', delay: 5000 }, - }, - ); - - this.logger.log(`Backup ${backupRecordId} completed successfully`); - } catch (error) { - await this.handleProcessingError(backup, error, job.attemptsMade); - throw error; // Re-throw to trigger retry + @Process(JOB_NAMES.VERIFY_BACKUP) + async handleVerifyBackup(job: Job) { + const { backupRecordId } = job.data; + this.logger.log(`Verifying backup integrity: ${backupRecordId}`); + try { + const isValid = await this.dataIntegrityService.verifyBackupIntegrity(backupRecordId); + if (isValid) { + await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.COMPLETED, { + integrityVerified: true, + verifiedAt: new Date(), + completedAt: new Date(), + }); + this.logger.log(`Backup ${backupRecordId} verified successfully`); + } + else { + throw new Error('Backup integrity verification failed'); + } + } + catch (error) { + await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.FAILED, { + errorMessage: `Verification failed: ${error.message}`, + }); + throw error; + } } - } - - @Process(JOB_NAMES.VERIFY_BACKUP) - async handleVerifyBackup(job: Job) { - const { backupRecordId } = job.data; - this.logger.log(`Verifying backup integrity: ${backupRecordId}`); - - try { - const isValid = await this.dataIntegrityService.verifyBackupIntegrity(backupRecordId); - - if (isValid) { - await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.COMPLETED, { - integrityVerified: true, - verifiedAt: new Date(), - completedAt: new Date(), + @Process(JOB_NAMES.RECOVERY_TEST) + async handleRecoveryTest(_job: Job) { + this.logger.log('Recovery test processing handled by RecoveryTestingService'); + // Delegated to RecoveryTestingService.executeRecoveryTest() + } + @Process(JOB_NAMES.DELETE_BACKUP) + async handleDeleteBackup(job: Job<{ + backupRecordId: string; + }>) { + const { backupRecordId } = job.data; + this.logger.log(`Deleting expired backup: ${backupRecordId}`); + const backup = await this.backupRepository.findOne({ + where: { id: backupRecordId }, }); - this.logger.log(`Backup ${backupRecordId} verified successfully`); - } else { - throw new Error('Backup integrity verification failed'); - } - } catch (error) { - await this.backupService.updateBackupStatus(backupRecordId, BackupStatus.FAILED, { - errorMessage: `Verification failed: ${error.message}`, - }); - throw error; + if (!backup) { + this.logger.warn(`Backup ${backupRecordId} not found, skipping deletion`); + return; + } + try { + const bucketName = this.configService.get('AWS_S3_BUCKET', ''); + // Delete from primary region + if (backup.encryptedStorageKey) { + await this.s3Client.send(new DeleteObjectCommand({ + Bucket: bucketName, + Key: backup.encryptedStorageKey, + })); + } + // Delete from secondary region (if replicated) + if (backup.replicatedStorageKey) { + const secondaryBucket = this.configService.get('AWS_S3_BUCKET_SECONDARY', bucketName); + const secondaryS3Client = new S3Client({ + region: this.configService.get('BACKUP_SECONDARY_REGION', 'us-west-2'), + }); + await secondaryS3Client.send(new DeleteObjectCommand({ + Bucket: secondaryBucket, + Key: backup.replicatedStorageKey, + })); + } + // Delete from database + await this.backupRepository.softRemove(backup); + this.logger.log(`Backup ${backupRecordId} deleted successfully`); + } + catch (error) { + this.logger.error(`Failed to delete backup ${backupRecordId}:`, error); + throw error; + } + } + private buildPgDumpCommand(databaseName: string, outputFile: string): string { + const host = this.configService.get('DB_HOST', 'localhost'); + const port = this.configService.get('DB_PORT', '5432'); + const username = this.configService.get('DB_USERNAME', 'postgres'); + const password = this.configService.get('DB_PASSWORD', ''); + return `PGPASSWORD="${password}" pg_dump -h ${host} -p ${port} -U ${username} -F c -b -v -f ${outputFile} ${databaseName}`; } - } - - @Process(JOB_NAMES.RECOVERY_TEST) - async handleRecoveryTest(_job: Job) { - this.logger.log('Recovery test processing handled by RecoveryTestingService'); - // Delegated to RecoveryTestingService.executeRecoveryTest() - } - - @Process(JOB_NAMES.DELETE_BACKUP) - async handleDeleteBackup(job: Job<{ backupRecordId: string }>) { - const { backupRecordId } = job.data; - this.logger.log(`Deleting expired backup: ${backupRecordId}`); - - const backup = await this.backupRepository.findOne({ - where: { id: backupRecordId }, - }); - - if (!backup) { - this.logger.warn(`Backup ${backupRecordId} not found, skipping deletion`); - return; + private async encryptBackup(storageKey: string, kmsKeyId: string): Promise { + const encryptedKey = `${storageKey}.encrypted`; + // Download from S3, encrypt with KMS, re-upload + const fileBuffer = await this.fileStorageService.downloadFile(storageKey); + const command = new EncryptCommand({ + KeyId: kmsKeyId, + Plaintext: fileBuffer, + }); + const response = await this.kmsClient.send(command); + await this.fileStorageService.uploadProcessedFile(Buffer.from(response.CiphertextBlob), encryptedKey, 'application/octet-stream'); + return encryptedKey; } - - try { - const bucketName = this.configService.get('AWS_S3_BUCKET', ''); - - // Delete from primary region - if (backup.encryptedStorageKey) { - await this.s3Client.send( - new DeleteObjectCommand({ - Bucket: bucketName, - Key: backup.encryptedStorageKey, - }), - ); - } - - // Delete from secondary region (if replicated) - if (backup.replicatedStorageKey) { - const secondaryBucket = this.configService.get( - 'AWS_S3_BUCKET_SECONDARY', - bucketName, - ); - const secondaryS3Client = new S3Client({ - region: this.configService.get('BACKUP_SECONDARY_REGION', 'us-west-2'), + private async replicateToRegion(storageKey: string, targetRegion: string): Promise { + this.logger.log(`Replicating ${storageKey} to ${targetRegion}`); + const sourceBucket = this.configService.get('AWS_S3_BUCKET', ''); + const targetBucket = this.configService.get('AWS_S3_BUCKET_SECONDARY', sourceBucket); + const targetKey = storageKey.replace('backups/', `backups-${targetRegion}/`); + const copyCommand = new CopyObjectCommand({ + CopySource: `${sourceBucket}/${storageKey}`, + Bucket: targetBucket, + Key: targetKey, }); - - await secondaryS3Client.send( - new DeleteObjectCommand({ - Bucket: secondaryBucket, - Key: backup.replicatedStorageKey, - }), - ); - } - - // Delete from database - await this.backupRepository.softRemove(backup); - - this.logger.log(`Backup ${backupRecordId} deleted successfully`); - } catch (error) { - this.logger.error(`Failed to delete backup ${backupRecordId}:`, error); - throw error; + await this.s3Client.send(copyCommand); + return targetKey; } - } - - private buildPgDumpCommand(databaseName: string, outputFile: string): string { - const host = this.configService.get('DB_HOST', 'localhost'); - const port = this.configService.get('DB_PORT', '5432'); - const username = this.configService.get('DB_USERNAME', 'postgres'); - const password = this.configService.get('DB_PASSWORD', ''); - - return `PGPASSWORD="${password}" pg_dump -h ${host} -p ${port} -U ${username} -F c -b -v -f ${outputFile} ${databaseName}`; - } - - private async encryptBackup(storageKey: string, kmsKeyId: string): Promise { - const encryptedKey = `${storageKey}.encrypted`; - - // Download from S3, encrypt with KMS, re-upload - const fileBuffer = await this.fileStorageService.downloadFile(storageKey); - - const command = new EncryptCommand({ - KeyId: kmsKeyId, - Plaintext: fileBuffer, - }); - - const response = await this.kmsClient.send(command); - - await this.fileStorageService.uploadProcessedFile( - Buffer.from(response.CiphertextBlob), - encryptedKey, - 'application/octet-stream', - ); - - return encryptedKey; - } - - private async replicateToRegion(storageKey: string, targetRegion: string): Promise { - this.logger.log(`Replicating ${storageKey} to ${targetRegion}`); - - const sourceBucket = this.configService.get('AWS_S3_BUCKET', ''); - const targetBucket = this.configService.get('AWS_S3_BUCKET_SECONDARY', sourceBucket); - - const targetKey = storageKey.replace('backups/', `backups-${targetRegion}/`); - - const copyCommand = new CopyObjectCommand({ - CopySource: `${sourceBucket}/${storageKey}`, - Bucket: targetBucket, - Key: targetKey, - }); - - await this.s3Client.send(copyCommand); - - return targetKey; - } - - private async handleProcessingError( - backup: BackupRecord, - error: Error, - attemptsMade: number, - ): Promise { - this.logger.error(`Backup processing failed for ${backup.id}:`, error); - - backup.retryCount = attemptsMade; - backup.errorMessage = error.message; - - if (attemptsMade >= MAX_RETRIES) { - backup.status = BackupStatus.FAILED; - this.logger.error(`Max retries exceeded for backup ${backup.id}`); + private async handleProcessingError(backup: BackupRecord, error: Error, attemptsMade: number): Promise { + this.logger.error(`Backup processing failed for ${backup.id}:`, error); + backup.retryCount = attemptsMade; + backup.errorMessage = error.message; + if (attemptsMade >= MAX_RETRIES) { + backup.status = BackupStatus.FAILED; + this.logger.error(`Max retries exceeded for backup ${backup.id}`); + } + await this.backupRepository.save(backup); } - - await this.backupRepository.save(backup); - } } diff --git a/src/backup/testing/recovery-testing.service.ts b/src/backup/testing/recovery-testing.service.ts index 3b5762ec..37c3ec54 100644 --- a/src/backup/testing/recovery-testing.service.ts +++ b/src/backup/testing/recovery-testing.service.ts @@ -17,251 +17,189 @@ import { Client } from 'pg'; import { exec } from 'child_process'; import { promisify } from 'util'; import * as fs from 'fs'; - const execAsync = promisify(exec); - @Injectable() export class RecoveryTestingService { - private readonly logger = new Logger(RecoveryTestingService.name); - private readonly kmsClient: KMSClient; - - constructor( + private readonly logger = new Logger(RecoveryTestingService.name); + private readonly kmsClient: KMSClient; + constructor( @InjectRepository(RecoveryTest) - private readonly recoveryTestRepository: Repository, + private readonly recoveryTestRepository: Repository, @InjectQueue(QUEUE_NAMES.BACKUP_PROCESSING) - private readonly backupQueue: Queue, - private readonly backupService: BackupService, - private readonly fileStorageService: FileStorageService, - private readonly configService: ConfigService, - private readonly alertingService: AlertingService, - ) { - const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); - this.kmsClient = new KMSClient({ region: awsRegion }); - } - - async createRecoveryTest(backupId: string): Promise { - const backup = await this.backupService.getLatestBackup(); - if (!backup) { - throw new NotFoundException('No verified backup found'); + private readonly backupQueue: Queue, private readonly backupService: BackupService, private readonly fileStorageService: FileStorageService, private readonly configService: ConfigService, private readonly alertingService: AlertingService) { + const awsRegion = this.configService.get('AWS_REGION', 'us-east-1'); + this.kmsClient = new KMSClient({ region: awsRegion }); } - - const testDatabaseName = this.configService.get( - 'BACKUP_TEST_DATABASE', - 'teachlink_backup_test', - ); - - const recoveryTest = this.recoveryTestRepository.create({ - backupRecordId: backupId, - status: RecoveryTestStatus.PENDING, - testDatabaseName, - }); - - await this.recoveryTestRepository.save(recoveryTest); - - // Queue recovery test job - await this.backupQueue.add( - JOB_NAMES.RECOVERY_TEST, - { - recoveryTestId: recoveryTest.id, - backupRecordId: backupId, - testDatabaseName, - } as RecoveryTestJobData, - { - attempts: 3, - backoff: { type: 'exponential', delay: 10000 }, - }, - ); - - return this.toResponseDto(recoveryTest); - } - - async executeRecoveryTest(testId: string): Promise { - const test = await this.recoveryTestRepository.findOne({ - where: { id: testId }, - relations: ['backupRecord'], - }); - - if (!test) { - throw new NotFoundException(`Recovery test ${testId} not found`); + async createRecoveryTest(backupId: string): Promise { + const backup = await this.backupService.getLatestBackup(); + if (!backup) { + throw new NotFoundException('No verified backup found'); + } + const testDatabaseName = this.configService.get('BACKUP_TEST_DATABASE', 'teachlink_backup_test'); + const recoveryTest = this.recoveryTestRepository.create({ + backupRecordId: backupId, + status: RecoveryTestStatus.PENDING, + testDatabaseName, + }); + await this.recoveryTestRepository.save(recoveryTest); + // Queue recovery test job + await this.backupQueue.add(JOB_NAMES.RECOVERY_TEST, { + recoveryTestId: recoveryTest.id, + backupRecordId: backupId, + testDatabaseName, + } as RecoveryTestJobData, { + attempts: 3, + backoff: { type: 'exponential', delay: 10000 }, + }); + return this.toResponseDto(recoveryTest); } - - const totalStartTime = Date.now(); - - try { - test.status = RecoveryTestStatus.RUNNING; - await this.recoveryTestRepository.save(test); - - // Step 1: Download encrypted backup - const downloadStartTime = Date.now(); - const backupData = await this.fileStorageService.downloadFile( - test.backupRecord.encryptedStorageKey, - ); - const downloadDuration = Date.now() - downloadStartTime; - - // Step 2: Decrypt - const decryptStartTime = Date.now(); - const decryptCommand = new DecryptCommand({ - CiphertextBlob: backupData, - }); - const decryptResponse = await this.kmsClient.send(decryptCommand); - const decryptedData = Buffer.from(decryptResponse.Plaintext); - const decryptionDuration = Date.now() - decryptStartTime; - - // Save to temp file - const tempFile = `/tmp/recovery-test-${testId}.sql`; - await fs.promises.writeFile(tempFile, decryptedData); - - // Step 3: Create test database - await this.createTestDatabase(test.testDatabaseName); - - // Step 4: Restore - const restoreStartTime = Date.now(); - await this.restoreDatabase(test.testDatabaseName, tempFile); - const restoreDuration = Date.now() - restoreStartTime; - - // Step 5: Validate - const validationStartTime = Date.now(); - const validationResults = await this.validateRestoredDatabase(test.testDatabaseName); - const validationDuration = Date.now() - validationStartTime; - - // Step 6: Cleanup - await this.dropTestDatabase(test.testDatabaseName); - await fs.promises.unlink(tempFile); - - // Update test results - test.status = validationResults.connectionSuccessful - ? RecoveryTestStatus.PASSED - : RecoveryTestStatus.FAILED; - test.validationResults = validationResults; - test.performanceMetrics = { - downloadDuration, - decryptionDuration, - restoreDuration, - validationDuration, - totalDuration: Date.now() - totalStartTime, - }; - test.testCompletedAt = new Date(); - - await this.recoveryTestRepository.save(test); - - // Send alert - this.alertingService.sendAlert( - 'RECOVERY_TEST_COMPLETED', - `Recovery test ${testId} ${test.status}`, - test.status === RecoveryTestStatus.PASSED ? 'INFO' : 'CRITICAL', - ); - } catch (error) { - this.logger.error(`Recovery test ${testId} failed:`, error); - test.status = RecoveryTestStatus.FAILED; - test.errorMessage = error.message; - await this.recoveryTestRepository.save(test); - - this.alertingService.sendAlert( - 'RECOVERY_TEST_FAILED', - `Recovery test ${testId} failed: ${error.message}`, - 'CRITICAL', - ); + async executeRecoveryTest(testId: string): Promise { + const test = await this.recoveryTestRepository.findOne({ + where: { id: testId }, + relations: ['backupRecord'], + }); + if (!test) { + throw new NotFoundException(`Recovery test ${testId} not found`); + } + const totalStartTime = Date.now(); + try { + test.status = RecoveryTestStatus.RUNNING; + await this.recoveryTestRepository.save(test); + // Step 1: Download encrypted backup + const downloadStartTime = Date.now(); + const backupData = await this.fileStorageService.downloadFile(test.backupRecord.encryptedStorageKey); + const downloadDuration = Date.now() - downloadStartTime; + // Step 2: Decrypt + const decryptStartTime = Date.now(); + const decryptCommand = new DecryptCommand({ + CiphertextBlob: backupData, + }); + const decryptResponse = await this.kmsClient.send(decryptCommand); + const decryptedData = Buffer.from(decryptResponse.Plaintext); + const decryptionDuration = Date.now() - decryptStartTime; + // Save to temp file + const tempFile = `/tmp/recovery-test-${testId}.sql`; + await fs.promises.writeFile(tempFile, decryptedData); + // Step 3: Create test database + await this.createTestDatabase(test.testDatabaseName); + // Step 4: Restore + const restoreStartTime = Date.now(); + await this.restoreDatabase(test.testDatabaseName, tempFile); + const restoreDuration = Date.now() - restoreStartTime; + // Step 5: Validate + const validationStartTime = Date.now(); + const validationResults = await this.validateRestoredDatabase(test.testDatabaseName); + const validationDuration = Date.now() - validationStartTime; + // Step 6: Cleanup + await this.dropTestDatabase(test.testDatabaseName); + await fs.promises.unlink(tempFile); + // Update test results + test.status = validationResults.connectionSuccessful + ? RecoveryTestStatus.PASSED + : RecoveryTestStatus.FAILED; + test.validationResults = validationResults; + test.performanceMetrics = { + downloadDuration, + decryptionDuration, + restoreDuration, + validationDuration, + totalDuration: Date.now() - totalStartTime, + }; + test.testCompletedAt = new Date(); + await this.recoveryTestRepository.save(test); + // Send alert + this.alertingService.sendAlert('RECOVERY_TEST_COMPLETED', `Recovery test ${testId} ${test.status}`, test.status === RecoveryTestStatus.PASSED ? 'INFO' : 'CRITICAL'); + } + catch (error) { + this.logger.error(`Recovery test ${testId} failed:`, error); + test.status = RecoveryTestStatus.FAILED; + test.errorMessage = error.message; + await this.recoveryTestRepository.save(test); + this.alertingService.sendAlert('RECOVERY_TEST_FAILED', `Recovery test ${testId} failed: ${error.message}`, 'CRITICAL'); + } } - } - - async getTestResults(testId: string): Promise { - const test = await this.recoveryTestRepository.findOne({ - where: { id: testId }, - relations: ['backupRecord'], - }); - - if (!test) { - throw new NotFoundException(`Recovery test ${testId} not found`); + async getTestResults(testId: string): Promise { + const test = await this.recoveryTestRepository.findOne({ + where: { id: testId }, + relations: ['backupRecord'], + }); + if (!test) { + throw new NotFoundException(`Recovery test ${testId} not found`); + } + return this.toResponseDto(test); } - - return this.toResponseDto(test); - } - - private async createTestDatabase(dbName: string): Promise { - const client = new Client({ - host: this.configService.get('DB_HOST', 'localhost'), - port: parseInt(this.configService.get('DB_PORT', '5432')), - user: this.configService.get('DB_USERNAME', 'postgres'), - password: this.configService.get('DB_PASSWORD', ''), - database: 'postgres', - }); - - await client.connect(); - await client.query(`DROP DATABASE IF EXISTS ${dbName}`); - await client.query(`CREATE DATABASE ${dbName}`); - await client.end(); - } - - private async restoreDatabase(dbName: string, backupFile: string): Promise { - const host = this.configService.get('DB_HOST', 'localhost'); - const port = this.configService.get('DB_PORT', '5432'); - const username = this.configService.get('DB_USERNAME', 'postgres'); - const password = this.configService.get('DB_PASSWORD', ''); - - const restoreCommand = `PGPASSWORD="${password}" pg_restore -h ${host} -p ${port} -U ${username} -d ${dbName} ${backupFile}`; - - await execAsync(restoreCommand); - } - - private async validateRestoredDatabase(dbName: string): Promise { - const client = new Client({ - host: this.configService.get('DB_HOST', 'localhost'), - port: parseInt(this.configService.get('DB_PORT', '5432')), - user: this.configService.get('DB_USERNAME', 'postgres'), - password: this.configService.get('DB_PASSWORD', ''), - database: dbName, - }); - - try { - await client.connect(); - - // Run validation queries - const tableCountResult = await client.query( - "SELECT count(*) FROM information_schema.tables WHERE table_schema = 'public'", - ); - const tableCount = parseInt(tableCountResult.rows[0].count); - - await client.end(); - - return { - tableCountMatch: tableCount > 0, - connectionSuccessful: true, - queriesExecuted: 1, - }; - } catch (error) { - return { - connectionSuccessful: false, - errors: [error.message], - }; + private async createTestDatabase(dbName: string): Promise { + const client = new Client({ + host: this.configService.get('DB_HOST', 'localhost'), + port: parseInt(this.configService.get('DB_PORT', '5432')), + user: this.configService.get('DB_USERNAME', 'postgres'), + password: this.configService.get('DB_PASSWORD', ''), + database: 'postgres', + }); + await client.connect(); + await client.query(`DROP DATABASE IF EXISTS ${dbName}`); + await client.query(`CREATE DATABASE ${dbName}`); + await client.end(); + } + private async restoreDatabase(dbName: string, backupFile: string): Promise { + const host = this.configService.get('DB_HOST', 'localhost'); + const port = this.configService.get('DB_PORT', '5432'); + const username = this.configService.get('DB_USERNAME', 'postgres'); + const password = this.configService.get('DB_PASSWORD', ''); + const restoreCommand = `PGPASSWORD="${password}" pg_restore -h ${host} -p ${port} -U ${username} -d ${dbName} ${backupFile}`; + await execAsync(restoreCommand); + } + private async validateRestoredDatabase(dbName: string): Promise { + const client = new Client({ + host: this.configService.get('DB_HOST', 'localhost'), + port: parseInt(this.configService.get('DB_PORT', '5432')), + user: this.configService.get('DB_USERNAME', 'postgres'), + password: this.configService.get('DB_PASSWORD', ''), + database: dbName, + }); + try { + await client.connect(); + // Run validation queries + const tableCountResult = await client.query("SELECT count(*) FROM information_schema.tables WHERE table_schema = 'public'"); + const tableCount = parseInt(tableCountResult.rows[0].count); + await client.end(); + return { + tableCountMatch: tableCount > 0, + connectionSuccessful: true, + queriesExecuted: 1, + }; + } + catch (error) { + return { + connectionSuccessful: false, + errors: [error.message], + }; + } + } + private async dropTestDatabase(dbName: string): Promise { + const client = new Client({ + host: this.configService.get('DB_HOST', 'localhost'), + port: parseInt(this.configService.get('DB_PORT', '5432')), + user: this.configService.get('DB_USERNAME', 'postgres'), + password: this.configService.get('DB_PASSWORD', ''), + database: 'postgres', + }); + await client.connect(); + await client.query(`DROP DATABASE IF EXISTS ${dbName}`); + await client.end(); + } + private toResponseDto(test: RecoveryTest): RecoveryTestResponseDto { + return { + id: test.id, + backupRecordId: test.backupRecordId, + status: test.status, + testDatabaseName: test.testDatabaseName, + validationResults: test.validationResults, + performanceMetrics: test.performanceMetrics + ? { totalDuration: test.performanceMetrics.totalDuration } + : undefined, + createdAt: test.createdAt, + testCompletedAt: test.testCompletedAt, + }; } - } - - private async dropTestDatabase(dbName: string): Promise { - const client = new Client({ - host: this.configService.get('DB_HOST', 'localhost'), - port: parseInt(this.configService.get('DB_PORT', '5432')), - user: this.configService.get('DB_USERNAME', 'postgres'), - password: this.configService.get('DB_PASSWORD', ''), - database: 'postgres', - }); - - await client.connect(); - await client.query(`DROP DATABASE IF EXISTS ${dbName}`); - await client.end(); - } - - private toResponseDto(test: RecoveryTest): RecoveryTestResponseDto { - return { - id: test.id, - backupRecordId: test.backupRecordId, - status: test.status, - testDatabaseName: test.testDatabaseName, - validationResults: test.validationResults, - performanceMetrics: test.performanceMetrics - ? { totalDuration: test.performanceMetrics.totalDuration } - : undefined, - createdAt: test.createdAt, - testCompletedAt: test.testCompletedAt, - }; - } } diff --git a/src/caching/analytics/cache-analytics.service.ts b/src/caching/analytics/cache-analytics.service.ts index 5281c3f0..847dd310 100644 --- a/src/caching/analytics/cache-analytics.service.ts +++ b/src/caching/analytics/cache-analytics.service.ts @@ -1,307 +1,263 @@ import { Injectable, Logger, OnModuleInit, OnModuleDestroy } from '@nestjs/common'; import { CachingService } from '../caching.service'; - export interface CacheMetric { - key: string; - pattern: string; - hits: number; - misses: number; - avgResponseTime: number; - totalRequests: number; - lastAccess: Date; + key: string; + pattern: string; + hits: number; + misses: number; + avgResponseTime: number; + totalRequests: number; + lastAccess: Date; } - export interface CacheAnalyticsSummary { - totalHits: number; - totalMisses: number; - hitRate: number; - missRate: number; - totalKeys: number; - memoryUsage: string; - topKeys: CacheMetric[]; - patternStats: Map; + totalHits: number; + totalMisses: number; + hitRate: number; + missRate: number; + totalKeys: number; + memoryUsage: string; + topKeys: CacheMetric[]; + patternStats: Map; } - @Injectable() export class CacheAnalyticsService implements OnModuleInit, OnModuleDestroy { - private readonly logger = new Logger(CacheAnalyticsService.name); - private metrics: Map = new Map(); - private flushInterval?: NodeJS.Timeout; - private readonly flushIntervalMs = 60000; // Flush every minute - - constructor(private readonly cachingService: CachingService) {} - - onModuleInit(): void { - // Start periodic metrics aggregation - this.startMetricsFlush(); - this.logger.log('Cache analytics service initialized'); - } - - onModuleDestroy(): void { - if (this.flushInterval) { - clearInterval(this.flushInterval); + private readonly logger = new Logger(CacheAnalyticsService.name); + private metrics: Map = new Map(); + private flushInterval?: NodeJS.Timeout; + private readonly flushIntervalMs = 60000; // Flush every minute + constructor(private readonly cachingService: CachingService) { } + onModuleInit(): void { + // Start periodic metrics aggregation + this.startMetricsFlush(); + this.logger.log('Cache analytics service initialized'); } - this.metrics.clear(); - } - - /** - * Record a cache hit - */ - recordHit(key: string, responseTime?: number): void { - const pattern = this.extractPattern(key); - const metric = this.getOrCreateMetric(key, pattern); - - metric.hits++; - metric.totalRequests++; - metric.lastAccess = new Date(); - - if (responseTime !== undefined) { - this.updateAvgResponseTime(metric, responseTime); + onModuleDestroy(): void { + if (this.flushInterval) { + clearInterval(this.flushInterval); + } + this.metrics.clear(); } - } - - /** - * Record a cache miss - */ - recordMiss(key: string, responseTime?: number): void { - const pattern = this.extractPattern(key); - const metric = this.getOrCreateMetric(key, pattern); - - metric.misses++; - metric.totalRequests++; - metric.lastAccess = new Date(); - - if (responseTime !== undefined) { - this.updateAvgResponseTime(metric, responseTime); + /** + * Record a cache hit + */ + recordHit(key: string, responseTime?: number): void { + const pattern = this.extractPattern(key); + const metric = this.getOrCreateMetric(key, pattern); + metric.hits++; + metric.totalRequests++; + metric.lastAccess = new Date(); + if (responseTime !== undefined) { + this.updateAvgResponseTime(metric, responseTime); + } } - } - - /** - * Get metrics for a specific key - */ - getMetric(key: string): CacheMetric | undefined { - return this.metrics.get(key); - } - - /** - * Get all metrics - */ - getAllMetrics(): CacheMetric[] { - return Array.from(this.metrics.values()); - } - - /** - * Get metrics by pattern - */ - getMetricsByPattern(pattern: string): CacheMetric[] { - return this.getAllMetrics().filter((m) => m.pattern === pattern); - } - - /** - * Get analytics summary - */ - async getSummary(): Promise { - const allMetrics = this.getAllMetrics(); - const stats = await this.cachingService.getStats(); - - let totalHits = 0; - let totalMisses = 0; - const patternStats = new Map(); - - for (const metric of allMetrics) { - totalHits += metric.hits; - totalMisses += metric.misses; - - const patternStat = patternStats.get(metric.pattern) || { hits: 0, misses: 0 }; - patternStat.hits += metric.hits; - patternStat.misses += metric.misses; - patternStats.set(metric.pattern, patternStat); + /** + * Record a cache miss + */ + recordMiss(key: string, responseTime?: number): void { + const pattern = this.extractPattern(key); + const metric = this.getOrCreateMetric(key, pattern); + metric.misses++; + metric.totalRequests++; + metric.lastAccess = new Date(); + if (responseTime !== undefined) { + this.updateAvgResponseTime(metric, responseTime); + } } - - const totalRequests = totalHits + totalMisses; - const hitRate = totalRequests > 0 ? (totalHits / totalRequests) * 100 : 0; - const missRate = totalRequests > 0 ? (totalMisses / totalRequests) * 100 : 0; - - // Get top 10 keys by total requests - const topKeys = allMetrics.sort((a, b) => b.totalRequests - a.totalRequests).slice(0, 10); - - return { - totalHits, - totalMisses, - hitRate: Math.round(hitRate * 100) / 100, - missRate: Math.round(missRate * 100) / 100, - totalKeys: stats.keys, - memoryUsage: stats.memory, - topKeys, - patternStats, - }; - } - - /** - * Get hit rate for a specific pattern - */ - getPatternHitRate(pattern: string): number { - const metrics = this.getMetricsByPattern(pattern); - if (metrics.length === 0) return 0; - - let hits = 0; - let total = 0; - - for (const metric of metrics) { - hits += metric.hits; - total += metric.totalRequests; + /** + * Get metrics for a specific key + */ + getMetric(key: string): CacheMetric | undefined { + return this.metrics.get(key); } - - return total > 0 ? (hits / total) * 100 : 0; - } - - /** - * Reset all metrics - */ - resetMetrics(): void { - this.metrics.clear(); - this.logger.log('Cache metrics reset'); - } - - /** - * Reset metrics for a specific pattern - */ - resetPatternMetrics(pattern: string): void { - for (const [key, metric] of this.metrics.entries()) { - if (metric.pattern === pattern) { - this.metrics.delete(key); - } + /** + * Get all metrics + */ + getAllMetrics(): CacheMetric[] { + return Array.from(this.metrics.values()); } - this.logger.log(`Reset metrics for pattern: ${pattern}`); - } - - /** - * Export metrics for external monitoring - */ - exportMetrics(): Record { - const summary: Record = { - timestamp: new Date().toISOString(), - metrics: {}, - }; - - for (const [key, metric] of this.metrics.entries()) { - summary.metrics[key] = { - hits: metric.hits, - misses: metric.misses, - hitRate: metric.totalRequests > 0 ? (metric.hits / metric.totalRequests) * 100 : 0, - missRate: metric.totalRequests > 0 ? (metric.misses / metric.totalRequests) * 100 : 0, - avgResponseTime: metric.avgResponseTime, - }; + /** + * Get metrics by pattern + */ + getMetricsByPattern(pattern: string): CacheMetric[] { + return this.getAllMetrics().filter((m) => m.pattern === pattern); } - - return summary; - } - - /** - * Get or create a metric entry - */ - private getOrCreateMetric(key: string, pattern: string): CacheMetric { - let metric = this.metrics.get(key); - - if (!metric) { - metric = { - key, - pattern, - hits: 0, - misses: 0, - avgResponseTime: 0, - totalRequests: 0, - lastAccess: new Date(), - }; - this.metrics.set(key, metric); + /** + * Get analytics summary + */ + async getSummary(): Promise { + const allMetrics = this.getAllMetrics(); + const stats = await this.cachingService.getStats(); + let totalHits = 0; + let totalMisses = 0; + const patternStats = new Map(); + for (const metric of allMetrics) { + totalHits += metric.hits; + totalMisses += metric.misses; + const patternStat = patternStats.get(metric.pattern) || { hits: 0, misses: 0 }; + patternStat.hits += metric.hits; + patternStat.misses += metric.misses; + patternStats.set(metric.pattern, patternStat); + } + const totalRequests = totalHits + totalMisses; + const hitRate = totalRequests > 0 ? (totalHits / totalRequests) * 100 : 0; + const missRate = totalRequests > 0 ? (totalMisses / totalRequests) * 100 : 0; + // Get top 10 keys by total requests + const topKeys = allMetrics.sort((a, b) => b.totalRequests - a.totalRequests).slice(0, 10); + return { + totalHits, + totalMisses, + hitRate: Math.round(hitRate * 100) / 100, + missRate: Math.round(missRate * 100) / 100, + totalKeys: stats.keys, + memoryUsage: stats.memory, + topKeys, + patternStats, + }; } - - return metric; - } - - /** - * Extract pattern from key - * e.g., 'cache:course:123' -> 'cache:course:*' - */ - private extractPattern(key: string): string { - const parts = key.split(':'); - if (parts.length <= 2) { - return key; + /** + * Get hit rate for a specific pattern + */ + getPatternHitRate(pattern: string): number { + const metrics = this.getMetricsByPattern(pattern); + if (metrics.length === 0) + return 0; + let hits = 0; + let total = 0; + for (const metric of metrics) { + hits += metric.hits; + total += metric.totalRequests; + } + return total > 0 ? (hits / total) * 100 : 0; } - - // Keep prefix and first segment, replace rest with * - return `${parts[0]}:${parts[1]}:*`; - } - - /** - * Update average response time using exponential moving average - */ - private updateAvgResponseTime(metric: CacheMetric, responseTime: number): void { - const alpha = 0.2; // Smoothing factor - metric.avgResponseTime = alpha * responseTime + (1 - alpha) * metric.avgResponseTime; - } - - /** - * Start periodic metrics flush to prevent memory bloat - */ - private startMetricsFlush(): void { - this.flushInterval = setInterval(() => { - this.flushOldMetrics(); - }, this.flushIntervalMs); - } - - /** - * Remove old metrics that haven't been accessed recently - */ - private flushOldMetrics(): void { - const maxAge = 24 * 60 * 60 * 1000; // 24 hours - const now = Date.now(); - let flushed = 0; - - for (const [key, metric] of this.metrics.entries()) { - const age = now - metric.lastAccess.getTime(); - if (age > maxAge) { - this.metrics.delete(key); - flushed++; - } + /** + * Reset all metrics + */ + resetMetrics(): void { + this.metrics.clear(); + this.logger.log('Cache metrics reset'); } - - if (flushed > 0) { - this.logger.debug(`Flushed ${flushed} old metrics entries`); + /** + * Reset metrics for a specific pattern + */ + resetPatternMetrics(pattern: string): void { + for (const [key, metric] of this.metrics.entries()) { + if (metric.pattern === pattern) { + this.metrics.delete(key); + } + } + this.logger.log(`Reset metrics for pattern: ${pattern}`); } - } - - /** - * Get Prometheus-style metrics - */ - getPrometheusMetrics(): string { - const lines: string[] = []; - - lines.push('# HELP cache_hits_total Total number of cache hits'); - lines.push('# TYPE cache_hits_total counter'); - - for (const metric of this.metrics.values()) { - lines.push(`cache_hits_total{pattern="${metric.pattern}"} ${metric.hits}`); + /** + * Export metrics for external monitoring + */ + exportMetrics(): Record { + const summary: Record = { + timestamp: new Date().toISOString(), + metrics: {}, + }; + for (const [key, metric] of this.metrics.entries()) { + summary.metrics[key] = { + hits: metric.hits, + misses: metric.misses, + hitRate: metric.totalRequests > 0 ? (metric.hits / metric.totalRequests) * 100 : 0, + missRate: metric.totalRequests > 0 ? (metric.misses / metric.totalRequests) * 100 : 0, + avgResponseTime: metric.avgResponseTime, + }; + } + return summary; } - - lines.push(''); - lines.push('# HELP cache_misses_total Total number of cache misses'); - lines.push('# TYPE cache_misses_total counter'); - - for (const metric of this.metrics.values()) { - lines.push(`cache_misses_total{pattern="${metric.pattern}"} ${metric.misses}`); + /** + * Get or create a metric entry + */ + private getOrCreateMetric(key: string, pattern: string): CacheMetric { + let metric = this.metrics.get(key); + if (!metric) { + metric = { + key, + pattern, + hits: 0, + misses: 0, + avgResponseTime: 0, + totalRequests: 0, + lastAccess: new Date(), + }; + this.metrics.set(key, metric); + } + return metric; } - - lines.push(''); - lines.push('# HELP cache_avg_response_time_ms Average response time in ms'); - lines.push('# TYPE cache_avg_response_time_ms gauge'); - - for (const metric of this.metrics.values()) { - lines.push( - `cache_avg_response_time_ms{pattern="${metric.pattern}"} ${Math.round(metric.avgResponseTime)}`, - ); + /** + * Extract pattern from key + * e.g., 'cache:course:123' -> 'cache:course:*' + */ + private extractPattern(key: string): string { + const parts = key.split(':'); + if (parts.length <= 2) { + return key; + } + // Keep prefix and first segment, replace rest with * + return `${parts[0]}:${parts[1]}:*`; + } + /** + * Update average response time using exponential moving average + */ + private updateAvgResponseTime(metric: CacheMetric, responseTime: number): void { + const alpha = 0.2; // Smoothing factor + metric.avgResponseTime = alpha * responseTime + (1 - alpha) * metric.avgResponseTime; + } + /** + * Start periodic metrics flush to prevent memory bloat + */ + private startMetricsFlush(): void { + this.flushInterval = setInterval(() => { + this.flushOldMetrics(); + }, this.flushIntervalMs); + } + /** + * Remove old metrics that haven't been accessed recently + */ + private flushOldMetrics(): void { + const maxAge = 24 * 60 * 60 * 1000; // 24 hours + const now = Date.now(); + let flushed = 0; + for (const [key, metric] of this.metrics.entries()) { + const age = now - metric.lastAccess.getTime(); + if (age > maxAge) { + this.metrics.delete(key); + flushed++; + } + } + if (flushed > 0) { + this.logger.debug(`Flushed ${flushed} old metrics entries`); + } + } + /** + * Get Prometheus-style metrics + */ + getPrometheusMetrics(): string { + const lines: string[] = []; + lines.push('# HELP cache_hits_total Total number of cache hits'); + lines.push('# TYPE cache_hits_total counter'); + for (const metric of this.metrics.values()) { + lines.push(`cache_hits_total{pattern="${metric.pattern}"} ${metric.hits}`); + } + lines.push(''); + lines.push('# HELP cache_misses_total Total number of cache misses'); + lines.push('# TYPE cache_misses_total counter'); + for (const metric of this.metrics.values()) { + lines.push(`cache_misses_total{pattern="${metric.pattern}"} ${metric.misses}`); + } + lines.push(''); + lines.push('# HELP cache_avg_response_time_ms Average response time in ms'); + lines.push('# TYPE cache_avg_response_time_ms gauge'); + for (const metric of this.metrics.values()) { + lines.push(`cache_avg_response_time_ms{pattern="${metric.pattern}"} ${Math.round(metric.avgResponseTime)}`); + } + return lines.join('\n'); } - - return lines.join('\n'); - } } diff --git a/src/caching/cache-management.controller.spec.ts b/src/caching/cache-management.controller.spec.ts index c40503f6..1663628b 100644 --- a/src/caching/cache-management.controller.spec.ts +++ b/src/caching/cache-management.controller.spec.ts @@ -5,273 +5,225 @@ import { CacheAnalyticsService } from './analytics/cache-analytics.service'; import { CacheInvalidationService } from './invalidation/invalidation.service'; import { CacheWarmingService } from './warming/cache-warming.service'; import { CacheStrategiesService } from './strategies/cache-strategies.service'; - describe('CacheManagementController', () => { - let controller: CacheManagementController; - let cachingService: jest.Mocked; - let analyticsService: jest.Mocked; - let invalidationService: jest.Mocked; - let warmingService: jest.Mocked; - let strategiesService: jest.Mocked; - - beforeEach(async () => { - const mockCachingService = { - getStats: jest.fn(), - get: jest.fn(), - getTtl: jest.fn(), - delPattern: jest.fn(), - clearAll: jest.fn(), - getTTLConstants: jest.fn(), - }; - - const mockAnalyticsService = { - getSummary: jest.fn(), - getAllMetrics: jest.fn(), - getPrometheusMetrics: jest.fn(), - resetMetrics: jest.fn(), - resetPatternMetrics: jest.fn(), - }; - - const mockInvalidationService = { - invalidateCourse: jest.fn(), - invalidateUser: jest.fn(), - invalidateSearch: jest.fn(), - getStats: jest.fn(), - }; - - const mockWarmingService = { - getStats: jest.fn(), - getWarmedKeys: jest.fn(), - refreshAll: jest.fn(), - }; - - const mockStrategiesService = { - getAllStrategies: jest.fn(), - }; - - const module: TestingModule = await Test.createTestingModule({ - controllers: [CacheManagementController], - providers: [ - { - provide: CachingService, - useValue: mockCachingService, - }, - { - provide: CacheAnalyticsService, - useValue: mockAnalyticsService, - }, - { - provide: CacheInvalidationService, - useValue: mockInvalidationService, - }, - { - provide: CacheWarmingService, - useValue: mockWarmingService, - }, - { - provide: CacheStrategiesService, - useValue: mockStrategiesService, - }, - ], - }).compile(); - - controller = module.get(CacheManagementController); - cachingService = module.get(CachingService); - analyticsService = module.get(CacheAnalyticsService); - invalidationService = module.get(CacheInvalidationService); - warmingService = module.get(CacheWarmingService); - strategiesService = module.get(CacheStrategiesService); - }); - - describe('getStats', () => { - it('should return combined stats', async () => { - cachingService.getStats.mockResolvedValue({ - keys: 100, - memory: '1.5M', - hits: 1000, - misses: 100, - }); - analyticsService.getSummary.mockResolvedValue({ - totalHits: 1000, - totalMisses: 100, - hitRate: 90.91, - missRate: 9.09, - totalKeys: 100, - memoryUsage: '1.5M', - topKeys: [], - patternStats: new Map(), - }); - warmingService.getStats.mockReturnValue({ - totalKeys: 10, - byType: { popular: 5 }, - lastWarmup: new Date(), - }); - invalidationService.getStats.mockReturnValue({ - registeredTags: 5, - totalTrackedKeys: 20, - }); - - const result = await controller.getStats(); - - expect(result).toHaveProperty('redis'); - expect(result).toHaveProperty('analytics'); - expect(result).toHaveProperty('warming'); - expect(result).toHaveProperty('invalidation'); + let controller: CacheManagementController; + let cachingService: jest.Mocked; + let analyticsService: jest.Mocked; + let invalidationService: jest.Mocked; + let warmingService: jest.Mocked; + let strategiesService: jest.Mocked; + beforeEach(async () => { + const mockCachingService = { + getStats: jest.fn(), + get: jest.fn(), + getTtl: jest.fn(), + delPattern: jest.fn(), + clearAll: jest.fn(), + getTTLConstants: jest.fn(), + }; + const mockAnalyticsService = { + getSummary: jest.fn(), + getAllMetrics: jest.fn(), + getPrometheusMetrics: jest.fn(), + resetMetrics: jest.fn(), + resetPatternMetrics: jest.fn(), + }; + const mockInvalidationService = { + invalidateCourse: jest.fn(), + invalidateUser: jest.fn(), + invalidateSearch: jest.fn(), + getStats: jest.fn(), + }; + const mockWarmingService = { + getStats: jest.fn(), + getWarmedKeys: jest.fn(), + refreshAll: jest.fn(), + }; + const mockStrategiesService = { + getAllStrategies: jest.fn(), + }; + const module: TestingModule = await Test.createTestingModule({ + controllers: [CacheManagementController], + providers: [ + { + provide: CachingService, + useValue: mockCachingService, + }, + { + provide: CacheAnalyticsService, + useValue: mockAnalyticsService, + }, + { + provide: CacheInvalidationService, + useValue: mockInvalidationService, + }, + { + provide: CacheWarmingService, + useValue: mockWarmingService, + }, + { + provide: CacheStrategiesService, + useValue: mockStrategiesService, + }, + ], + }).compile(); + controller = module.get(CacheManagementController); + cachingService = module.get(CachingService); + analyticsService = module.get(CacheAnalyticsService); + invalidationService = module.get(CacheInvalidationService); + warmingService = module.get(CacheWarmingService); + strategiesService = module.get(CacheStrategiesService); }); - }); - - describe('getAnalytics', () => { - it('should return analytics', async () => { - analyticsService.getSummary.mockResolvedValue({ - totalHits: 100, - totalMisses: 10, - hitRate: 90.91, - missRate: 9.09, - totalKeys: 50, - memoryUsage: '1M', - topKeys: [], - patternStats: new Map([['cache:course:*', { hits: 50, misses: 5 }]]), - }); - analyticsService.getAllMetrics.mockReturnValue([]); - - const result = await controller.getAnalytics(); - - expect(result).toHaveProperty('summary'); - expect(result).toHaveProperty('metrics'); - expect(result).toHaveProperty('patternStats'); + describe('getStats', () => { + it('should return combined stats', async () => { + cachingService.getStats.mockResolvedValue({ + keys: 100, + memory: '1.5M', + hits: 1000, + misses: 100, + }); + analyticsService.getSummary.mockResolvedValue({ + totalHits: 1000, + totalMisses: 100, + hitRate: 90.91, + missRate: 9.09, + totalKeys: 100, + memoryUsage: '1.5M', + topKeys: [], + patternStats: new Map(), + }); + warmingService.getStats.mockReturnValue({ + totalKeys: 10, + byType: { popular: 5 }, + lastWarmup: new Date(), + }); + invalidationService.getStats.mockReturnValue({ + registeredTags: 5, + totalTrackedKeys: 20, + }); + const result = await controller.getStats(); + expect(result).toHaveProperty('redis'); + expect(result).toHaveProperty('analytics'); + expect(result).toHaveProperty('warming'); + expect(result).toHaveProperty('invalidation'); + }); }); - }); - - describe('getPrometheusMetrics', () => { - it('should return prometheus metrics', () => { - analyticsService.getPrometheusMetrics.mockReturnValue('# HELP cache_hits_total'); - - const result = controller.getPrometheusMetrics(); - - expect(result).toContain('cache_hits_total'); + describe('getAnalytics', () => { + it('should return analytics', async () => { + analyticsService.getSummary.mockResolvedValue({ + totalHits: 100, + totalMisses: 10, + hitRate: 90.91, + missRate: 9.09, + totalKeys: 50, + memoryUsage: '1M', + topKeys: [], + patternStats: new Map([['cache:course:*', { hits: 50, misses: 5 }]]), + }); + analyticsService.getAllMetrics.mockReturnValue([]); + const result = await controller.getAnalytics(); + expect(result).toHaveProperty('summary'); + expect(result).toHaveProperty('metrics'); + expect(result).toHaveProperty('patternStats'); + }); }); - }); - - describe('getStrategies', () => { - it('should return strategies', () => { - strategiesService.getAllStrategies.mockReturnValue([]); - cachingService.getTTLConstants.mockReturnValue({} as any); - - const result = controller.getStrategies(); - - expect(result).toHaveProperty('strategies'); - expect(result).toHaveProperty('ttlConstants'); + describe('getPrometheusMetrics', () => { + it('should return prometheus metrics', () => { + analyticsService.getPrometheusMetrics.mockReturnValue('# HELP cache_hits_total'); + const result = controller.getPrometheusMetrics(); + expect(result).toContain('cache_hits_total'); + }); }); - }); - - describe('getWarmedKeys', () => { - it('should return warmed keys', () => { - warmingService.getStats.mockReturnValue({ - totalKeys: 5, - byType: {}, - lastWarmup: new Date(), - }); - warmingService.getWarmedKeys.mockReturnValue([]); - - const result = controller.getWarmedKeys(); - - expect(result).toHaveProperty('stats'); - expect(result).toHaveProperty('keys'); + describe('getStrategies', () => { + it('should return strategies', () => { + strategiesService.getAllStrategies.mockReturnValue([]); + cachingService.getTTLConstants.mockReturnValue({} as unknown); + const result = controller.getStrategies(); + expect(result).toHaveProperty('strategies'); + expect(result).toHaveProperty('ttlConstants'); + }); }); - }); - - describe('getKey', () => { - it('should return key value', async () => { - cachingService.get.mockResolvedValue({ data: 'value' }); - cachingService.getTtl.mockResolvedValue(120); - - const result = await controller.getKey('test-key'); - - expect(result.key).toBe('test-key'); - expect(result.value).toEqual({ data: 'value' }); - expect(result.exists).toBe(true); + describe('getWarmedKeys', () => { + it('should return warmed keys', () => { + warmingService.getStats.mockReturnValue({ + totalKeys: 5, + byType: {}, + lastWarmup: new Date(), + }); + warmingService.getWarmedKeys.mockReturnValue([]); + const result = controller.getWarmedKeys(); + expect(result).toHaveProperty('stats'); + expect(result).toHaveProperty('keys'); + }); }); - }); - - describe('clearAll', () => { - it('should clear all cache', async () => { - cachingService.clearAll.mockResolvedValue(100); - - await controller.clearAll(); - - expect(cachingService.clearAll).toHaveBeenCalled(); + describe('getKey', () => { + it('should return key value', async () => { + cachingService.get.mockResolvedValue({ data: 'value' }); + cachingService.getTtl.mockResolvedValue(120); + const result = await controller.getKey('test-key'); + expect(result.key).toBe('test-key'); + expect(result.value).toEqual({ data: 'value' }); + expect(result.exists).toBe(true); + }); }); - }); - - describe('clearByPattern', () => { - it('should clear by pattern', async () => { - cachingService.delPattern.mockResolvedValue(5); - - await controller.clearByPattern('course:*'); - - expect(cachingService.delPattern).toHaveBeenCalledWith('cache:course:*'); + describe('clearAll', () => { + it('should clear all cache', async () => { + cachingService.clearAll.mockResolvedValue(100); + await controller.clearAll(); + expect(cachingService.clearAll).toHaveBeenCalled(); + }); }); - }); - - describe('invalidateCourse', () => { - it('should invalidate course cache', async () => { - invalidationService.invalidateCourse.mockResolvedValue([]); - - await controller.invalidateCourse('course-123'); - - expect(invalidationService.invalidateCourse).toHaveBeenCalledWith('course-123'); + describe('clearByPattern', () => { + it('should clear by pattern', async () => { + cachingService.delPattern.mockResolvedValue(5); + await controller.clearByPattern('course:*'); + expect(cachingService.delPattern).toHaveBeenCalledWith('cache:course:*'); + }); }); - }); - - describe('invalidateUser', () => { - it('should invalidate user cache', async () => { - invalidationService.invalidateUser.mockResolvedValue([]); - - await controller.invalidateUser('user-123'); - - expect(invalidationService.invalidateUser).toHaveBeenCalledWith('user-123'); + describe('invalidateCourse', () => { + it('should invalidate course cache', async () => { + invalidationService.invalidateCourse.mockResolvedValue([]); + await controller.invalidateCourse('course-123'); + expect(invalidationService.invalidateCourse).toHaveBeenCalledWith('course-123'); + }); }); - }); - - describe('invalidateSearch', () => { - it('should invalidate search cache', async () => { - invalidationService.invalidateSearch.mockResolvedValue([]); - - await controller.invalidateSearch(); - - expect(invalidationService.invalidateSearch).toHaveBeenCalled(); + describe('invalidateUser', () => { + it('should invalidate user cache', async () => { + invalidationService.invalidateUser.mockResolvedValue([]); + await controller.invalidateUser('user-123'); + expect(invalidationService.invalidateUser).toHaveBeenCalledWith('user-123'); + }); }); - }); - - describe('warmCache', () => { - it('should trigger cache warming', async () => { - warmingService.refreshAll.mockResolvedValue(undefined); - warmingService.getStats.mockReturnValue({ - totalKeys: 10, - byType: {}, - lastWarmup: new Date(), - }); - - const result = await controller.warmCache(); - - expect(result).toHaveProperty('message'); - expect(result).toHaveProperty('stats'); + describe('invalidateSearch', () => { + it('should invalidate search cache', async () => { + invalidationService.invalidateSearch.mockResolvedValue([]); + await controller.invalidateSearch(); + expect(invalidationService.invalidateSearch).toHaveBeenCalled(); + }); }); - }); - - describe('resetAnalytics', () => { - it('should reset all analytics', async () => { - const result = await controller.resetAnalytics(); - - expect(analyticsService.resetMetrics).toHaveBeenCalled(); - expect(result.message).toContain('All analytics reset'); + describe('warmCache', () => { + it('should trigger cache warming', async () => { + warmingService.refreshAll.mockResolvedValue(undefined); + warmingService.getStats.mockReturnValue({ + totalKeys: 10, + byType: {}, + lastWarmup: new Date(), + }); + const result = await controller.warmCache(); + expect(result).toHaveProperty('message'); + expect(result).toHaveProperty('stats'); + }); }); - - it('should reset pattern analytics', async () => { - const result = await controller.resetAnalytics('course:*'); - - expect(analyticsService.resetPatternMetrics).toHaveBeenCalledWith('course:*'); - expect(result.message).toContain('course:*'); + describe('resetAnalytics', () => { + it('should reset all analytics', async () => { + const result = await controller.resetAnalytics(); + expect(analyticsService.resetMetrics).toHaveBeenCalled(); + expect(result.message).toContain('All analytics reset'); + }); + it('should reset pattern analytics', async () => { + const result = await controller.resetAnalytics('course:*'); + expect(analyticsService.resetPatternMetrics).toHaveBeenCalledWith('course:*'); + expect(result.message).toContain('course:*'); + }); }); - }); }); diff --git a/src/caching/cache-management.controller.ts b/src/caching/cache-management.controller.ts index 5331f8c2..8a1624bf 100644 --- a/src/caching/cache-management.controller.ts +++ b/src/caching/cache-management.controller.ts @@ -1,14 +1,4 @@ -import { - Controller, - Get, - Post, - Delete, - Param, - Query, - HttpCode, - HttpStatus, - UseGuards, -} from '@nestjs/common'; +import { Controller, Get, Post, Delete, Param, Query, HttpCode, HttpStatus, UseGuards, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiParam } from '@nestjs/swagger'; import { CachingService } from './caching.service'; import { CacheAnalyticsService } from './analytics/cache-analytics.service'; @@ -17,223 +7,210 @@ import { CacheWarmingService } from './warming/cache-warming.service'; import { CacheStrategiesService } from './strategies/cache-strategies.service'; import { RolesGuard } from '../common/guards/roles.guard'; import { Roles, Role } from '../common/decorators/roles.decorator'; - @ApiTags('Cache Management') @ApiBearerAuth() @Controller('cache') @UseGuards(RolesGuard) @Roles(Role.ADMIN) export class CacheManagementController { - constructor( - private readonly cachingService: CachingService, - private readonly analyticsService: CacheAnalyticsService, - private readonly invalidationService: CacheInvalidationService, - private readonly warmingService: CacheWarmingService, - private readonly strategiesService: CacheStrategiesService, - ) {} - - @Get('stats') - @ApiOperation({ summary: 'Get cache statistics' }) - @ApiResponse({ - status: 200, - description: 'Returns cache statistics including hit/miss rates and memory usage', - }) - async getStats() { - const [redisStats, analyticsSummary, warmingStats, invalidationStats] = await Promise.all([ - this.cachingService.getStats(), - this.analyticsService.getSummary(), - this.warmingService.getStats(), - this.invalidationService.getStats(), - ]); - - return { - redis: redisStats, - analytics: { - totalHits: analyticsSummary.totalHits, - totalMisses: analyticsSummary.totalMisses, - hitRate: `${analyticsSummary.hitRate}%`, - missRate: `${analyticsSummary.missRate}%`, - totalKeys: analyticsSummary.totalKeys, - memoryUsage: analyticsSummary.memoryUsage, - topKeys: analyticsSummary.topKeys, - }, - warming: warmingStats, - invalidation: invalidationStats, - }; - } - - @Get('analytics') - @ApiOperation({ summary: 'Get detailed cache analytics' }) - @ApiResponse({ - status: 200, - description: 'Returns detailed cache analytics including metrics per key', - }) - async getAnalytics() { - const summary = await this.analyticsService.getSummary(); - const allMetrics = this.analyticsService.getAllMetrics(); - - return { - summary: { - totalHits: summary.totalHits, - totalMisses: summary.totalMisses, - hitRate: summary.hitRate, - missRate: summary.missRate, - }, - metrics: allMetrics, - patternStats: Object.fromEntries(summary.patternStats), - }; - } - - @Get('metrics/prometheus') - @ApiOperation({ summary: 'Get Prometheus-compatible metrics' }) - @ApiResponse({ - status: 200, - description: 'Returns metrics in Prometheus format', - }) - getPrometheusMetrics() { - return this.analyticsService.getPrometheusMetrics(); - } - - @Get('strategies') - @ApiOperation({ summary: 'Get all cache strategies' }) - @ApiResponse({ - status: 200, - description: 'Returns all registered cache strategies', - }) - getStrategies() { - return { - strategies: this.strategiesService.getAllStrategies(), - ttlConstants: this.cachingService.getTTLConstants(), - }; - } - - @Get('warmed') - @ApiOperation({ summary: 'Get warmed cache keys' }) - @ApiResponse({ - status: 200, - description: 'Returns all keys that have been warmed', - }) - getWarmedKeys() { - return { - stats: this.warmingService.getStats(), - keys: this.warmingService.getWarmedKeys(), - }; - } - - @Get('key/:key') - @ApiOperation({ summary: 'Get a cached value by key' }) - @ApiParam({ name: 'key', description: 'Cache key to retrieve' }) - @ApiResponse({ - status: 200, - description: 'Returns the cached value', - }) - @ApiResponse({ - status: 404, - description: 'Key not found in cache', - }) - async getKey(@Param('key') key: string) { - const value = await this.cachingService.get(key); - const ttl = await this.cachingService.getTtl(key); - - return { - key, - value, - ttl, - exists: value !== null, - }; - } - - @Delete('clear') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Clear all cache' }) - @ApiResponse({ - status: 204, - description: 'All cache cleared successfully', - }) - async clearAll() { - await this.cachingService.clearAll(); - } - - @Delete('clear/:pattern') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Clear cache by pattern' }) - @ApiParam({ name: 'pattern', description: 'Pattern to match (use * as wildcard)' }) - @ApiResponse({ - status: 204, - description: 'Cache entries matching pattern cleared successfully', - }) - async clearByPattern(@Param('pattern') pattern: string) { - // Decode URL-encoded pattern - const decodedPattern = decodeURIComponent(pattern); - await this.cachingService.delPattern(`cache:${decodedPattern}`); - } - - @Delete('invalidate/course/:courseId') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Invalidate all cache for a specific course' }) - @ApiParam({ name: 'courseId', description: 'Course ID to invalidate cache for' }) - @ApiResponse({ - status: 204, - description: 'Course cache invalidated successfully', - }) - async invalidateCourse(@Param('courseId') courseId: string) { - await this.invalidationService.invalidateCourse(courseId); - } - - @Delete('invalidate/user/:userId') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Invalidate all cache for a specific user' }) - @ApiParam({ name: 'userId', description: 'User ID to invalidate cache for' }) - @ApiResponse({ - status: 204, - description: 'User cache invalidated successfully', - }) - async invalidateUser(@Param('userId') userId: string) { - await this.invalidationService.invalidateUser(userId); - } - - @Delete('invalidate/search') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Invalidate all search cache' }) - @ApiResponse({ - status: 204, - description: 'Search cache invalidated successfully', - }) - async invalidateSearch() { - await this.invalidationService.invalidateSearch(); - } - - @Post('warm') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Manually trigger cache warming' }) - @ApiResponse({ - status: 200, - description: 'Cache warming triggered successfully', - }) - async warmCache() { - await this.warmingService.refreshAll(); - return { - message: 'Cache warming completed', - stats: this.warmingService.getStats(), - }; - } - - @Post('analytics/reset') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Reset analytics metrics' }) - @ApiResponse({ - status: 200, - description: 'Analytics metrics reset successfully', - }) - async resetAnalytics(@Query('pattern') pattern?: string) { - if (pattern) { - this.analyticsService.resetPatternMetrics(pattern); - } else { - this.analyticsService.resetMetrics(); + constructor(private readonly cachingService: CachingService, private readonly analyticsService: CacheAnalyticsService, private readonly invalidationService: CacheInvalidationService, private readonly warmingService: CacheWarmingService, private readonly strategiesService: CacheStrategiesService) { } + @Get('stats') + @ApiOperation({ summary: 'Get cache statistics' }) + @ApiResponse({ + status: 200, + description: 'Returns cache statistics including hit/miss rates and memory usage', + }) + async getStats() { + const [redisStats, analyticsSummary, warmingStats, invalidationStats] = await Promise.all([ + this.cachingService.getStats(), + this.analyticsService.getSummary(), + this.warmingService.getStats(), + this.invalidationService.getStats(), + ]); + return { + redis: redisStats, + analytics: { + totalHits: analyticsSummary.totalHits, + totalMisses: analyticsSummary.totalMisses, + hitRate: `${analyticsSummary.hitRate}%`, + missRate: `${analyticsSummary.missRate}%`, + totalKeys: analyticsSummary.totalKeys, + memoryUsage: analyticsSummary.memoryUsage, + topKeys: analyticsSummary.topKeys, + }, + warming: warmingStats, + invalidation: invalidationStats, + }; + } + @Get('analytics') + @ApiOperation({ summary: 'Get detailed cache analytics' }) + @ApiResponse({ + status: 200, + description: 'Returns detailed cache analytics including metrics per key', + }) + async getAnalytics() { + const summary = await this.analyticsService.getSummary(); + const allMetrics = this.analyticsService.getAllMetrics(); + return { + summary: { + totalHits: summary.totalHits, + totalMisses: summary.totalMisses, + hitRate: summary.hitRate, + missRate: summary.missRate, + }, + metrics: allMetrics, + patternStats: Object.fromEntries(summary.patternStats), + }; + } + @Get('metrics/prometheus') + @ApiOperation({ summary: 'Get Prometheus-compatible metrics' }) + @ApiResponse({ + status: 200, + description: 'Returns metrics in Prometheus format', + }) + getPrometheusMetrics() { + return this.analyticsService.getPrometheusMetrics(); + } + @Get('strategies') + @ApiOperation({ summary: 'Get all cache strategies' }) + @ApiResponse({ + status: 200, + description: 'Returns all registered cache strategies', + }) + getStrategies() { + return { + strategies: this.strategiesService.getAllStrategies(), + ttlConstants: this.cachingService.getTTLConstants(), + }; + } + @Get('warmed') + @ApiOperation({ summary: 'Get warmed cache keys' }) + @ApiResponse({ + status: 200, + description: 'Returns all keys that have been warmed', + }) + getWarmedKeys() { + return { + stats: this.warmingService.getStats(), + keys: this.warmingService.getWarmedKeys(), + }; + } + @Get('key/:key') + @ApiOperation({ summary: 'Get a cached value by key' }) + @ApiParam({ name: 'key', description: 'Cache key to retrieve' }) + @ApiResponse({ + status: 200, + description: 'Returns the cached value', + }) + @ApiResponse({ + status: 404, + description: 'Key not found in cache', + }) + async getKey( + @Param('key') + key: string) { + const value = await this.cachingService.get(key); + const ttl = await this.cachingService.getTtl(key); + return { + key, + value, + ttl, + exists: value !== null, + }; + } + @Delete('clear') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Clear all cache' }) + @ApiResponse({ + status: 204, + description: 'All cache cleared successfully', + }) + async clearAll() { + await this.cachingService.clearAll(); + } + @Delete('clear/:pattern') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Clear cache by pattern' }) + @ApiParam({ name: 'pattern', description: 'Pattern to match (use * as wildcard)' }) + @ApiResponse({ + status: 204, + description: 'Cache entries matching pattern cleared successfully', + }) + async clearByPattern( + @Param('pattern') + pattern: string) { + // Decode URL-encoded pattern + const decodedPattern = decodeURIComponent(pattern); + await this.cachingService.delPattern(`cache:${decodedPattern}`); + } + @Delete('invalidate/course/:courseId') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Invalidate all cache for a specific course' }) + @ApiParam({ name: 'courseId', description: 'Course ID to invalidate cache for' }) + @ApiResponse({ + status: 204, + description: 'Course cache invalidated successfully', + }) + async invalidateCourse( + @Param('courseId') + courseId: string) { + await this.invalidationService.invalidateCourse(courseId); + } + @Delete('invalidate/user/:userId') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Invalidate all cache for a specific user' }) + @ApiParam({ name: 'userId', description: 'User ID to invalidate cache for' }) + @ApiResponse({ + status: 204, + description: 'User cache invalidated successfully', + }) + async invalidateUser( + @Param('userId') + userId: string) { + await this.invalidationService.invalidateUser(userId); + } + @Delete('invalidate/search') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Invalidate all search cache' }) + @ApiResponse({ + status: 204, + description: 'Search cache invalidated successfully', + }) + async invalidateSearch() { + await this.invalidationService.invalidateSearch(); + } + @Post('warm') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Manually trigger cache warming' }) + @ApiResponse({ + status: 200, + description: 'Cache warming triggered successfully', + }) + async warmCache() { + await this.warmingService.refreshAll(); + return { + message: 'Cache warming completed', + stats: this.warmingService.getStats(), + }; + } + @Post('analytics/reset') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Reset analytics metrics' }) + @ApiResponse({ + status: 200, + description: 'Analytics metrics reset successfully', + }) + async resetAnalytics( + @Query('pattern') + pattern?: string) { + if (pattern) { + this.analyticsService.resetPatternMetrics(pattern); + } + else { + this.analyticsService.resetMetrics(); + } + return { + message: pattern ? `Analytics reset for pattern: ${pattern}` : 'All analytics reset', + }; } - - return { - message: pattern ? `Analytics reset for pattern: ${pattern}` : 'All analytics reset', - }; - } } diff --git a/src/caching/caching.constants.ts b/src/caching/caching.constants.ts index 7b911828..1934c003 100644 --- a/src/caching/caching.constants.ts +++ b/src/caching/caching.constants.ts @@ -1,37 +1,34 @@ export const CACHE_REDIS_CLIENT = 'CACHE_REDIS_CLIENT'; - export const CACHE_TTL = { - USER_SESSION: 604800, // 7 days - COURSE_METADATA: 900, // 15 minutes - COURSE_DETAILS: 300, // 5 minutes - SEARCH_RESULTS: 120, // 2 minutes - USER_PROFILE: 600, // 10 minutes - STATIC_CONTENT: 3600, // 1 hour - POPULAR_COURSES: 1800, // 30 minutes - ENROLLMENT_DATA: 300, // 5 minutes + USER_SESSION: 604800, // 7 days + COURSE_METADATA: 900, // 15 minutes + COURSE_DETAILS: 300, // 5 minutes + SEARCH_RESULTS: 120, // 2 minutes + USER_PROFILE: 600, // 10 minutes + STATIC_CONTENT: 3600, // 1 hour + POPULAR_COURSES: 1800, // 30 minutes + ENROLLMENT_DATA: 300, // 5 minutes } as const; - export const CACHE_PREFIXES = { - COURSE: 'cache:course', - COURSES_LIST: 'cache:courses:list', - USER: 'cache:user', - USER_PROFILE: 'cache:user:profile', - SEARCH: 'cache:search', - POPULAR: 'cache:popular', - ENROLLMENT: 'cache:enrollment', - FEATURED: 'cache:featured', - SYSTEM_CONFIG: 'cache:system:config', - USERS_LIST: 'cache:users:list', + COURSE: 'cache:course', + COURSES_LIST: 'cache:courses:list', + USER: 'cache:user', + USER_PROFILE: 'cache:user:profile', + SEARCH: 'cache:search', + POPULAR: 'cache:popular', + ENROLLMENT: 'cache:enrollment', + FEATURED: 'cache:featured', + SYSTEM_CONFIG: 'cache:system:config', + USERS_LIST: 'cache:users:list', } as const; - export const CACHE_EVENTS = { - COURSE_UPDATED: 'cache.course.updated', - COURSE_DELETED: 'cache.course.deleted', - USER_UPDATED: 'cache.user.updated', - USER_DELETED: 'cache.user.deleted', - ENROLLMENT_CREATED: 'cache.enrollment.created', - ENROLLMENT_UPDATED: 'cache.enrollment.updated', - SEARCH_INDEX_UPDATED: 'cache.search.updated', - CACHE_INVALIDATED: 'cache.invalidated', - CACHE_PURGED: 'cache.purged', + COURSE_UPDATED: 'cache.course.updated', + COURSE_DELETED: 'cache.course.deleted', + USER_UPDATED: 'cache.user.updated', + USER_DELETED: 'cache.user.deleted', + ENROLLMENT_CREATED: 'cache.enrollment.created', + ENROLLMENT_UPDATED: 'cache.enrollment.updated', + SEARCH_INDEX_UPDATED: 'cache.search.updated', + CACHE_INVALIDATED: 'cache.invalidated', + CACHE_PURGED: 'cache.purged', } as const; diff --git a/src/caching/caching.module.ts b/src/caching/caching.module.ts index 304054ca..ddc1e4fb 100644 --- a/src/caching/caching.module.ts +++ b/src/caching/caching.module.ts @@ -9,52 +9,48 @@ import { CacheInvalidationService } from './invalidation/invalidation.service'; import { CacheWarmingService } from './warming/cache-warming.service'; import { CacheAnalyticsService } from './analytics/cache-analytics.service'; import { CacheManagementController } from './cache-management.controller'; - @Global() @Module({ - imports: [ConfigModule, EventEmitterModule], - controllers: [CacheManagementController], - providers: [ - // Redis client provider - { - provide: CACHE_REDIS_CLIENT, - inject: [ConfigService], - useFactory: (configService: ConfigService) => { - const client = new Redis({ - host: configService.get('REDIS_HOST') || 'localhost', - port: parseInt(configService.get('REDIS_PORT') || '6379', 10), - lazyConnect: false, - maxRetriesPerRequest: null, - enableReadyCheck: true, - }); - - client.on('error', () => { - // Prevent unhandled error events when Redis is temporarily unavailable. - }); - - return client; - }, - }, - // Cache services - CachingService, - CacheStrategiesService, - CacheInvalidationService, - CacheWarmingService, - CacheAnalyticsService, - ], - exports: [ - CACHE_REDIS_CLIENT, - CachingService, - CacheStrategiesService, - CacheInvalidationService, - CacheWarmingService, - CacheAnalyticsService, - ], + imports: [ConfigModule, EventEmitterModule], + controllers: [CacheManagementController], + providers: [ + // Redis client provider + { + provide: CACHE_REDIS_CLIENT, + inject: [ConfigService], + useFactory: (configService: ConfigService) => { + const client = new Redis({ + host: configService.get('REDIS_HOST') || 'localhost', + port: parseInt(configService.get('REDIS_PORT') || '6379', 10), + lazyConnect: false, + maxRetriesPerRequest: null, + enableReadyCheck: true, + }); + client.on('error', () => { + // Prevent unhandled error events when Redis is temporarily unavailable. + }); + return client; + }, + }, + // Cache services + CachingService, + CacheStrategiesService, + CacheInvalidationService, + CacheWarmingService, + CacheAnalyticsService, + ], + exports: [ + CACHE_REDIS_CLIENT, + CachingService, + CacheStrategiesService, + CacheInvalidationService, + CacheWarmingService, + CacheAnalyticsService, + ], }) export class CachingModule implements OnModuleDestroy { - constructor(private readonly cachingService: CachingService) {} - - onModuleDestroy() { - // Cleanup is handled by CachingService - } + constructor(private readonly cachingService: CachingService) { } + onModuleDestroy() { + // Cleanup is handled by CachingService + } } diff --git a/src/caching/caching.service.spec.ts b/src/caching/caching.service.spec.ts index 79a3b35d..37b15d22 100644 --- a/src/caching/caching.service.spec.ts +++ b/src/caching/caching.service.spec.ts @@ -4,261 +4,189 @@ import { CachingService } from './caching.service'; import { CACHE_REDIS_CLIENT } from './caching.constants'; import { createMockRedisClient, createMockConfigService } from 'test/utils/mock-factories'; import Redis from 'ioredis'; - describe('CachingService', () => { - let service: CachingService; - let mockRedis: jest.Mocked; - - beforeEach(async () => { - // ─── Initialize Mocks ────────────────────────────────────────────────── - mockRedis = createMockRedisClient(); - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - CachingService, - { - provide: ConfigService, - useValue: createMockConfigService({ CACHE_TTL: 300 }), - }, - { - provide: CACHE_REDIS_CLIENT, - useValue: mockRedis, - }, - ], - }).compile(); - - service = module.get(CachingService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - describe('get', () => { - it('should return cached value', async () => { - mockRedis.get.mockResolvedValue(JSON.stringify({ id: '1', name: 'Test' })); - - const result = await service.get('test:key'); - - expect(result).toEqual({ id: '1', name: 'Test' }); - expect(mockRedis.get).toHaveBeenCalledWith('test:key'); + let service: CachingService; + let mockRedis: jest.Mocked; + beforeEach(async () => { + // ─── Initialize Mocks ────────────────────────────────────────────────── + mockRedis = createMockRedisClient(); + const module: TestingModule = await Test.createTestingModule({ + providers: [ + CachingService, + { + provide: ConfigService, + useValue: createMockConfigService({ CACHE_TTL: 300 }), + }, + { + provide: CACHE_REDIS_CLIENT, + useValue: mockRedis, + }, + ], + }).compile(); + service = module.get(CachingService); + }); + afterEach(() => { + jest.clearAllMocks(); + }); + describe('get', () => { + it('should return cached value', async () => { + mockRedis.get.mockResolvedValue(JSON.stringify({ id: '1', name: 'Test' })); + const result = await service.get('test:key'); + expect(result).toEqual({ id: '1', name: 'Test' }); + expect(mockRedis.get).toHaveBeenCalledWith('test:key'); + }); + it('should return null when key not found', async () => { + mockRedis.get.mockResolvedValue(null); + const result = await service.get('test:key'); + expect(result).toBeNull(); + }); + it('should return raw string for non-JSON values', async () => { + mockRedis.get.mockResolvedValue('raw-string-value'); + const result = await service.get('test:key'); + expect(result).toBe('raw-string-value'); + }); + }); + describe('set', () => { + it('should set value with TTL', async () => { + await service.set('test:key', { data: 'value' }, 60); + expect(mockRedis.set).toHaveBeenCalledWith('test:key', JSON.stringify({ data: 'value' }), 'EX', 60); + }); + it('should set value without TTL when ttl is 0', async () => { + await service.set('test:key', { data: 'value' }, 0); + expect(mockRedis.set).toHaveBeenCalledWith('test:key', JSON.stringify({ data: 'value' })); + }); + it('should set string value directly', async () => { + await service.set('test:key', 'string-value'); + expect(mockRedis.set).toHaveBeenCalledWith('test:key', 'string-value', 'EX', 300); + }); + }); + describe('del', () => { + it('should delete key', async () => { + await service.del('test:key'); + expect(mockRedis.del).toHaveBeenCalledWith('test:key'); + }); + }); + describe('delPattern', () => { + it('should delete keys matching pattern', async () => { + mockRedis.scan.mockResolvedValueOnce(['0', ['key1', 'key2']]); + const result = await service.delPattern('test:*'); + expect(result).toBe(2); + expect(mockRedis.del).toHaveBeenCalledWith('key1', 'key2'); + }); + it('should return 0 when no keys match', async () => { + mockRedis.scan.mockResolvedValueOnce(['0', []]); + const result = await service.delPattern('test:*'); + expect(result).toBe(0); + }); + }); + describe('getOrSet', () => { + it('should return cached value when available', async () => { + mockRedis.get.mockResolvedValue(JSON.stringify({ cached: true })); + const factory = jest.fn(); + const result = await service.getOrSet('test:key', factory, 60); + expect(result).toEqual({ cached: true }); + expect(factory).not.toHaveBeenCalled(); + }); + it('should call factory and cache result when not available', async () => { + mockRedis.get.mockResolvedValue(null); + const factory = jest.fn().mockResolvedValue({ new: 'data' }); + const result = await service.getOrSet('test:key', factory, 60); + expect(result).toEqual({ new: 'data' }); + expect(factory).toHaveBeenCalled(); + expect(mockRedis.set).toHaveBeenCalled(); + }); + }); + describe('exists', () => { + it('should return true when key exists', async () => { + mockRedis.exists.mockResolvedValue(1); + const result = await service.exists('test:key'); + expect(result).toBe(true); + }); + it('should return false when key does not exist', async () => { + mockRedis.exists.mockResolvedValue(0); + const result = await service.exists('test:key'); + expect(result).toBe(false); + }); + }); + describe('getTtl', () => { + it('should return TTL of key', async () => { + mockRedis.ttl.mockResolvedValue(120); + const result = await service.getTtl('test:key'); + expect(result).toBe(120); + }); + }); + describe('incr', () => { + it('should increment by 1 by default', async () => { + mockRedis.incr.mockResolvedValue(1); + const result = await service.incr('counter'); + expect(result).toBe(1); + expect(mockRedis.incr).toHaveBeenCalledWith('counter'); + }); + it('should increment by specified amount', async () => { + mockRedis.incrby.mockResolvedValue(5); + const result = await service.incr('counter', 5); + expect(result).toBe(5); + expect(mockRedis.incrby).toHaveBeenCalledWith('counter', 5); + }); + }); + describe('mget', () => { + it('should get multiple values', async () => { + mockRedis.mget.mockResolvedValue([ + JSON.stringify({ id: 1 }), + null, + JSON.stringify({ id: 3 }), + ]); + const result = await service.mget(['key1', 'key2', 'key3']); + expect(result).toEqual([{ id: 1 }, null, { id: 3 }]); + }); + it('should return empty array for empty input', async () => { + const result = await service.mget([]); + expect(result).toEqual([]); + expect(mockRedis.mget).not.toHaveBeenCalled(); + }); + }); + describe('mset', () => { + it('should set multiple values', async () => { + const mockPipeline = { + set: jest.fn().mockReturnThis(), + exec: jest.fn().mockResolvedValue([]), + }; + mockRedis.pipeline.mockReturnValue(mockPipeline as unknown); + await service.mset([ + { key: 'key1', value: { id: 1 } }, + { key: 'key2', value: { id: 2 } }, + ], 60); + expect(mockRedis.pipeline).toHaveBeenCalled(); + expect(mockPipeline.exec).toHaveBeenCalled(); + }); + }); + describe('getStats', () => { + it('should return cache statistics', async () => { + mockRedis.info + .mockResolvedValueOnce('used_memory_human:1.5M\n') + .mockResolvedValueOnce('db0:keys=100\n') + .mockResolvedValueOnce('keyspace_hits:1000\nkeyspace_misses:100\n'); + const result = await service.getStats(); + expect(result).toEqual({ + keys: 100, + memory: '1.5M', + hits: 1000, + misses: 100, + }); + }); + }); + describe('generateKey', () => { + it('should generate key with prefix and parts', () => { + const result = service.generateKey('prefix', 'part1', 'part2'); + expect(result).toBe('prefix:part1:part2'); + }); + }); + describe('getTTLConstants', () => { + it('should return TTL constants', () => { + const result = service.getTTLConstants(); + expect(result).toHaveProperty('USER_SESSION'); + expect(result).toHaveProperty('COURSE_DETAILS'); + expect(result).toHaveProperty('SEARCH_RESULTS'); + }); }); - - it('should return null when key not found', async () => { - mockRedis.get.mockResolvedValue(null); - - const result = await service.get('test:key'); - - expect(result).toBeNull(); - }); - - it('should return raw string for non-JSON values', async () => { - mockRedis.get.mockResolvedValue('raw-string-value'); - - const result = await service.get('test:key'); - - expect(result).toBe('raw-string-value'); - }); - }); - - describe('set', () => { - it('should set value with TTL', async () => { - await service.set('test:key', { data: 'value' }, 60); - - expect(mockRedis.set).toHaveBeenCalledWith( - 'test:key', - JSON.stringify({ data: 'value' }), - 'EX', - 60, - ); - }); - - it('should set value without TTL when ttl is 0', async () => { - await service.set('test:key', { data: 'value' }, 0); - - expect(mockRedis.set).toHaveBeenCalledWith('test:key', JSON.stringify({ data: 'value' })); - }); - - it('should set string value directly', async () => { - await service.set('test:key', 'string-value'); - - expect(mockRedis.set).toHaveBeenCalledWith('test:key', 'string-value', 'EX', 300); - }); - }); - - describe('del', () => { - it('should delete key', async () => { - await service.del('test:key'); - - expect(mockRedis.del).toHaveBeenCalledWith('test:key'); - }); - }); - - describe('delPattern', () => { - it('should delete keys matching pattern', async () => { - mockRedis.scan.mockResolvedValueOnce(['0', ['key1', 'key2']]); - - const result = await service.delPattern('test:*'); - - expect(result).toBe(2); - expect(mockRedis.del).toHaveBeenCalledWith('key1', 'key2'); - }); - - it('should return 0 when no keys match', async () => { - mockRedis.scan.mockResolvedValueOnce(['0', []]); - - const result = await service.delPattern('test:*'); - - expect(result).toBe(0); - }); - }); - - describe('getOrSet', () => { - it('should return cached value when available', async () => { - mockRedis.get.mockResolvedValue(JSON.stringify({ cached: true })); - - const factory = jest.fn(); - const result = await service.getOrSet('test:key', factory, 60); - - expect(result).toEqual({ cached: true }); - expect(factory).not.toHaveBeenCalled(); - }); - - it('should call factory and cache result when not available', async () => { - mockRedis.get.mockResolvedValue(null); - const factory = jest.fn().mockResolvedValue({ new: 'data' }); - - const result = await service.getOrSet('test:key', factory, 60); - - expect(result).toEqual({ new: 'data' }); - expect(factory).toHaveBeenCalled(); - expect(mockRedis.set).toHaveBeenCalled(); - }); - }); - - describe('exists', () => { - it('should return true when key exists', async () => { - mockRedis.exists.mockResolvedValue(1); - - const result = await service.exists('test:key'); - - expect(result).toBe(true); - }); - - it('should return false when key does not exist', async () => { - mockRedis.exists.mockResolvedValue(0); - - const result = await service.exists('test:key'); - - expect(result).toBe(false); - }); - }); - - describe('getTtl', () => { - it('should return TTL of key', async () => { - mockRedis.ttl.mockResolvedValue(120); - - const result = await service.getTtl('test:key'); - - expect(result).toBe(120); - }); - }); - - describe('incr', () => { - it('should increment by 1 by default', async () => { - mockRedis.incr.mockResolvedValue(1); - - const result = await service.incr('counter'); - - expect(result).toBe(1); - expect(mockRedis.incr).toHaveBeenCalledWith('counter'); - }); - - it('should increment by specified amount', async () => { - mockRedis.incrby.mockResolvedValue(5); - - const result = await service.incr('counter', 5); - - expect(result).toBe(5); - expect(mockRedis.incrby).toHaveBeenCalledWith('counter', 5); - }); - }); - - describe('mget', () => { - it('should get multiple values', async () => { - mockRedis.mget.mockResolvedValue([ - JSON.stringify({ id: 1 }), - null, - JSON.stringify({ id: 3 }), - ]); - - const result = await service.mget(['key1', 'key2', 'key3']); - - expect(result).toEqual([{ id: 1 }, null, { id: 3 }]); - }); - - it('should return empty array for empty input', async () => { - const result = await service.mget([]); - - expect(result).toEqual([]); - expect(mockRedis.mget).not.toHaveBeenCalled(); - }); - }); - - describe('mset', () => { - it('should set multiple values', async () => { - const mockPipeline = { - set: jest.fn().mockReturnThis(), - exec: jest.fn().mockResolvedValue([]), - }; - mockRedis.pipeline.mockReturnValue(mockPipeline as any); - - await service.mset( - [ - { key: 'key1', value: { id: 1 } }, - { key: 'key2', value: { id: 2 } }, - ], - 60, - ); - - expect(mockRedis.pipeline).toHaveBeenCalled(); - expect(mockPipeline.exec).toHaveBeenCalled(); - }); - }); - - describe('getStats', () => { - it('should return cache statistics', async () => { - mockRedis.info - .mockResolvedValueOnce('used_memory_human:1.5M\n') - .mockResolvedValueOnce('db0:keys=100\n') - .mockResolvedValueOnce('keyspace_hits:1000\nkeyspace_misses:100\n'); - - const result = await service.getStats(); - - expect(result).toEqual({ - keys: 100, - memory: '1.5M', - hits: 1000, - misses: 100, - }); - }); - }); - - describe('generateKey', () => { - it('should generate key with prefix and parts', () => { - const result = service.generateKey('prefix', 'part1', 'part2'); - - expect(result).toBe('prefix:part1:part2'); - }); - }); - - describe('getTTLConstants', () => { - it('should return TTL constants', () => { - const result = service.getTTLConstants(); - - expect(result).toHaveProperty('USER_SESSION'); - expect(result).toHaveProperty('COURSE_DETAILS'); - expect(result).toHaveProperty('SEARCH_RESULTS'); - }); - }); }); diff --git a/src/caching/caching.service.ts b/src/caching/caching.service.ts index bc2fba34..afe33d92 100644 --- a/src/caching/caching.service.ts +++ b/src/caching/caching.service.ts @@ -2,330 +2,307 @@ import { Inject, Injectable, Logger, OnModuleDestroy } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import Redis from 'ioredis'; import { CACHE_REDIS_CLIENT, CACHE_TTL } from './caching.constants'; - export interface CacheOptions { - ttl?: number; - prefix?: string; + ttl?: number; + prefix?: string; } - @Injectable() export class CachingService implements OnModuleDestroy { - private readonly logger = new Logger(CachingService.name); - private readonly defaultTtl: number; - - constructor( - @Inject(CACHE_REDIS_CLIENT) private readonly redis: Redis, - private readonly configService: ConfigService, - ) { - this.defaultTtl = parseInt(this.configService.get('REDIS_TTL') || '300', 10); - } - - async onModuleDestroy(): Promise { - if (this.redis.status !== 'end') { - await this.redis.quit(); + private readonly logger = new Logger(CachingService.name); + private readonly defaultTtl: number; + constructor( + @Inject(CACHE_REDIS_CLIENT) + private readonly redis: Redis, private readonly configService: ConfigService) { + this.defaultTtl = parseInt(this.configService.get('REDIS_TTL') || '300', 10); } - } - - /** - * Get a value from cache - * @param key - Cache key - * @returns The cached value or null if not found - */ - async get(key: string): Promise { - try { - const data = await this.redis.get(key); - if (!data) { - return null; - } - - try { - return JSON.parse(data) as T; - } catch { - // Return raw string if not valid JSON - return data as unknown as T; - } - } catch (error) { - this.logger.error(`Failed to get cache key: ${key}`, error); - return null; + async onModuleDestroy(): Promise { + if (this.redis.status !== 'end') { + await this.redis.quit(); + } } - } - - /** - * Set a value in cache - * @param key - Cache key - * @param value - Value to cache - * @param ttl - Time to live in seconds (optional) - */ - async set(key: string, value: T, ttl?: number): Promise { - try { - const serializedValue = typeof value === 'string' ? value : JSON.stringify(value); - const effectiveTtl = ttl ?? this.defaultTtl; - - if (effectiveTtl > 0) { - await this.redis.set(key, serializedValue, 'EX', effectiveTtl); - } else { - await this.redis.set(key, serializedValue); - } - } catch (error) { - this.logger.error(`Failed to set cache key: ${key}`, error); + /** + * Get a value from cache + * @param key - Cache key + * @returns The cached value or null if not found + */ + async get(key: string): Promise { + try { + const data = await this.redis.get(key); + if (!data) { + return null; + } + try { + return JSON.parse(data) as T; + } + catch { + // Return raw string if not valid JSON + return data as unknown as T; + } + } + catch (error) { + this.logger.error(`Failed to get cache key: ${key}`, error); + return null; + } } - } - - /** - * Delete a key from cache - * @param key - Cache key to delete - */ - async del(key: string): Promise { - try { - await this.redis.del(key); - } catch (error) { - this.logger.error(`Failed to delete cache key: ${key}`, error); + /** + * Set a value in cache + * @param key - Cache key + * @param value - Value to cache + * @param ttl - Time to live in seconds (optional) + */ + async set(key: string, value: T, ttl?: number): Promise { + try { + const serializedValue = typeof value === 'string' ? value : JSON.stringify(value); + const effectiveTtl = ttl ?? this.defaultTtl; + if (effectiveTtl > 0) { + await this.redis.set(key, serializedValue, 'EX', effectiveTtl); + } + else { + await this.redis.set(key, serializedValue); + } + } + catch (error) { + this.logger.error(`Failed to set cache key: ${key}`, error); + } } - } - - /** - * Delete all keys matching a pattern - * @param pattern - Pattern to match (e.g., 'cache:course:*') - */ - async delPattern(pattern: string): Promise { - try { - const keys = await this.scanKeys(pattern); - if (keys.length === 0) { - return 0; - } - - await this.redis.del(...keys); - this.logger.debug(`Deleted ${keys.length} keys matching pattern: ${pattern}`); - return keys.length; - } catch (error) { - this.logger.error(`Failed to delete cache pattern: ${pattern}`, error); - return 0; + /** + * Delete a key from cache + * @param key - Cache key to delete + */ + async del(key: string): Promise { + try { + await this.redis.del(key); + } + catch (error) { + this.logger.error(`Failed to delete cache key: ${key}`, error); + } } - } - - /** - * Get a value from cache, or execute factory function and cache the result - * @param key - Cache key - * @param factory - Function to execute if value not in cache - * @param ttl - Time to live in seconds (optional) - * @returns The cached or freshly computed value - */ - async getOrSet(key: string, factory: () => Promise, ttl?: number): Promise { - const cached = await this.get(key); - if (cached !== null) { - return cached; + /** + * Delete all keys matching a pattern + * @param pattern - Pattern to match (e.g., 'cache:course:*') + */ + async delPattern(pattern: string): Promise { + try { + const keys = await this.scanKeys(pattern); + if (keys.length === 0) { + return 0; + } + await this.redis.del(...keys); + this.logger.debug(`Deleted ${keys.length} keys matching pattern: ${pattern}`); + return keys.length; + } + catch (error) { + this.logger.error(`Failed to delete cache pattern: ${pattern}`, error); + return 0; + } } - - const value = await factory(); - await this.set(key, value, ttl); - return value; - } - - /** - * Check if a key exists in cache - * @param key - Cache key - * @returns True if key exists - */ - async exists(key: string): Promise { - try { - const result = await this.redis.exists(key); - return result === 1; - } catch (error) { - this.logger.error(`Failed to check cache key existence: ${key}`, error); - return false; + /** + * Get a value from cache, or execute factory function and cache the result + * @param key - Cache key + * @param factory - Function to execute if value not in cache + * @param ttl - Time to live in seconds (optional) + * @returns The cached or freshly computed value + */ + async getOrSet(key: string, factory: () => Promise, ttl?: number): Promise { + const cached = await this.get(key); + if (cached !== null) { + return cached; + } + const value = await factory(); + await this.set(key, value, ttl); + return value; } - } - - /** - * Get the TTL of a key - * @param key - Cache key - * @returns TTL in seconds, -1 if no expiry, -2 if key doesn't exist - */ - async getTtl(key: string): Promise { - try { - return await this.redis.ttl(key); - } catch (error) { - this.logger.error(`Failed to get TTL for key: ${key}`, error); - return -2; + /** + * Check if a key exists in cache + * @param key - Cache key + * @returns True if key exists + */ + async exists(key: string): Promise { + try { + const result = await this.redis.exists(key); + return result === 1; + } + catch (error) { + this.logger.error(`Failed to check cache key existence: ${key}`, error); + return false; + } } - } - - /** - * Set the TTL of an existing key - * @param key - Cache key - * @param ttl - New TTL in seconds - */ - async expire(key: string, ttl: number): Promise { - try { - await this.redis.expire(key, ttl); - } catch (error) { - this.logger.error(`Failed to set expiry for key: ${key}`, error); + /** + * Get the TTL of a key + * @param key - Cache key + * @returns TTL in seconds, -1 if no expiry, -2 if key doesn't exist + */ + async getTtl(key: string): Promise { + try { + return await this.redis.ttl(key); + } + catch (error) { + this.logger.error(`Failed to get TTL for key: ${key}`, error); + return -2; + } } - } - - /** - * Increment a counter - * @param key - Cache key - * @param increment - Amount to increment (default: 1) - * @returns New value after increment - */ - async incr(key: string, increment = 1): Promise { - try { - if (increment === 1) { - return await this.redis.incr(key); - } - return await this.redis.incrby(key, increment); - } catch (error) { - this.logger.error(`Failed to increment key: ${key}`, error); - return 0; + /** + * Set the TTL of an existing key + * @param key - Cache key + * @param ttl - New TTL in seconds + */ + async expire(key: string, ttl: number): Promise { + try { + await this.redis.expire(key, ttl); + } + catch (error) { + this.logger.error(`Failed to set expiry for key: ${key}`, error); + } } - } - - /** - * Get multiple values at once - * @param keys - Array of cache keys - * @returns Array of values (null for missing keys) - */ - async mget(keys: string[]): Promise> { - if (keys.length === 0) { - return []; + /** + * Increment a counter + * @param key - Cache key + * @param increment - Amount to increment (default: 1) + * @returns New value after increment + */ + async incr(key: string, increment = 1): Promise { + try { + if (increment === 1) { + return await this.redis.incr(key); + } + return await this.redis.incrby(key, increment); + } + catch (error) { + this.logger.error(`Failed to increment key: ${key}`, error); + return 0; + } + } + /** + * Get multiple values at once + * @param keys - Array of cache keys + * @returns Array of values (null for missing keys) + */ + async mget(keys: string[]): Promise> { + if (keys.length === 0) { + return []; + } + try { + const values = await this.redis.mget(...keys); + return values.map((data) => { + if (!data) + return null; + try { + return JSON.parse(data) as T; + } + catch { + return data as unknown as T; + } + }); + } + catch (error) { + this.logger.error('Failed to get multiple cache keys', error); + return keys.map(() => null); + } } - - try { - const values = await this.redis.mget(...keys); - return values.map((data) => { - if (!data) return null; + /** + * Set multiple values at once + * @param entries - Array of key-value pairs + * @param ttl - Time to live in seconds (optional) + */ + async mset(entries: Array<{ + key: string; + value: T; + }>, ttl?: number): Promise { + if (entries.length === 0) { + return; + } try { - return JSON.parse(data) as T; - } catch { - return data as unknown as T; + const pipeline = this.redis.pipeline(); + for (const { key, value } of entries) { + const serializedValue = typeof value === 'string' ? value : JSON.stringify(value); + if (ttl && ttl > 0) { + pipeline.set(key, serializedValue, 'EX', ttl); + } + else { + pipeline.set(key, serializedValue); + } + } + await pipeline.exec(); + } + catch (error) { + this.logger.error('Failed to set multiple cache keys', error); } - }); - } catch (error) { - this.logger.error('Failed to get multiple cache keys', error); - return keys.map(() => null); } - } - - /** - * Set multiple values at once - * @param entries - Array of key-value pairs - * @param ttl - Time to live in seconds (optional) - */ - async mset(entries: Array<{ key: string; value: T }>, ttl?: number): Promise { - if (entries.length === 0) { - return; + /** + * Scan keys matching a pattern + * @param pattern - Pattern to match + * @param count - Approximate number of keys to return per iteration + * @returns Array of matching keys + */ + private async scanKeys(pattern: string, count = 100): Promise { + const keys: string[] = []; + let cursor = '0'; + do { + const [nextCursor, matchedKeys] = await this.redis.scan(cursor, 'MATCH', pattern, 'COUNT', count); + cursor = nextCursor; + keys.push(...matchedKeys); + } while (cursor !== '0'); + return keys; } - - try { - const pipeline = this.redis.pipeline(); - - for (const { key, value } of entries) { - const serializedValue = typeof value === 'string' ? value : JSON.stringify(value); - - if (ttl && ttl > 0) { - pipeline.set(key, serializedValue, 'EX', ttl); - } else { - pipeline.set(key, serializedValue); + /** + * Get cache statistics + * @returns Cache statistics object + */ + async getStats(): Promise<{ + keys: number; + memory: string; + hits: number; + misses: number; + }> { + try { + const info = await this.redis.info('memory'); + const keyspaceInfo = await this.redis.info('keyspace'); + // Parse memory usage + const memoryMatch = info.match(/used_memory_human:(\S+)/); + const memory = memoryMatch ? memoryMatch[1] : 'unknown'; + // Parse key count + const dbMatch = keyspaceInfo.match(/db\d+:keys=(\d+)/); + const keys = dbMatch ? parseInt(dbMatch[1], 10) : 0; + // Parse hit/miss stats + const statsInfo = await this.redis.info('stats'); + const hitsMatch = statsInfo.match(/keyspace_hits:(\d+)/); + const missesMatch = statsInfo.match(/keyspace_misses:(\d+)/); + return { + keys, + memory, + hits: hitsMatch ? parseInt(hitsMatch[1], 10) : 0, + misses: missesMatch ? parseInt(missesMatch[1], 10) : 0, + }; + } + catch (error) { + this.logger.error('Failed to get cache stats', error); + return { + keys: 0, + memory: 'unknown', + hits: 0, + misses: 0, + }; } - } - - await pipeline.exec(); - } catch (error) { - this.logger.error('Failed to set multiple cache keys', error); } - } - - /** - * Scan keys matching a pattern - * @param pattern - Pattern to match - * @param count - Approximate number of keys to return per iteration - * @returns Array of matching keys - */ - private async scanKeys(pattern: string, count = 100): Promise { - const keys: string[] = []; - let cursor = '0'; - - do { - const [nextCursor, matchedKeys] = await this.redis.scan( - cursor, - 'MATCH', - pattern, - 'COUNT', - count, - ); - cursor = nextCursor; - keys.push(...matchedKeys); - } while (cursor !== '0'); - - return keys; - } - - /** - * Get cache statistics - * @returns Cache statistics object - */ - async getStats(): Promise<{ - keys: number; - memory: string; - hits: number; - misses: number; - }> { - try { - const info = await this.redis.info('memory'); - const keyspaceInfo = await this.redis.info('keyspace'); - - // Parse memory usage - const memoryMatch = info.match(/used_memory_human:(\S+)/); - const memory = memoryMatch ? memoryMatch[1] : 'unknown'; - - // Parse key count - const dbMatch = keyspaceInfo.match(/db\d+:keys=(\d+)/); - const keys = dbMatch ? parseInt(dbMatch[1], 10) : 0; - - // Parse hit/miss stats - const statsInfo = await this.redis.info('stats'); - const hitsMatch = statsInfo.match(/keyspace_hits:(\d+)/); - const missesMatch = statsInfo.match(/keyspace_misses:(\d+)/); - - return { - keys, - memory, - hits: hitsMatch ? parseInt(hitsMatch[1], 10) : 0, - misses: missesMatch ? parseInt(missesMatch[1], 10) : 0, - }; - } catch (error) { - this.logger.error('Failed to get cache stats', error); - return { - keys: 0, - memory: 'unknown', - hits: 0, - misses: 0, - }; + /** + * Clear all cache keys with the application prefix + */ + async clearAll(): Promise { + return this.delPattern('cache:*'); + } + /** + * Generate a cache key with prefix + * @param prefix - Key prefix + * @param parts - Key parts to join + * @returns Formatted cache key + */ + generateKey(prefix: string, ...parts: Array): string { + return `${prefix}:${parts.join(':')}`; + } + /** + * Get TTL constants for external use + */ + getTTLConstants(): typeof CACHE_TTL { + return CACHE_TTL; } - } - - /** - * Clear all cache keys with the application prefix - */ - async clearAll(): Promise { - return this.delPattern('cache:*'); - } - - /** - * Generate a cache key with prefix - * @param prefix - Key prefix - * @param parts - Key parts to join - * @returns Formatted cache key - */ - generateKey(prefix: string, ...parts: Array): string { - return `${prefix}:${parts.join(':')}`; - } - - /** - * Get TTL constants for external use - */ - getTTLConstants(): typeof CACHE_TTL { - return CACHE_TTL; - } } diff --git a/src/caching/decorators/cache.decorator.ts b/src/caching/decorators/cache.decorator.ts index 88e2b09e..df954d36 100644 --- a/src/caching/decorators/cache.decorator.ts +++ b/src/caching/decorators/cache.decorator.ts @@ -1,54 +1,46 @@ import { SetMetadata } from '@nestjs/common'; - export const CACHE_KEY_METADATA = 'cache:key'; export const CACHE_TTL_METADATA = 'cache:ttl'; export const CACHE_EVICT_METADATA = 'cache:evict'; export const CACHE_PREFIX_METADATA = 'cache:prefix'; export const CACHE_CONDITION_METADATA = 'cache:condition'; - /** * Options for cacheable decorator */ export interface CacheableOptions { - /** - * Time to live in seconds - */ - ttl?: number; - - /** - * Cache key prefix - */ - prefix?: string; - - /** - * Custom cache key generator - * If provided, this function will be used to generate the cache key - */ - keyGenerator?: (...args: any[]) => string; - - /** - * Condition to determine if result should be cached - * Return true to cache, false to skip caching - */ - condition?: (...args: any[]) => boolean; + /** + * Time to live in seconds + */ + ttl?: number; + /** + * Cache key prefix + */ + prefix?: string; + /** + * Custom cache key generator + * If provided, this function will be used to generate the cache key + */ + keyGenerator?: (...args: unknown[]) => string; + /** + * Condition to determine if result should be cached + * Return true to cache, false to skip caching + */ + condition?: (...args: unknown[]) => boolean; } - /** * Options for cache evict decorator */ export interface CacheEvictOptions { - /** - * Pattern(s) to evict (supports wildcards) - */ - patterns: string | string[]; - - /** - * Whether to evict before method execution - * Default: false (evict after successful execution) - */ - beforeInvocation?: boolean; + /** + * Pattern(s) to evict (supports wildcards) + */ + patterns: string | string[]; + /** + * Whether to evict before method execution + * Default: false (evict after successful execution) + */ + beforeInvocation?: boolean; } - /** * Decorator to cache method result * @@ -69,34 +61,23 @@ export interface CacheEvictOptions { * ``` */ export function Cacheable(ttlOrOptions?: number | CacheableOptions): MethodDecorator { - const options: CacheableOptions = - typeof ttlOrOptions === 'number' ? { ttl: ttlOrOptions } : (ttlOrOptions ?? {}); - - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - if (options.ttl) { - SetMetadata(CACHE_TTL_METADATA, options.ttl)(target, propertyKey, descriptor); - } - - if (options.prefix) { - SetMetadata(CACHE_PREFIX_METADATA, options.prefix)(target, propertyKey, descriptor); - } - - if (options.keyGenerator) { - SetMetadata(CACHE_KEY_METADATA, options.keyGenerator)(target, propertyKey, descriptor); - } - - if (options.condition) { - SetMetadata(CACHE_CONDITION_METADATA, options.condition)(target, propertyKey, descriptor); - } - - return descriptor; - }; + const options: CacheableOptions = typeof ttlOrOptions === 'number' ? { ttl: ttlOrOptions } : (ttlOrOptions ?? {}); + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + if (options.ttl) { + SetMetadata(CACHE_TTL_METADATA, options.ttl)(target, propertyKey, descriptor); + } + if (options.prefix) { + SetMetadata(CACHE_PREFIX_METADATA, options.prefix)(target, propertyKey, descriptor); + } + if (options.keyGenerator) { + SetMetadata(CACHE_KEY_METADATA, options.keyGenerator)(target, propertyKey, descriptor); + } + if (options.condition) { + SetMetadata(CACHE_CONDITION_METADATA, options.condition)(target, propertyKey, descriptor); + } + return descriptor; + }; } - /** * Decorator to evict cache entries when method is executed * @@ -115,26 +96,17 @@ export function Cacheable(ttlOrOptions?: number | CacheableOptions): MethodDecor * } * ``` */ -export function CacheEvict( - patternsOrOptions: string | string[] | CacheEvictOptions, -): MethodDecorator { - const options: CacheEvictOptions = - typeof patternsOrOptions === 'string' - ? { patterns: [patternsOrOptions] } - : Array.isArray(patternsOrOptions) - ? { patterns: patternsOrOptions } - : patternsOrOptions; - - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - SetMetadata(CACHE_EVICT_METADATA, options)(target, propertyKey, descriptor); - return descriptor; - }; +export function CacheEvict(patternsOrOptions: string | string[] | CacheEvictOptions): MethodDecorator { + const options: CacheEvictOptions = typeof patternsOrOptions === 'string' + ? { patterns: [patternsOrOptions] } + : Array.isArray(patternsOrOptions) + ? { patterns: patternsOrOptions } + : patternsOrOptions; + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + SetMetadata(CACHE_EVICT_METADATA, options)(target, propertyKey, descriptor); + return descriptor; + }; } - /** * Decorator to set a custom cache key for a method * @@ -155,17 +127,12 @@ export function CacheEvict( * } * ``` */ -export function CacheKey(key: string | ((...args: any[]) => string)): MethodDecorator { - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - SetMetadata(CACHE_KEY_METADATA, key)(target, propertyKey, descriptor); - return descriptor; - }; +export function CacheKey(key: string | ((...args: unknown[]) => string)): MethodDecorator { + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + SetMetadata(CACHE_KEY_METADATA, key)(target, propertyKey, descriptor); + return descriptor; + }; } - /** * Decorator to set TTL for a cached method * @@ -181,16 +148,11 @@ export function CacheKey(key: string | ((...args: any[]) => string)): MethodDeco * ``` */ export function CacheTTL(ttl: number): MethodDecorator { - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - SetMetadata(CACHE_TTL_METADATA, ttl)(target, propertyKey, descriptor); - return descriptor; - }; + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + SetMetadata(CACHE_TTL_METADATA, ttl)(target, propertyKey, descriptor); + return descriptor; + }; } - /** * Decorator to set cache prefix for a method * @@ -206,16 +168,11 @@ export function CacheTTL(ttl: number): MethodDecorator { * ``` */ export function CachePrefix(prefix: string): MethodDecorator { - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - SetMetadata(CACHE_PREFIX_METADATA, prefix)(target, propertyKey, descriptor); - return descriptor; - }; + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + SetMetadata(CACHE_PREFIX_METADATA, prefix)(target, propertyKey, descriptor); + return descriptor; + }; } - /** * Decorator to conditionally cache based on method arguments * @@ -230,15 +187,9 @@ export function CachePrefix(prefix: string): MethodDecorator { * } * ``` */ -export function CacheCondition( - condition: (result: any, ...args: any[]) => boolean, -): MethodDecorator { - return ( - target: object, - propertyKey: string | symbol, - descriptor: TypedPropertyDescriptor, - ) => { - SetMetadata(CACHE_CONDITION_METADATA, condition)(target, propertyKey, descriptor); - return descriptor; - }; +export function CacheCondition(condition: (result: unknown, ...args: unknown[]) => boolean): MethodDecorator { + return (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => { + SetMetadata(CACHE_CONDITION_METADATA, condition)(target, propertyKey, descriptor); + return descriptor; + }; } diff --git a/src/caching/interceptors/cache.interceptor.ts b/src/caching/interceptors/cache.interceptor.ts index a79d4cb7..73e11646 100644 --- a/src/caching/interceptors/cache.interceptor.ts +++ b/src/caching/interceptors/cache.interceptor.ts @@ -1,213 +1,150 @@ -import { - Injectable, - NestInterceptor, - ExecutionContext, - CallHandler, - Inject, - Optional, -} from '@nestjs/common'; +import { Injectable, NestInterceptor, ExecutionContext, CallHandler, Inject, Optional, } from '@nestjs/common'; import { Reflector } from '@nestjs/core'; import { Observable, of, from } from 'rxjs'; import { mergeMap, tap } from 'rxjs/operators'; import { CachingService } from '../caching.service'; import { CacheAnalyticsService } from '../analytics/cache-analytics.service'; -import { - CACHE_KEY_METADATA, - CACHE_TTL_METADATA, - CACHE_EVICT_METADATA, - CACHE_PREFIX_METADATA, - CACHE_CONDITION_METADATA, - CacheEvictOptions, -} from '../decorators/cache.decorator'; +import { CACHE_KEY_METADATA, CACHE_TTL_METADATA, CACHE_EVICT_METADATA, CACHE_PREFIX_METADATA, CACHE_CONDITION_METADATA, CacheEvictOptions, } from '../decorators/cache.decorator'; import { CACHE_TTL } from '../caching.constants'; - export interface CacheInterceptorOptions { - /** - * Default TTL in seconds - */ - defaultTtl?: number; - - /** - * Default key prefix - */ - defaultPrefix?: string; - - /** - * Methods to cache (default: GET) - */ - methods?: string[]; - - /** - * Whether to track analytics - */ - trackAnalytics?: boolean; + /** + * Default TTL in seconds + */ + defaultTtl?: number; + /** + * Default key prefix + */ + defaultPrefix?: string; + /** + * Methods to cache (default: GET) + */ + methods?: string[]; + /** + * Whether to track analytics + */ + trackAnalytics?: boolean; } - @Injectable() export class CacheInterceptor implements NestInterceptor { - private readonly defaultTtl: number; - private readonly defaultPrefix: string; - private readonly cachedMethods: string[]; - private readonly trackAnalytics: boolean; - - constructor( - private readonly cachingService: CachingService, - private readonly reflector: Reflector, - @Optional() @Inject('CACHE_INTERCEPTOR_OPTIONS') options?: CacheInterceptorOptions, - @Optional() private readonly analyticsService?: CacheAnalyticsService, - ) { - this.defaultTtl = options?.defaultTtl ?? CACHE_TTL.COURSE_DETAILS; - this.defaultPrefix = options?.defaultPrefix ?? 'cache:http'; - this.cachedMethods = options?.methods ?? ['GET']; - this.trackAnalytics = options?.trackAnalytics ?? true; - } - - intercept(context: ExecutionContext, next: CallHandler): Observable { - const request = context.switchToHttp().getRequest(); - - // Only cache specified HTTP methods - if (!request || !this.cachedMethods.includes(request.method)) { - return next.handle(); + private readonly defaultTtl: number; + private readonly defaultPrefix: string; + private readonly cachedMethods: string[]; + private readonly trackAnalytics: boolean; + constructor(private readonly cachingService: CachingService, private readonly reflector: Reflector, + @Optional() + @Inject('CACHE_INTERCEPTOR_OPTIONS') + options?: CacheInterceptorOptions, + @Optional() + private readonly analyticsService?: CacheAnalyticsService) { + this.defaultTtl = options?.defaultTtl ?? CACHE_TTL.COURSE_DETAILS; + this.defaultPrefix = options?.defaultPrefix ?? 'cache:http'; + this.cachedMethods = options?.methods ?? ['GET']; + this.trackAnalytics = options?.trackAnalytics ?? true; } - - // Get metadata from decorator - const handler = context.getHandler(); - const ttl = this.reflector.get(CACHE_TTL_METADATA, handler) ?? this.defaultTtl; - const prefix = this.reflector.get(CACHE_PREFIX_METADATA, handler) ?? this.defaultPrefix; - const customKey = this.reflector.get string)>( - CACHE_KEY_METADATA, - handler, - ); - const evictOptions = this.reflector.get(CACHE_EVICT_METADATA, handler); - const condition = this.reflector.get<(...args: any[]) => boolean>( - CACHE_CONDITION_METADATA, - handler, - ); - - // Generate cache key - const cacheKey = this.generateCacheKey(request, prefix, customKey); - - // Handle cache eviction - if (evictOptions) { - return this.handleEviction(next, evictOptions, cacheKey); - } - - // Check condition - if (condition && !condition(request.params, request.query, request.body)) { - return next.handle(); + intercept(context: ExecutionContext, next: CallHandler): Observable { + const request = context.switchToHttp().getRequest(); + // Only cache specified HTTP methods + if (!request || !this.cachedMethods.includes(request.method)) { + return next.handle(); + } + // Get metadata from decorator + const handler = context.getHandler(); + const ttl = this.reflector.get(CACHE_TTL_METADATA, handler) ?? this.defaultTtl; + const prefix = this.reflector.get(CACHE_PREFIX_METADATA, handler) ?? this.defaultPrefix; + const customKey = this.reflector.get string)>(CACHE_KEY_METADATA, handler); + const evictOptions = this.reflector.get(CACHE_EVICT_METADATA, handler); + const condition = this.reflector.get<(...args: unknown[]) => boolean>(CACHE_CONDITION_METADATA, handler); + // Generate cache key + const cacheKey = this.generateCacheKey(request, prefix, customKey); + // Handle cache eviction + if (evictOptions) { + return this.handleEviction(next, evictOptions, cacheKey); + } + // Check condition + if (condition && !condition(request.params, request.query, request.body)) { + return next.handle(); + } + // Try to get from cache + return from(this.cachingService.get(cacheKey)).pipe(mergeMap((cachedResponse) => { + if (cachedResponse !== null) { + // Cache hit + if (this.trackAnalytics && this.analyticsService) { + this.analyticsService.recordHit(cacheKey); + } + return of(cachedResponse); + } + // Cache miss - execute handler and cache result + return next.handle().pipe(tap({ + next: (response) => { + // Cache the response + if (this.trackAnalytics && this.analyticsService) { + this.analyticsService.recordMiss(cacheKey); + } + from(this.cachingService.set(cacheKey, response, ttl)).subscribe(); + }, + })); + })); } - - // Try to get from cache - return from(this.cachingService.get(cacheKey)).pipe( - mergeMap((cachedResponse) => { - if (cachedResponse !== null) { - // Cache hit - if (this.trackAnalytics && this.analyticsService) { - this.analyticsService.recordHit(cacheKey); - } - return of(cachedResponse); + /** + * Handle cache eviction before or after method execution + */ + private handleEviction(next: CallHandler, evictOptions: CacheEvictOptions, _currentKey: string): Observable { + const patterns = Array.isArray(evictOptions.patterns) + ? evictOptions.patterns + : [evictOptions.patterns]; + if (evictOptions.beforeInvocation) { + // Evict before method execution + return from(this.evictPatterns(patterns)).pipe(mergeMap(() => next.handle())); } - - // Cache miss - execute handler and cache result - return next.handle().pipe( - tap({ - next: (response) => { - // Cache the response - if (this.trackAnalytics && this.analyticsService) { - this.analyticsService.recordMiss(cacheKey); - } - from(this.cachingService.set(cacheKey, response, ttl)).subscribe(); + // Evict after successful method execution + return next.handle().pipe(tap({ + next: () => { + from(this.evictPatterns(patterns)).subscribe(); }, - }), - ); - }), - ); - } - - /** - * Handle cache eviction before or after method execution - */ - private handleEviction( - next: CallHandler, - evictOptions: CacheEvictOptions, - _currentKey: string, - ): Observable { - const patterns = Array.isArray(evictOptions.patterns) - ? evictOptions.patterns - : [evictOptions.patterns]; - - if (evictOptions.beforeInvocation) { - // Evict before method execution - return from(this.evictPatterns(patterns)).pipe(mergeMap(() => next.handle())); + })); } - - // Evict after successful method execution - return next.handle().pipe( - tap({ - next: () => { - from(this.evictPatterns(patterns)).subscribe(); - }, - }), - ); - } - - /** - * Evict cache entries matching patterns - */ - private async evictPatterns(patterns: string[]): Promise { - for (const pattern of patterns) { - await this.cachingService.delPattern(pattern); + /** + * Evict cache entries matching patterns + */ + private async evictPatterns(patterns: string[]): Promise { + for (const pattern of patterns) { + await this.cachingService.delPattern(pattern); + } } - } - - /** - * Generate a cache key from request - */ - private generateCacheKey( - request: any, - prefix: string, - customKey?: string | ((...args: any[]) => string), - ): string { - if (customKey) { - if (typeof customKey === 'function') { - return customKey(request.params, request.query, request.body); - } - return customKey; + /** + * Generate a cache key from request + */ + private generateCacheKey(request: unknown, prefix: string, customKey?: string | ((...args: unknown[]) => string)): string { + if (customKey) { + if (typeof customKey === 'function') { + return customKey(request.params, request.query, request.body); + } + return customKey; + } + const route = request.route?.path ?? request.url; + const params = JSON.stringify(request.params ?? {}); + const query = JSON.stringify(request.query ?? {}); + const userId = request.user?.id ?? 'anonymous'; + // Create a hash-like key from route, params, query, and user + const keyParts = [prefix, route.replace(/\//g, ':'), userId, this.hashString(params + query)]; + return keyParts.filter(Boolean).join(':'); } - - const route = request.route?.path ?? request.url; - const params = JSON.stringify(request.params ?? {}); - const query = JSON.stringify(request.query ?? {}); - const userId = request.user?.id ?? 'anonymous'; - - // Create a hash-like key from route, params, query, and user - const keyParts = [prefix, route.replace(/\//g, ':'), userId, this.hashString(params + query)]; - - return keyParts.filter(Boolean).join(':'); - } - - /** - * Simple string hash for cache key generation - */ - private hashString(str: string): string { - let hash = 0; - for (let i = 0; i < str.length; i++) { - const char = str.charCodeAt(i); - hash = (hash << 5) - hash + char; - hash = hash & hash; // Convert to 32-bit integer + /** + * Simple string hash for cache key generation + */ + private hashString(str: string): string { + let hash = 0; + for (let i = 0; i < str.length; i++) { + const char = str.charCodeAt(i); + hash = (hash << 5) - hash + char; + hash = hash & hash; // Convert to 32-bit integer + } + return Math.abs(hash).toString(36); } - return Math.abs(hash).toString(36); - } } - /** * Factory function to create a configured cache interceptor */ -export function createCacheInterceptor( - cachingService: CachingService, - reflector: Reflector, - options?: CacheInterceptorOptions, - analyticsService?: CacheAnalyticsService, -): CacheInterceptor { - return new CacheInterceptor(cachingService, reflector, options, analyticsService); +export function createCacheInterceptor(cachingService: CachingService, reflector: Reflector, options?: CacheInterceptorOptions, analyticsService?: CacheAnalyticsService): CacheInterceptor { + return new CacheInterceptor(cachingService, reflector, options, analyticsService); } diff --git a/src/caching/invalidation/invalidation.service.spec.ts b/src/caching/invalidation/invalidation.service.spec.ts index 97463514..8e5e6d2c 100644 --- a/src/caching/invalidation/invalidation.service.spec.ts +++ b/src/caching/invalidation/invalidation.service.spec.ts @@ -4,176 +4,138 @@ import { CacheInvalidationService } from './invalidation.service'; import { CachingService } from '../caching.service'; import { CacheStrategiesService } from '../strategies/cache-strategies.service'; import { CACHE_EVENTS } from '../caching.constants'; - describe('CacheInvalidationService', () => { - let service: CacheInvalidationService; - let cachingService: jest.Mocked; - let eventEmitter: jest.Mocked; - - beforeEach(async () => { - const mockCachingService = { - delPattern: jest.fn(), - del: jest.fn(), - clearAll: jest.fn(), - }; - - const mockEventEmitter = { - emit: jest.fn(), - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - CacheInvalidationService, - CacheStrategiesService, - { - provide: CachingService, - useValue: mockCachingService, - }, - { - provide: EventEmitter2, - useValue: mockEventEmitter, - }, - ], - }).compile(); - - service = module.get(CacheInvalidationService); - cachingService = module.get(CachingService); - eventEmitter = module.get(EventEmitter2); - }); - - describe('invalidateByPattern', () => { - it('should invalidate by pattern', async () => { - cachingService.delPattern.mockResolvedValue(5); - - const result = await service.invalidateByPattern('cache:course:*'); - - expect(result.pattern).toBe('cache:course:*'); - expect(result.keysDeleted).toBe(5); - expect(cachingService.delPattern).toHaveBeenCalledWith('cache:course:*'); + let service: CacheInvalidationService; + let cachingService: jest.Mocked; + let eventEmitter: jest.Mocked; + beforeEach(async () => { + const mockCachingService = { + delPattern: jest.fn(), + del: jest.fn(), + clearAll: jest.fn(), + }; + const mockEventEmitter = { + emit: jest.fn(), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + CacheInvalidationService, + CacheStrategiesService, + { + provide: CachingService, + useValue: mockCachingService, + }, + { + provide: EventEmitter2, + useValue: mockEventEmitter, + }, + ], + }).compile(); + service = module.get(CacheInvalidationService); + cachingService = module.get(CachingService); + eventEmitter = module.get(EventEmitter2); }); - }); - - describe('invalidateByPatterns', () => { - it('should invalidate multiple patterns', async () => { - cachingService.delPattern.mockResolvedValue(3); - - const results = await service.invalidateByPatterns(['cache:course:*', 'cache:user:*']); - - expect(results).toHaveLength(2); - expect(cachingService.delPattern).toHaveBeenCalledTimes(2); + describe('invalidateByPattern', () => { + it('should invalidate by pattern', async () => { + cachingService.delPattern.mockResolvedValue(5); + const result = await service.invalidateByPattern('cache:course:*'); + expect(result.pattern).toBe('cache:course:*'); + expect(result.keysDeleted).toBe(5); + expect(cachingService.delPattern).toHaveBeenCalledWith('cache:course:*'); + }); }); - }); - - describe('invalidateByTag', () => { - it('should invalidate by tag', async () => { - service.registerKeyWithTag('courses', 'cache:course:1'); - service.registerKeyWithTag('courses', 'cache:course:2'); - - const result = await service.invalidateByTag('courses'); - - expect(result).toBe(2); - expect(cachingService.del).toHaveBeenCalledTimes(2); + describe('invalidateByPatterns', () => { + it('should invalidate multiple patterns', async () => { + cachingService.delPattern.mockResolvedValue(3); + const results = await service.invalidateByPatterns(['cache:course:*', 'cache:user:*']); + expect(results).toHaveLength(2); + expect(cachingService.delPattern).toHaveBeenCalledTimes(2); + }); }); - - it('should return 0 for non-existent tag', async () => { - const result = await service.invalidateByTag('nonexistent'); - - expect(result).toBe(0); + describe('invalidateByTag', () => { + it('should invalidate by tag', async () => { + service.registerKeyWithTag('courses', 'cache:course:1'); + service.registerKeyWithTag('courses', 'cache:course:2'); + const result = await service.invalidateByTag('courses'); + expect(result).toBe(2); + expect(cachingService.del).toHaveBeenCalledTimes(2); + }); + it('should return 0 for non-existent tag', async () => { + const result = await service.invalidateByTag('nonexistent'); + expect(result).toBe(0); + }); }); - }); - - describe('registerKeyWithTag', () => { - it('should register key with tag', () => { - service.registerKeyWithTag('courses', 'cache:course:1'); - - const stats = service.getStats(); - expect(stats.registeredTags).toBe(1); - expect(stats.totalTrackedKeys).toBe(1); + describe('registerKeyWithTag', () => { + it('should register key with tag', () => { + service.registerKeyWithTag('courses', 'cache:course:1'); + const stats = service.getStats(); + expect(stats.registeredTags).toBe(1); + expect(stats.totalTrackedKeys).toBe(1); + }); }); - }); - - describe('unregisterKeyFromTag', () => { - it('should unregister key from tag', () => { - service.registerKeyWithTag('courses', 'cache:course:1'); - service.unregisterKeyFromTag('courses', 'cache:course:1'); - - const stats = service.getStats(); - expect(stats.totalTrackedKeys).toBe(0); + describe('unregisterKeyFromTag', () => { + it('should unregister key from tag', () => { + service.registerKeyWithTag('courses', 'cache:course:1'); + service.unregisterKeyFromTag('courses', 'cache:course:1'); + const stats = service.getStats(); + expect(stats.totalTrackedKeys).toBe(0); + }); }); - }); - - describe('invalidateCourse', () => { - it('should invalidate course cache', async () => { - cachingService.delPattern.mockResolvedValue(5); - - const results = await service.invalidateCourse('course-123'); - - expect(results.length).toBeGreaterThan(0); - expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.COURSE_UPDATED, { - courseId: 'course-123', - }); + describe('invalidateCourse', () => { + it('should invalidate course cache', async () => { + cachingService.delPattern.mockResolvedValue(5); + const results = await service.invalidateCourse('course-123'); + expect(results.length).toBeGreaterThan(0); + expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.COURSE_UPDATED, { + courseId: 'course-123', + }); + }); }); - }); - - describe('invalidateUser', () => { - it('should invalidate user cache', async () => { - cachingService.delPattern.mockResolvedValue(3); - - const results = await service.invalidateUser('user-123'); - - expect(results.length).toBeGreaterThan(0); - expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.USER_UPDATED, { - userId: 'user-123', - }); + describe('invalidateUser', () => { + it('should invalidate user cache', async () => { + cachingService.delPattern.mockResolvedValue(3); + const results = await service.invalidateUser('user-123'); + expect(results.length).toBeGreaterThan(0); + expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.USER_UPDATED, { + userId: 'user-123', + }); + }); }); - }); - - describe('invalidateEnrollment', () => { - it('should invalidate enrollment cache', async () => { - cachingService.delPattern.mockResolvedValue(2); - - const results = await service.invalidateEnrollment('enroll-123', 'course-123'); - - expect(results.length).toBeGreaterThan(0); - expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.ENROLLMENT_UPDATED, { - enrollmentId: 'enroll-123', - courseId: 'course-123', - }); + describe('invalidateEnrollment', () => { + it('should invalidate enrollment cache', async () => { + cachingService.delPattern.mockResolvedValue(2); + const results = await service.invalidateEnrollment('enroll-123', 'course-123'); + expect(results.length).toBeGreaterThan(0); + expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.ENROLLMENT_UPDATED, { + enrollmentId: 'enroll-123', + courseId: 'course-123', + }); + }); }); - }); - - describe('invalidateSearch', () => { - it('should invalidate search cache', async () => { - cachingService.delPattern.mockResolvedValue(10); - - const results = await service.invalidateSearch(); - - expect(results.length).toBeGreaterThan(0); - expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.SEARCH_INDEX_UPDATED); + describe('invalidateSearch', () => { + it('should invalidate search cache', async () => { + cachingService.delPattern.mockResolvedValue(10); + const results = await service.invalidateSearch(); + expect(results.length).toBeGreaterThan(0); + expect(eventEmitter.emit).toHaveBeenCalledWith(CACHE_EVENTS.SEARCH_INDEX_UPDATED); + }); }); - }); - - describe('clearAll', () => { - it('should clear all cache', async () => { - cachingService.clearAll.mockResolvedValue(100); - - const result = await service.clearAll(); - - expect(result).toBe(100); - expect(cachingService.clearAll).toHaveBeenCalled(); + describe('clearAll', () => { + it('should clear all cache', async () => { + cachingService.clearAll.mockResolvedValue(100); + const result = await service.clearAll(); + expect(result).toBe(100); + expect(cachingService.clearAll).toHaveBeenCalled(); + }); }); - }); - - describe('getStats', () => { - it('should return stats', () => { - service.registerKeyWithTag('tag1', 'key1'); - service.registerKeyWithTag('tag1', 'key2'); - service.registerKeyWithTag('tag2', 'key3'); - - const stats = service.getStats(); - - expect(stats.registeredTags).toBe(2); - expect(stats.totalTrackedKeys).toBe(3); + describe('getStats', () => { + it('should return stats', () => { + service.registerKeyWithTag('tag1', 'key1'); + service.registerKeyWithTag('tag1', 'key2'); + service.registerKeyWithTag('tag2', 'key3'); + const stats = service.getStats(); + expect(stats.registeredTags).toBe(2); + expect(stats.totalTrackedKeys).toBe(3); + }); }); - }); }); diff --git a/src/caching/invalidation/invalidation.service.ts b/src/caching/invalidation/invalidation.service.ts index 9343ccae..7fa99002 100644 --- a/src/caching/invalidation/invalidation.service.ts +++ b/src/caching/invalidation/invalidation.service.ts @@ -3,243 +3,210 @@ import { EventEmitter2, OnEvent } from '@nestjs/event-emitter'; import { CachingService } from '../caching.service'; import { CacheStrategiesService } from '../strategies/cache-strategies.service'; import { CACHE_EVENTS } from '../caching.constants'; - export interface InvalidationResult { - pattern: string; - keysDeleted: number; + pattern: string; + keysDeleted: number; } - @Injectable() export class CacheInvalidationService implements OnModuleInit, OnModuleDestroy { - private readonly logger = new Logger(CacheInvalidationService.name); - private readonly tagIndex: Map> = new Map(); - - constructor( - private readonly cachingService: CachingService, - private readonly strategiesService: CacheStrategiesService, - private readonly eventEmitter: EventEmitter2, - ) {} - - onModuleInit() { - this.logger.log('Cache invalidation service initialized'); - } - - onModuleDestroy() { - this.tagIndex.clear(); - } - - /** - * Invalidate cache by pattern - * @param pattern - Pattern to match (e.g., 'cache:course:*') - */ - async invalidateByPattern(pattern: string): Promise { - this.logger.debug(`Invalidating cache pattern: ${pattern}`); - const keysDeleted = await this.cachingService.delPattern(pattern); - - return { pattern, keysDeleted }; - } - - /** - * Invalidate cache by multiple patterns - * @param patterns - Array of patterns to invalidate - */ - async invalidateByPatterns(patterns: string[]): Promise { - const results: InvalidationResult[] = []; - - for (const pattern of patterns) { - const result = await this.invalidateByPattern(pattern); - results.push(result); - } - - return results; - } - - /** - * Invalidate cache by tag - * @param tag - Tag to invalidate - */ - async invalidateByTag(tag: string): Promise { - const keys = this.tagIndex.get(tag); - if (!keys || keys.size === 0) { - return 0; - } - - let deleted = 0; - for (const key of keys) { - await this.cachingService.del(key); - deleted++; - } - - this.tagIndex.delete(tag); - this.logger.debug(`Invalidated ${deleted} keys for tag: ${tag}`); - return deleted; - } - - /** - * Register a key with a tag for tag-based invalidation - * @param tag - Tag to associate with the key - * @param key - Cache key - */ - registerKeyWithTag(tag: string, key: string): void { - if (!this.tagIndex.has(tag)) { - this.tagIndex.set(tag, new Set()); - } - const tagSet = this.tagIndex.get(tag); - if (tagSet) { - tagSet.add(key); - } - } - - /** - * Unregister a key from a tag - * @param tag - Tag to dissociate from the key - * @param key - Cache key - */ - unregisterKeyFromTag(tag: string, key: string): void { - const keys = this.tagIndex.get(tag); - if (keys) { - keys.delete(key); - if (keys.size === 0) { + private readonly logger = new Logger(CacheInvalidationService.name); + private readonly tagIndex: Map> = new Map(); + constructor(private readonly cachingService: CachingService, private readonly strategiesService: CacheStrategiesService, private readonly eventEmitter: EventEmitter2) { } + onModuleInit() { + this.logger.log('Cache invalidation service initialized'); + } + onModuleDestroy() { + this.tagIndex.clear(); + } + /** + * Invalidate cache by pattern + * @param pattern - Pattern to match (e.g., 'cache:course:*') + */ + async invalidateByPattern(pattern: string): Promise { + this.logger.debug(`Invalidating cache pattern: ${pattern}`); + const keysDeleted = await this.cachingService.delPattern(pattern); + return { pattern, keysDeleted }; + } + /** + * Invalidate cache by multiple patterns + * @param patterns - Array of patterns to invalidate + */ + async invalidateByPatterns(patterns: string[]): Promise { + const results: InvalidationResult[] = []; + for (const pattern of patterns) { + const result = await this.invalidateByPattern(pattern); + results.push(result); + } + return results; + } + /** + * Invalidate cache by tag + * @param tag - Tag to invalidate + */ + async invalidateByTag(tag: string): Promise { + const keys = this.tagIndex.get(tag); + if (!keys || keys.size === 0) { + return 0; + } + let deleted = 0; + for (const key of keys) { + await this.cachingService.del(key); + deleted++; + } this.tagIndex.delete(tag); - } - } - } - - /** - * Invalidate all cache entries related to a course - */ - async invalidateCourse(courseId: string): Promise { - const patterns = [ - `cache:course:${courseId}:*`, - `cache:course:${courseId}`, - 'cache:courses:list:*', - 'cache:search:*', - 'cache:popular:*', - ]; - - this.eventEmitter.emit(CACHE_EVENTS.COURSE_UPDATED, { courseId }); - return this.invalidateByPatterns(patterns); - } - - /** - * Invalidate all cache entries related to a user - */ - async invalidateUser(userId: string): Promise { - const patterns = [ - `cache:user:${userId}:*`, - `cache:user:${userId}`, - `cache:user:profile:${userId}`, - ]; - - this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId }); - return this.invalidateByPatterns(patterns); - } - - /** - * Invalidate all cache entries related to an enrollment - */ - async invalidateEnrollment( - enrollmentId: string, - courseId?: string, - ): Promise { - const patterns = [`cache:enrollment:${enrollmentId}:*`, `cache:enrollment:${enrollmentId}`]; - - if (courseId) { - patterns.push(`cache:course:${courseId}:enrollments:*`); - } - - this.eventEmitter.emit(CACHE_EVENTS.ENROLLMENT_UPDATED, { enrollmentId, courseId }); - return this.invalidateByPatterns(patterns); - } - - /** - * Invalidate search cache - */ - async invalidateSearch(): Promise { - const patterns = ['cache:search:*']; - this.eventEmitter.emit(CACHE_EVENTS.SEARCH_INDEX_UPDATED); - return this.invalidateByPatterns(patterns); - } - - /** - * Clear all application cache - */ - async clearAll(): Promise { - return this.cachingService.clearAll(); - } - - // Event handlers for automatic invalidation - - @OnEvent(CACHE_EVENTS.COURSE_UPDATED) - async handleCourseUpdatedEvent(payload: { courseId: string }): Promise { - this.logger.debug(`Handling course updated event for: ${payload.courseId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.COURSE_UPDATED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.COURSE_DELETED) - async handleCourseDeletedEvent(payload: { courseId: string }): Promise { - this.logger.debug(`Handling course deleted event for: ${payload.courseId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.COURSE_DELETED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.USER_UPDATED) - async handleUserUpdatedEvent(payload: { userId: string }): Promise { - this.logger.debug(`Handling user updated event for: ${payload.userId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.USER_UPDATED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.USER_DELETED) - async handleUserDeletedEvent(payload: { userId: string }): Promise { - this.logger.debug(`Handling user deleted event for: ${payload.userId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.USER_DELETED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.ENROLLMENT_CREATED) - async handleEnrollmentCreatedEvent(payload: { - enrollmentId: string; - courseId: string; - }): Promise { - this.logger.debug(`Handling enrollment created event for: ${payload.enrollmentId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.ENROLLMENT_CREATED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.ENROLLMENT_UPDATED) - async handleEnrollmentUpdatedEvent(payload: { - enrollmentId: string; - courseId?: string; - }): Promise { - this.logger.debug(`Handling enrollment updated event for: ${payload.enrollmentId}`); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.ENROLLMENT_UPDATED); - await this.invalidateByPatterns(patterns); - } - - @OnEvent(CACHE_EVENTS.SEARCH_INDEX_UPDATED) - async handleSearchIndexUpdatedEvent(): Promise { - this.logger.debug('Handling search index updated event'); - const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.SEARCH_INDEX_UPDATED); - await this.invalidateByPatterns(patterns); - } - - /** - * Get invalidation statistics - */ - getStats(): { - registeredTags: number; - totalTrackedKeys: number; - } { - let totalKeys = 0; - for (const keys of this.tagIndex.values()) { - totalKeys += keys.size; - } - - return { - registeredTags: this.tagIndex.size, - totalTrackedKeys: totalKeys, - }; - } + this.logger.debug(`Invalidated ${deleted} keys for tag: ${tag}`); + return deleted; + } + /** + * Register a key with a tag for tag-based invalidation + * @param tag - Tag to associate with the key + * @param key - Cache key + */ + registerKeyWithTag(tag: string, key: string): void { + if (!this.tagIndex.has(tag)) { + this.tagIndex.set(tag, new Set()); + } + const tagSet = this.tagIndex.get(tag); + if (tagSet) { + tagSet.add(key); + } + } + /** + * Unregister a key from a tag + * @param tag - Tag to dissociate from the key + * @param key - Cache key + */ + unregisterKeyFromTag(tag: string, key: string): void { + const keys = this.tagIndex.get(tag); + if (keys) { + keys.delete(key); + if (keys.size === 0) { + this.tagIndex.delete(tag); + } + } + } + /** + * Invalidate all cache entries related to a course + */ + async invalidateCourse(courseId: string): Promise { + const patterns = [ + `cache:course:${courseId}:*`, + `cache:course:${courseId}`, + 'cache:courses:list:*', + 'cache:search:*', + 'cache:popular:*', + ]; + this.eventEmitter.emit(CACHE_EVENTS.COURSE_UPDATED, { courseId }); + return this.invalidateByPatterns(patterns); + } + /** + * Invalidate all cache entries related to a user + */ + async invalidateUser(userId: string): Promise { + const patterns = [ + `cache:user:${userId}:*`, + `cache:user:${userId}`, + `cache:user:profile:${userId}`, + ]; + this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId }); + return this.invalidateByPatterns(patterns); + } + /** + * Invalidate all cache entries related to an enrollment + */ + async invalidateEnrollment(enrollmentId: string, courseId?: string): Promise { + const patterns = [`cache:enrollment:${enrollmentId}:*`, `cache:enrollment:${enrollmentId}`]; + if (courseId) { + patterns.push(`cache:course:${courseId}:enrollments:*`); + } + this.eventEmitter.emit(CACHE_EVENTS.ENROLLMENT_UPDATED, { enrollmentId, courseId }); + return this.invalidateByPatterns(patterns); + } + /** + * Invalidate search cache + */ + async invalidateSearch(): Promise { + const patterns = ['cache:search:*']; + this.eventEmitter.emit(CACHE_EVENTS.SEARCH_INDEX_UPDATED); + return this.invalidateByPatterns(patterns); + } + /** + * Clear all application cache + */ + async clearAll(): Promise { + return this.cachingService.clearAll(); + } + // Event handlers for automatic invalidation + @OnEvent(CACHE_EVENTS.COURSE_UPDATED) + async handleCourseUpdatedEvent(payload: { + courseId: string; + }): Promise { + this.logger.debug(`Handling course updated event for: ${payload.courseId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.COURSE_UPDATED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.COURSE_DELETED) + async handleCourseDeletedEvent(payload: { + courseId: string; + }): Promise { + this.logger.debug(`Handling course deleted event for: ${payload.courseId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.COURSE_DELETED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.USER_UPDATED) + async handleUserUpdatedEvent(payload: { + userId: string; + }): Promise { + this.logger.debug(`Handling user updated event for: ${payload.userId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.USER_UPDATED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.USER_DELETED) + async handleUserDeletedEvent(payload: { + userId: string; + }): Promise { + this.logger.debug(`Handling user deleted event for: ${payload.userId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.USER_DELETED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.ENROLLMENT_CREATED) + async handleEnrollmentCreatedEvent(payload: { + enrollmentId: string; + courseId: string; + }): Promise { + this.logger.debug(`Handling enrollment created event for: ${payload.enrollmentId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.ENROLLMENT_CREATED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.ENROLLMENT_UPDATED) + async handleEnrollmentUpdatedEvent(payload: { + enrollmentId: string; + courseId?: string; + }): Promise { + this.logger.debug(`Handling enrollment updated event for: ${payload.enrollmentId}`); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.ENROLLMENT_UPDATED); + await this.invalidateByPatterns(patterns); + } + @OnEvent(CACHE_EVENTS.SEARCH_INDEX_UPDATED) + async handleSearchIndexUpdatedEvent(): Promise { + this.logger.debug('Handling search index updated event'); + const patterns = this.strategiesService.getPatternsForEvent(CACHE_EVENTS.SEARCH_INDEX_UPDATED); + await this.invalidateByPatterns(patterns); + } + /** + * Get invalidation statistics + */ + getStats(): { + registeredTags: number; + totalTrackedKeys: number; + } { + let totalKeys = 0; + for (const keys of this.tagIndex.values()) { + totalKeys += keys.size; + } + return { + registeredTags: this.tagIndex.size, + totalTrackedKeys: totalKeys, + }; + } } diff --git a/src/caching/strategies/cache-strategies.service.spec.ts b/src/caching/strategies/cache-strategies.service.spec.ts index 105c5ebb..9b8fd9c8 100644 --- a/src/caching/strategies/cache-strategies.service.spec.ts +++ b/src/caching/strategies/cache-strategies.service.spec.ts @@ -1,152 +1,117 @@ import { Test, TestingModule } from '@nestjs/testing'; import { CacheStrategiesService } from './cache-strategies.service'; import { CACHE_EVENTS } from '../caching.constants'; - describe('CacheStrategiesService', () => { - let service: CacheStrategiesService; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [CacheStrategiesService], - }).compile(); - - service = module.get(CacheStrategiesService); - }); - - describe('getStrategy', () => { - it('should return course details strategy', () => { - const strategy = service.getStrategy('course:details'); - - expect(strategy).toBeDefined(); - expect(strategy?.ttl).toBe(300); - expect(strategy?.prefix).toBe('cache:course'); + let service: CacheStrategiesService; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + providers: [CacheStrategiesService], + }).compile(); + service = module.get(CacheStrategiesService); + }); + describe('getStrategy', () => { + it('should return course details strategy', () => { + const strategy = service.getStrategy('course:details'); + expect(strategy).toBeDefined(); + expect(strategy?.ttl).toBe(300); + expect(strategy?.prefix).toBe('cache:course'); + }); + it('should return user profile strategy', () => { + const strategy = service.getStrategy('user:profile'); + expect(strategy).toBeDefined(); + expect(strategy?.ttl).toBe(600); + expect(strategy?.prefix).toBe('cache:user:profile'); + }); + it('should return undefined for unknown strategy', () => { + const strategy = service.getStrategy('unknown:strategy'); + expect(strategy).toBeUndefined(); + }); + }); + describe('getTtl', () => { + it('should return TTL for known strategy', () => { + const ttl = service.getTtl('course:details'); + expect(ttl).toBe(300); + }); + it('should return default TTL for unknown strategy', () => { + const ttl = service.getTtl('unknown'); + expect(ttl).toBe(300); + }); + }); + describe('getPrefix', () => { + it('should return prefix for known strategy', () => { + const prefix = service.getPrefix('course:details'); + expect(prefix).toBe('cache:course'); + }); + it('should return default prefix for unknown strategy', () => { + const prefix = service.getPrefix('unknown'); + expect(prefix).toBe('cache'); + }); + }); + describe('getInvalidationEvents', () => { + it('should return events for course strategy', () => { + const events = service.getInvalidationEvents('course:details'); + expect(events).toContain(CACHE_EVENTS.COURSE_UPDATED); + expect(events).toContain(CACHE_EVENTS.COURSE_DELETED); + }); + }); + describe('getRelatedPatterns', () => { + it('should return patterns for strategy', () => { + const patterns = service.getRelatedPatterns('course:details'); + expect(patterns).toContain('cache:course:*'); + expect(patterns).toContain('cache:courses:list:*'); + }); + }); + describe('getStrategiesForEvent', () => { + it('should return strategies for COURSE_UPDATED event', () => { + const strategies = service.getStrategiesForEvent(CACHE_EVENTS.COURSE_UPDATED); + expect(strategies.length).toBeGreaterThan(0); + expect(strategies.some((s) => s.name === 'course:details')).toBe(true); + }); + }); + describe('getPatternsForEvent', () => { + it('should return patterns for COURSE_UPDATED event', () => { + const patterns = service.getPatternsForEvent(CACHE_EVENTS.COURSE_UPDATED); + expect(patterns.length).toBeGreaterThan(0); + expect(patterns.some((p) => p.includes('course'))).toBe(true); + }); + }); + describe('getAllStrategies', () => { + it('should return all registered strategies', () => { + const strategies = service.getAllStrategies(); + expect(strategies.length).toBeGreaterThan(0); + expect(strategies.some((s) => s.name === 'course:details')).toBe(true); + expect(strategies.some((s) => s.name === 'user:profile')).toBe(true); + }); + }); + describe('buildKey', () => { + it('should build key with parts', () => { + const key = service.buildKey('course:details', '123'); + expect(key).toBe('cache:course:123'); + }); + it('should build key with multiple parts', () => { + const key = service.buildKey('course:details', '123', 'modules'); + expect(key).toBe('cache:course:123:modules'); + }); + }); + describe('getDefaultStrategy', () => { + it('should return default strategy', () => { + const strategy = service.getDefaultStrategy(); + expect(strategy.name).toBe('default'); + expect(strategy.ttl).toBe(300); + }); + }); + describe('registerStrategy', () => { + it('should register new strategy', () => { + service.registerStrategy({ + name: 'custom:strategy', + ttl: 1000, + prefix: 'cache:custom', + invalidateOnEvents: [], + relatedPatterns: ['cache:custom:*'], + }); + const strategy = service.getStrategy('custom:strategy'); + expect(strategy).toBeDefined(); + expect(strategy?.ttl).toBe(1000); + }); }); - - it('should return user profile strategy', () => { - const strategy = service.getStrategy('user:profile'); - - expect(strategy).toBeDefined(); - expect(strategy?.ttl).toBe(600); - expect(strategy?.prefix).toBe('cache:user:profile'); - }); - - it('should return undefined for unknown strategy', () => { - const strategy = service.getStrategy('unknown:strategy'); - - expect(strategy).toBeUndefined(); - }); - }); - - describe('getTtl', () => { - it('should return TTL for known strategy', () => { - const ttl = service.getTtl('course:details'); - - expect(ttl).toBe(300); - }); - - it('should return default TTL for unknown strategy', () => { - const ttl = service.getTtl('unknown'); - - expect(ttl).toBe(300); - }); - }); - - describe('getPrefix', () => { - it('should return prefix for known strategy', () => { - const prefix = service.getPrefix('course:details'); - - expect(prefix).toBe('cache:course'); - }); - - it('should return default prefix for unknown strategy', () => { - const prefix = service.getPrefix('unknown'); - - expect(prefix).toBe('cache'); - }); - }); - - describe('getInvalidationEvents', () => { - it('should return events for course strategy', () => { - const events = service.getInvalidationEvents('course:details'); - - expect(events).toContain(CACHE_EVENTS.COURSE_UPDATED); - expect(events).toContain(CACHE_EVENTS.COURSE_DELETED); - }); - }); - - describe('getRelatedPatterns', () => { - it('should return patterns for strategy', () => { - const patterns = service.getRelatedPatterns('course:details'); - - expect(patterns).toContain('cache:course:*'); - expect(patterns).toContain('cache:courses:list:*'); - }); - }); - - describe('getStrategiesForEvent', () => { - it('should return strategies for COURSE_UPDATED event', () => { - const strategies = service.getStrategiesForEvent(CACHE_EVENTS.COURSE_UPDATED); - - expect(strategies.length).toBeGreaterThan(0); - expect(strategies.some((s) => s.name === 'course:details')).toBe(true); - }); - }); - - describe('getPatternsForEvent', () => { - it('should return patterns for COURSE_UPDATED event', () => { - const patterns = service.getPatternsForEvent(CACHE_EVENTS.COURSE_UPDATED); - - expect(patterns.length).toBeGreaterThan(0); - expect(patterns.some((p) => p.includes('course'))).toBe(true); - }); - }); - - describe('getAllStrategies', () => { - it('should return all registered strategies', () => { - const strategies = service.getAllStrategies(); - - expect(strategies.length).toBeGreaterThan(0); - expect(strategies.some((s) => s.name === 'course:details')).toBe(true); - expect(strategies.some((s) => s.name === 'user:profile')).toBe(true); - }); - }); - - describe('buildKey', () => { - it('should build key with parts', () => { - const key = service.buildKey('course:details', '123'); - - expect(key).toBe('cache:course:123'); - }); - - it('should build key with multiple parts', () => { - const key = service.buildKey('course:details', '123', 'modules'); - - expect(key).toBe('cache:course:123:modules'); - }); - }); - - describe('getDefaultStrategy', () => { - it('should return default strategy', () => { - const strategy = service.getDefaultStrategy(); - - expect(strategy.name).toBe('default'); - expect(strategy.ttl).toBe(300); - }); - }); - - describe('registerStrategy', () => { - it('should register new strategy', () => { - service.registerStrategy({ - name: 'custom:strategy', - ttl: 1000, - prefix: 'cache:custom', - invalidateOnEvents: [], - relatedPatterns: ['cache:custom:*'], - }); - - const strategy = service.getStrategy('custom:strategy'); - expect(strategy).toBeDefined(); - expect(strategy?.ttl).toBe(1000); - }); - }); }); diff --git a/src/caching/strategies/cache-strategies.service.ts b/src/caching/strategies/cache-strategies.service.ts index 6fcc2bc3..a3988511 100644 --- a/src/caching/strategies/cache-strategies.service.ts +++ b/src/caching/strategies/cache-strategies.service.ts @@ -1,218 +1,190 @@ import { Injectable } from '@nestjs/common'; import { CACHE_TTL, CACHE_PREFIXES, CACHE_EVENTS } from '../caching.constants'; - export interface CacheStrategy { - ttl: number; - prefix: string; - invalidateOn: string[]; + ttl: number; + prefix: string; + invalidateOn: string[]; } - export interface CacheStrategyConfig { - name: string; - ttl: number; - prefix: string; - invalidateOnEvents: string[]; - relatedPatterns: string[]; + name: string; + ttl: number; + prefix: string; + invalidateOnEvents: string[]; + relatedPatterns: string[]; } - @Injectable() export class CacheStrategiesService { - private readonly strategies: Map = new Map(); - - constructor() { - this.initializeStrategies(); - } - - private initializeStrategies(): void { - // Course strategies - this.registerStrategy({ - name: 'course:details', - ttl: CACHE_TTL.COURSE_DETAILS, - prefix: CACHE_PREFIXES.COURSE, - invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED, CACHE_EVENTS.COURSE_DELETED], - relatedPatterns: ['cache:course:*', 'cache:courses:list:*'], - }); - - this.registerStrategy({ - name: 'course:metadata', - ttl: CACHE_TTL.COURSE_METADATA, - prefix: CACHE_PREFIXES.COURSE, - invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED], - relatedPatterns: ['cache:course:*'], - }); - - this.registerStrategy({ - name: 'courses:list', - ttl: CACHE_TTL.COURSE_METADATA, - prefix: CACHE_PREFIXES.COURSES_LIST, - invalidateOnEvents: [ - CACHE_EVENTS.COURSE_UPDATED, - CACHE_EVENTS.COURSE_DELETED, - CACHE_EVENTS.ENROLLMENT_CREATED, - ], - relatedPatterns: ['cache:courses:list:*'], - }); - - // User strategies - this.registerStrategy({ - name: 'user:profile', - ttl: CACHE_TTL.USER_PROFILE, - prefix: CACHE_PREFIXES.USER_PROFILE, - invalidateOnEvents: [CACHE_EVENTS.USER_UPDATED, CACHE_EVENTS.USER_DELETED], - relatedPatterns: ['cache:user:*', 'cache:user:profile:*'], - }); - - this.registerStrategy({ - name: 'user:session', - ttl: CACHE_TTL.USER_SESSION, - prefix: CACHE_PREFIXES.USER, - invalidateOnEvents: [CACHE_EVENTS.USER_DELETED], - relatedPatterns: ['cache:user:*'], - }); - - // Search strategies - this.registerStrategy({ - name: 'search:results', - ttl: CACHE_TTL.SEARCH_RESULTS, - prefix: CACHE_PREFIXES.SEARCH, - invalidateOnEvents: [ - CACHE_EVENTS.COURSE_UPDATED, - CACHE_EVENTS.COURSE_DELETED, - CACHE_EVENTS.SEARCH_INDEX_UPDATED, - ], - relatedPatterns: ['cache:search:*'], - }); - - // Popular content - this.registerStrategy({ - name: 'popular:courses', - ttl: CACHE_TTL.POPULAR_COURSES, - prefix: CACHE_PREFIXES.POPULAR, - invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED, CACHE_EVENTS.ENROLLMENT_CREATED], - relatedPatterns: ['cache:popular:*'], - }); - - // Enrollment strategies - this.registerStrategy({ - name: 'enrollment:data', - ttl: CACHE_TTL.ENROLLMENT_DATA, - prefix: CACHE_PREFIXES.ENROLLMENT, - invalidateOnEvents: [CACHE_EVENTS.ENROLLMENT_CREATED, CACHE_EVENTS.ENROLLMENT_UPDATED], - relatedPatterns: ['cache:enrollment:*'], - }); - - // Featured content - this.registerStrategy({ - name: 'featured:content', - ttl: CACHE_TTL.STATIC_CONTENT, - prefix: CACHE_PREFIXES.FEATURED, - invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED], - relatedPatterns: ['cache:featured:*'], - }); - } - - /** - * Register a new cache strategy - */ - registerStrategy(config: CacheStrategyConfig): void { - this.strategies.set(config.name, config); - } - - /** - * Get a strategy by name - */ - getStrategy(name: string): CacheStrategyConfig | undefined { - return this.strategies.get(name); - } - - /** - * Get TTL for a strategy - */ - getTtl(strategyName: string): number { - const strategy = this.strategies.get(strategyName); - return strategy?.ttl ?? CACHE_TTL.COURSE_DETAILS; - } - - /** - * Get prefix for a strategy - */ - getPrefix(strategyName: string): string { - const strategy = this.strategies.get(strategyName); - return strategy?.prefix ?? 'cache'; - } - - /** - * Get invalidation events for a strategy - */ - getInvalidationEvents(strategyName: string): string[] { - const strategy = this.strategies.get(strategyName); - return strategy?.invalidateOnEvents ?? []; - } - - /** - * Get related patterns to invalidate when a strategy is invalidated - */ - getRelatedPatterns(strategyName: string): string[] { - const strategy = this.strategies.get(strategyName); - return strategy?.relatedPatterns ?? []; - } - - /** - * Get all strategies that should be invalidated for a given event - */ - getStrategiesForEvent(eventName: string): CacheStrategyConfig[] { - const strategies: CacheStrategyConfig[] = []; - - for (const strategy of this.strategies.values()) { - if (strategy.invalidateOnEvents.includes(eventName)) { - strategies.push(strategy); - } + private readonly strategies: Map = new Map(); + constructor() { + this.initializeStrategies(); } - - return strategies; - } - - /** - * Get all patterns to invalidate for a given event - */ - getPatternsForEvent(eventName: string): string[] { - const strategies = this.getStrategiesForEvent(eventName); - const patterns = new Set(); - - for (const strategy of strategies) { - for (const pattern of strategy.relatedPatterns) { - patterns.add(pattern); - } + private initializeStrategies(): void { + // Course strategies + this.registerStrategy({ + name: 'course:details', + ttl: CACHE_TTL.COURSE_DETAILS, + prefix: CACHE_PREFIXES.COURSE, + invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED, CACHE_EVENTS.COURSE_DELETED], + relatedPatterns: ['cache:course:*', 'cache:courses:list:*'], + }); + this.registerStrategy({ + name: 'course:metadata', + ttl: CACHE_TTL.COURSE_METADATA, + prefix: CACHE_PREFIXES.COURSE, + invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED], + relatedPatterns: ['cache:course:*'], + }); + this.registerStrategy({ + name: 'courses:list', + ttl: CACHE_TTL.COURSE_METADATA, + prefix: CACHE_PREFIXES.COURSES_LIST, + invalidateOnEvents: [ + CACHE_EVENTS.COURSE_UPDATED, + CACHE_EVENTS.COURSE_DELETED, + CACHE_EVENTS.ENROLLMENT_CREATED, + ], + relatedPatterns: ['cache:courses:list:*'], + }); + // User strategies + this.registerStrategy({ + name: 'user:profile', + ttl: CACHE_TTL.USER_PROFILE, + prefix: CACHE_PREFIXES.USER_PROFILE, + invalidateOnEvents: [CACHE_EVENTS.USER_UPDATED, CACHE_EVENTS.USER_DELETED], + relatedPatterns: ['cache:user:*', 'cache:user:profile:*'], + }); + this.registerStrategy({ + name: 'user:session', + ttl: CACHE_TTL.USER_SESSION, + prefix: CACHE_PREFIXES.USER, + invalidateOnEvents: [CACHE_EVENTS.USER_DELETED], + relatedPatterns: ['cache:user:*'], + }); + // Search strategies + this.registerStrategy({ + name: 'search:results', + ttl: CACHE_TTL.SEARCH_RESULTS, + prefix: CACHE_PREFIXES.SEARCH, + invalidateOnEvents: [ + CACHE_EVENTS.COURSE_UPDATED, + CACHE_EVENTS.COURSE_DELETED, + CACHE_EVENTS.SEARCH_INDEX_UPDATED, + ], + relatedPatterns: ['cache:search:*'], + }); + // Popular content + this.registerStrategy({ + name: 'popular:courses', + ttl: CACHE_TTL.POPULAR_COURSES, + prefix: CACHE_PREFIXES.POPULAR, + invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED, CACHE_EVENTS.ENROLLMENT_CREATED], + relatedPatterns: ['cache:popular:*'], + }); + // Enrollment strategies + this.registerStrategy({ + name: 'enrollment:data', + ttl: CACHE_TTL.ENROLLMENT_DATA, + prefix: CACHE_PREFIXES.ENROLLMENT, + invalidateOnEvents: [CACHE_EVENTS.ENROLLMENT_CREATED, CACHE_EVENTS.ENROLLMENT_UPDATED], + relatedPatterns: ['cache:enrollment:*'], + }); + // Featured content + this.registerStrategy({ + name: 'featured:content', + ttl: CACHE_TTL.STATIC_CONTENT, + prefix: CACHE_PREFIXES.FEATURED, + invalidateOnEvents: [CACHE_EVENTS.COURSE_UPDATED], + relatedPatterns: ['cache:featured:*'], + }); + } + /** + * Register a new cache strategy + */ + registerStrategy(config: CacheStrategyConfig): void { + this.strategies.set(config.name, config); + } + /** + * Get a strategy by name + */ + getStrategy(name: string): CacheStrategyConfig | undefined { + return this.strategies.get(name); + } + /** + * Get TTL for a strategy + */ + getTtl(strategyName: string): number { + const strategy = this.strategies.get(strategyName); + return strategy?.ttl ?? CACHE_TTL.COURSE_DETAILS; + } + /** + * Get prefix for a strategy + */ + getPrefix(strategyName: string): string { + const strategy = this.strategies.get(strategyName); + return strategy?.prefix ?? 'cache'; + } + /** + * Get invalidation events for a strategy + */ + getInvalidationEvents(strategyName: string): string[] { + const strategy = this.strategies.get(strategyName); + return strategy?.invalidateOnEvents ?? []; + } + /** + * Get related patterns to invalidate when a strategy is invalidated + */ + getRelatedPatterns(strategyName: string): string[] { + const strategy = this.strategies.get(strategyName); + return strategy?.relatedPatterns ?? []; + } + /** + * Get all strategies that should be invalidated for a given event + */ + getStrategiesForEvent(eventName: string): CacheStrategyConfig[] { + const strategies: CacheStrategyConfig[] = []; + for (const strategy of this.strategies.values()) { + if (strategy.invalidateOnEvents.includes(eventName)) { + strategies.push(strategy); + } + } + return strategies; + } + /** + * Get all patterns to invalidate for a given event + */ + getPatternsForEvent(eventName: string): string[] { + const strategies = this.getStrategiesForEvent(eventName); + const patterns = new Set(); + for (const strategy of strategies) { + for (const pattern of strategy.relatedPatterns) { + patterns.add(pattern); + } + } + return Array.from(patterns); + } + /** + * Get all registered strategies + */ + getAllStrategies(): CacheStrategyConfig[] { + return Array.from(this.strategies.values()); + } + /** + * Build a cache key for a strategy + */ + buildKey(strategyName: string, ...parts: Array): string { + const prefix = this.getPrefix(strategyName); + return `${prefix}:${parts.join(':')}`; + } + /** + * Get default strategy configuration + */ + getDefaultStrategy(): CacheStrategyConfig { + return { + name: 'default', + ttl: CACHE_TTL.COURSE_DETAILS, + prefix: 'cache', + invalidateOnEvents: [], + relatedPatterns: ['cache:*'], + }; } - - return Array.from(patterns); - } - - /** - * Get all registered strategies - */ - getAllStrategies(): CacheStrategyConfig[] { - return Array.from(this.strategies.values()); - } - - /** - * Build a cache key for a strategy - */ - buildKey(strategyName: string, ...parts: Array): string { - const prefix = this.getPrefix(strategyName); - return `${prefix}:${parts.join(':')}`; - } - - /** - * Get default strategy configuration - */ - getDefaultStrategy(): CacheStrategyConfig { - return { - name: 'default', - ttl: CACHE_TTL.COURSE_DETAILS, - prefix: 'cache', - invalidateOnEvents: [], - relatedPatterns: ['cache:*'], - }; - } } diff --git a/src/caching/warming/cache-warming.service.ts b/src/caching/warming/cache-warming.service.ts index fd1a781e..e363f313 100644 --- a/src/caching/warming/cache-warming.service.ts +++ b/src/caching/warming/cache-warming.service.ts @@ -1,312 +1,258 @@ -import { - Injectable, - Logger, - OnModuleInit, - OnModuleDestroy, - Inject, - Optional, -} from '@nestjs/common'; +import { Injectable, Logger, OnModuleInit, OnModuleDestroy, Inject, Optional, } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import { CachingService } from '../caching.service'; import { CacheStrategiesService } from '../strategies/cache-strategies.service'; import { CACHE_TTL, CACHE_PREFIXES } from '../caching.constants'; - export interface CacheWarmingConfig { - /** - * Whether cache warming is enabled - */ - enabled: boolean; - - /** - * Warm popular courses on startup - */ - warmPopularCourses: boolean; - - /** - * Warm featured content on startup - */ - warmFeaturedContent: boolean; - - /** - * Number of popular courses to warm - */ - popularCoursesLimit: number; - - /** - * Warm system configuration - */ - warmSystemConfig: boolean; - - /** - * Delay before starting cache warming (ms) - */ - startupDelay: number; + /** + * Whether cache warming is enabled + */ + enabled: boolean; + /** + * Warm popular courses on startup + */ + warmPopularCourses: boolean; + /** + * Warm featured content on startup + */ + warmFeaturedContent: boolean; + /** + * Number of popular courses to warm + */ + popularCoursesLimit: number; + /** + * Warm system configuration + */ + warmSystemConfig: boolean; + /** + * Delay before starting cache warming (ms) + */ + startupDelay: number; } - export interface WarmedData { - key: string; - type: string; - timestamp: Date; + key: string; + type: string; + timestamp: Date; } - /** * Interface for data providers that can be warmed */ export interface CacheWarmableProvider { - /** - * Get data to warm into cache - */ - getWarmableData(): Promise>; + /** + * Get data to warm into cache + */ + getWarmableData(): Promise>; } - @Injectable() export class CacheWarmingService implements OnModuleInit, OnModuleDestroy { - private readonly logger = new Logger(CacheWarmingService.name); - private readonly config: CacheWarmingConfig; - private warmedKeys: Map = new Map(); - private warmupInterval?: NodeJS.Timeout; - private dataProviders: CacheWarmableProvider[] = []; - - constructor( - private readonly cachingService: CachingService, - private readonly strategiesService: CacheStrategiesService, - private readonly configService: ConfigService, - @Optional() @Inject('CACHE_WARMING_CONFIG') config?: Partial, - ) { - this.config = { - enabled: this.configService.get('CACHE_WARMING_ENABLED') !== 'false', - warmPopularCourses: true, - warmFeaturedContent: true, - popularCoursesLimit: parseInt( - this.configService.get('CACHE_WARMING_POPULAR_LIMIT') || '10', - 10, - ), - warmSystemConfig: true, - startupDelay: parseInt(this.configService.get('CACHE_WARMING_DELAY') || '5000', 10), - ...config, - }; - } - - /** - * Register a data provider for cache warming - */ - registerDataProvider(provider: CacheWarmableProvider): void { - this.dataProviders.push(provider); - } - - async onModuleInit(): Promise { - if (!this.config.enabled) { - this.logger.log('Cache warming is disabled'); - return; + private readonly logger = new Logger(CacheWarmingService.name); + private readonly config: CacheWarmingConfig; + private warmedKeys: Map = new Map(); + private warmupInterval?: NodeJS.Timeout; + private dataProviders: CacheWarmableProvider[] = []; + constructor(private readonly cachingService: CachingService, private readonly strategiesService: CacheStrategiesService, private readonly configService: ConfigService, + @Optional() + @Inject('CACHE_WARMING_CONFIG') + config?: Partial) { + this.config = { + enabled: this.configService.get('CACHE_WARMING_ENABLED') !== 'false', + warmPopularCourses: true, + warmFeaturedContent: true, + popularCoursesLimit: parseInt(this.configService.get('CACHE_WARMING_POPULAR_LIMIT') || '10', 10), + warmSystemConfig: true, + startupDelay: parseInt(this.configService.get('CACHE_WARMING_DELAY') || '5000', 10), + ...config, + }; } - - // Delay warming to allow the application to fully start - setTimeout(() => { - this.warmCache().catch((error) => { - this.logger.error('Failed to warm cache on startup', error); - }); - }, this.config.startupDelay); - - // Schedule periodic refresh of warmed data - this.schedulePeriodicWarmup(); - } - - onModuleDestroy(): void { - if (this.warmupInterval) { - clearInterval(this.warmupInterval); + /** + * Register a data provider for cache warming + */ + registerDataProvider(provider: CacheWarmableProvider): void { + this.dataProviders.push(provider); } - this.warmedKeys.clear(); - } - - /** - * Warm cache with critical data - */ - async warmCache(): Promise { - this.logger.log('Starting cache warming...'); - const startTime = Date.now(); - - try { - // Warm data from registered providers - await this.warmFromProviders(); - - // Warm built-in data types - if (this.config.warmPopularCourses) { - await this.warmPopularCourses(); - } - - if (this.config.warmFeaturedContent) { - await this.warmFeaturedContent(); - } - - if (this.config.warmSystemConfig) { - await this.warmSystemConfig(); - } - - const duration = Date.now() - startTime; - this.logger.log( - `Cache warming completed in ${duration}ms. Warmed ${this.warmedKeys.size} keys.`, - ); - } catch (error) { - this.logger.error('Cache warming failed', error); - throw error; + async onModuleInit(): Promise { + if (!this.config.enabled) { + this.logger.log('Cache warming is disabled'); + return; + } + // Delay warming to allow the application to fully start + setTimeout(() => { + this.warmCache().catch((error) => { + this.logger.error('Failed to warm cache on startup', error); + }); + }, this.config.startupDelay); + // Schedule periodic refresh of warmed data + this.schedulePeriodicWarmup(); + } + onModuleDestroy(): void { + if (this.warmupInterval) { + clearInterval(this.warmupInterval); + } + this.warmedKeys.clear(); + } + /** + * Warm cache with critical data + */ + async warmCache(): Promise { + this.logger.log('Starting cache warming...'); + const startTime = Date.now(); + try { + // Warm data from registered providers + await this.warmFromProviders(); + // Warm built-in data types + if (this.config.warmPopularCourses) { + await this.warmPopularCourses(); + } + if (this.config.warmFeaturedContent) { + await this.warmFeaturedContent(); + } + if (this.config.warmSystemConfig) { + await this.warmSystemConfig(); + } + const duration = Date.now() - startTime; + this.logger.log(`Cache warming completed in ${duration}ms. Warmed ${this.warmedKeys.size} keys.`); + } + catch (error) { + this.logger.error('Cache warming failed', error); + throw error; + } } - } - - /** - * Warm data from registered providers - */ - private async warmFromProviders(): Promise { - for (const provider of this.dataProviders) { - try { - const items = await provider.getWarmableData(); - for (const item of items) { - await this.cachingService.set(item.key, item.data, item.ttl); - this.trackWarmedKey(item.key, 'provider'); + /** + * Warm data from registered providers + */ + private async warmFromProviders(): Promise { + for (const provider of this.dataProviders) { + try { + const items = await provider.getWarmableData(); + for (const item of items) { + await this.cachingService.set(item.key, item.data, item.ttl); + this.trackWarmedKey(item.key, 'provider'); + } + } + catch (error) { + this.logger.warn('Failed to warm data from provider', error); + } } - } catch (error) { - this.logger.warn('Failed to warm data from provider', error); - } } - } - - /** - * Warm popular courses - * This is a placeholder - in production, this would fetch from CourseService - */ - private async warmPopularCourses(): Promise { - this.logger.debug('Warming popular courses...'); - - // Placeholder: In a real implementation, this would inject CourseService - // and fetch actual popular courses - const popularCoursesKey = `${CACHE_PREFIXES.POPULAR}:courses`; - - // Simulate warming with placeholder data - const placeholderData = { - courses: [], - warmedAt: new Date().toISOString(), - type: 'popular', - }; - - await this.cachingService.set(popularCoursesKey, placeholderData, CACHE_TTL.POPULAR_COURSES); - - this.trackWarmedKey(popularCoursesKey, 'popular_courses'); - this.logger.debug(`Warmed popular courses key: ${popularCoursesKey}`); - } - - /** - * Warm featured content - */ - private async warmFeaturedContent(): Promise { - this.logger.debug('Warming featured content...'); - - const featuredKey = `${CACHE_PREFIXES.FEATURED}:content`; - - const placeholderData = { - content: [], - warmedAt: new Date().toISOString(), - type: 'featured', - }; - - await this.cachingService.set(featuredKey, placeholderData, CACHE_TTL.STATIC_CONTENT); - - this.trackWarmedKey(featuredKey, 'featured_content'); - this.logger.debug(`Warmed featured content key: ${featuredKey}`); - } - - /** - * Warm system configuration - */ - private async warmSystemConfig(): Promise { - this.logger.debug('Warming system configuration...'); - - const configKey = CACHE_PREFIXES.SYSTEM_CONFIG; - - const configData = { - version: this.configService.get('npm_package_version') || '1.0.0', - environment: this.configService.get('NODE_ENV') || 'development', - warmedAt: new Date().toISOString(), - }; - - await this.cachingService.set(configKey, configData, CACHE_TTL.STATIC_CONTENT); - - this.trackWarmedKey(configKey, 'system_config'); - this.logger.debug(`Warmed system config key: ${configKey}`); - } - - /** - * Schedule periodic cache warming - */ - private schedulePeriodicWarmup(): void { - // Refresh warmed data every 30 minutes - const interval = 30 * 60 * 1000; - - this.warmupInterval = setInterval(async () => { - this.logger.debug('Running scheduled cache warming...'); - try { + /** + * Warm popular courses + * This is a placeholder - in production, this would fetch from CourseService + */ + private async warmPopularCourses(): Promise { + this.logger.debug('Warming popular courses...'); + // Placeholder: In a real implementation, this would inject CourseService + // and fetch actual popular courses + const popularCoursesKey = `${CACHE_PREFIXES.POPULAR}:courses`; + // Simulate warming with placeholder data + const placeholderData = { + courses: [], + warmedAt: new Date().toISOString(), + type: 'popular', + }; + await this.cachingService.set(popularCoursesKey, placeholderData, CACHE_TTL.POPULAR_COURSES); + this.trackWarmedKey(popularCoursesKey, 'popular_courses'); + this.logger.debug(`Warmed popular courses key: ${popularCoursesKey}`); + } + /** + * Warm featured content + */ + private async warmFeaturedContent(): Promise { + this.logger.debug('Warming featured content...'); + const featuredKey = `${CACHE_PREFIXES.FEATURED}:content`; + const placeholderData = { + content: [], + warmedAt: new Date().toISOString(), + type: 'featured', + }; + await this.cachingService.set(featuredKey, placeholderData, CACHE_TTL.STATIC_CONTENT); + this.trackWarmedKey(featuredKey, 'featured_content'); + this.logger.debug(`Warmed featured content key: ${featuredKey}`); + } + /** + * Warm system configuration + */ + private async warmSystemConfig(): Promise { + this.logger.debug('Warming system configuration...'); + const configKey = CACHE_PREFIXES.SYSTEM_CONFIG; + const configData = { + version: this.configService.get('npm_package_version') || '1.0.0', + environment: this.configService.get('NODE_ENV') || 'development', + warmedAt: new Date().toISOString(), + }; + await this.cachingService.set(configKey, configData, CACHE_TTL.STATIC_CONTENT); + this.trackWarmedKey(configKey, 'system_config'); + this.logger.debug(`Warmed system config key: ${configKey}`); + } + /** + * Schedule periodic cache warming + */ + private schedulePeriodicWarmup(): void { + // Refresh warmed data every 30 minutes + const interval = 30 * 60 * 1000; + this.warmupInterval = setInterval(async () => { + this.logger.debug('Running scheduled cache warming...'); + try { + await this.warmCache(); + } + catch (error) { + this.logger.error('Scheduled cache warming failed', error); + } + }, interval); + } + /** + * Track a warmed key + */ + private trackWarmedKey(key: string, type: string): void { + this.warmedKeys.set(key, { + key, + type, + timestamp: new Date(), + }); + } + /** + * Get statistics about warmed keys + */ + getStats(): { + totalKeys: number; + byType: Record; + lastWarmup: Date | null; + } { + const byType: Record = {}; + let lastWarmup: Date | null = null; + for (const warmed of this.warmedKeys.values()) { + byType[warmed.type] = (byType[warmed.type] || 0) + 1; + if (!lastWarmup || warmed.timestamp > lastWarmup) { + lastWarmup = warmed.timestamp; + } + } + return { + totalKeys: this.warmedKeys.size, + byType, + lastWarmup, + }; + } + /** + * Force refresh all warmed data + */ + async refreshAll(): Promise { + this.logger.log('Force refreshing all warmed data...'); + this.warmedKeys.clear(); await this.warmCache(); - } catch (error) { - this.logger.error('Scheduled cache warming failed', error); - } - }, interval); - } - - /** - * Track a warmed key - */ - private trackWarmedKey(key: string, type: string): void { - this.warmedKeys.set(key, { - key, - type, - timestamp: new Date(), - }); - } - - /** - * Get statistics about warmed keys - */ - getStats(): { - totalKeys: number; - byType: Record; - lastWarmup: Date | null; - } { - const byType: Record = {}; - let lastWarmup: Date | null = null; - - for (const warmed of this.warmedKeys.values()) { - byType[warmed.type] = (byType[warmed.type] || 0) + 1; - - if (!lastWarmup || warmed.timestamp > lastWarmup) { - lastWarmup = warmed.timestamp; - } } - - return { - totalKeys: this.warmedKeys.size, - byType, - lastWarmup, - }; - } - - /** - * Force refresh all warmed data - */ - async refreshAll(): Promise { - this.logger.log('Force refreshing all warmed data...'); - this.warmedKeys.clear(); - await this.warmCache(); - } - - /** - * Check if a key was warmed - */ - isWarmed(key: string): boolean { - return this.warmedKeys.has(key); - } - - /** - * Get all warmed keys - */ - getWarmedKeys(): WarmedData[] { - return Array.from(this.warmedKeys.values()); - } + /** + * Check if a key was warmed + */ + isWarmed(key: string): boolean { + return this.warmedKeys.has(key); + } + /** + * Get all warmed keys + */ + getWarmedKeys(): WarmedData[] { + return Array.from(this.warmedKeys.values()); + } } diff --git a/src/cdn/caching/edge-caching.service.ts b/src/cdn/caching/edge-caching.service.ts index 2af35d86..f2ef22aa 100644 --- a/src/cdn/caching/edge-caching.service.ts +++ b/src/cdn/caching/edge-caching.service.ts @@ -1,169 +1,142 @@ import { Injectable, Logger } from '@nestjs/common'; import { CloudflareService } from '../providers/cloudflare.service'; - export interface CacheEntry { - url: string; - ttl: number; - lastModified: Date; - etag?: string; + url: string; + ttl: number; + lastModified: Date; + etag?: string; } - export interface PurgeResult { - success: boolean; - purgedUrls: string[]; - failedUrls: string[]; - provider: string; + success: boolean; + purgedUrls: string[]; + failedUrls: string[]; + provider: string; } - @Injectable() export class EdgeCachingService { - private readonly logger = new Logger(EdgeCachingService.name); - - constructor(private cloudflareService: CloudflareService) {} - - async getEdgeUrl(originalUrl: string, location?: string): Promise { - // In a real implementation, this would return the CDN URL for the optimal edge location - // For now, just return the original URL with CDN prefix - const cdnUrl = originalUrl.replace(/^https?:\/\/[^/]+/, 'https://cdn.example.com'); - - // Add location-based routing if needed - if (location) { - return `${cdnUrl}?location=${location}`; + private readonly logger = new Logger(EdgeCachingService.name); + constructor(private cloudflareService: CloudflareService) { } + async getEdgeUrl(originalUrl: string, location?: string): Promise { + // In a real implementation, this would return the CDN URL for the optimal edge location + // For now, just return the original URL with CDN prefix + const cdnUrl = originalUrl.replace(/^https?:\/\/[^/]+/, 'https://cdn.example.com'); + // Add location-based routing if needed + if (location) { + return `${cdnUrl}?location=${location}`; + } + return cdnUrl; } - - return cdnUrl; - } - - async purgeContent(contentId: string): Promise { - const urls = await this.getContentUrls(contentId); - return this.purgeUrls(urls); - } - - async purgeUrls(urls: string[]): Promise { - this.logger.log(`Purging ${urls.length} URLs from edge cache`); - - const results: PurgeResult[] = []; - - // Purge from Cloudflare - try { - const cfResult = await this.cloudflareService.purgeUrls(urls); - results.push({ - success: cfResult.success, - purgedUrls: cfResult.purgedUrls, - failedUrls: cfResult.failedUrls, - provider: 'cloudflare', - }); - } catch (error) { - this.logger.error('Cloudflare purge failed:', error); - results.push({ - success: false, - purgedUrls: [], - failedUrls: urls, - provider: 'cloudflare', - }); + async purgeContent(contentId: string): Promise { + const urls = await this.getContentUrls(contentId); + return this.purgeUrls(urls); } - - // Combine results - const success = results.every((r) => r.success); - const purgedUrls = results.flatMap((r) => r.purgedUrls); - const failedUrls = results.flatMap((r) => r.failedUrls); - - return { - success, - purgedUrls, - failedUrls, - provider: 'all', - }; - } - - async purgeByTags(tags: string[]): Promise { - this.logger.log(`Purging content by tags: ${tags.join(', ')}`); - - // Get URLs associated with tags - const urls = await this.getUrlsByTags(tags); - - return this.purgeUrls(urls); - } - - async purgeByPattern(pattern: string): Promise { - this.logger.log(`Purging content by pattern: ${pattern}`); - - // Get URLs matching pattern - const urls = await this.getUrlsByPattern(pattern); - - return this.purgeUrls(urls); - } - - async warmCache(urls: string[]): Promise { - this.logger.log(`Warming cache for ${urls.length} URLs`); - - // Prefetch content to edge locations - for (const url of urls) { - try { - await this.prefetchToEdge(url); - } catch (error) { - this.logger.error(`Failed to prefetch ${url}:`, error); - } + async purgeUrls(urls: string[]): Promise { + this.logger.log(`Purging ${urls.length} URLs from edge cache`); + const results: PurgeResult[] = []; + // Purge from Cloudflare + try { + const cfResult = await this.cloudflareService.purgeUrls(urls); + results.push({ + success: cfResult.success, + purgedUrls: cfResult.purgedUrls, + failedUrls: cfResult.failedUrls, + provider: 'cloudflare', + }); + } + catch (error) { + this.logger.error('Cloudflare purge failed:', error); + results.push({ + success: false, + purgedUrls: [], + failedUrls: urls, + provider: 'cloudflare', + }); + } + // Combine results + const success = results.every((r) => r.success); + const purgedUrls = results.flatMap((r) => r.purgedUrls); + const failedUrls = results.flatMap((r) => r.failedUrls); + return { + success, + purgedUrls, + failedUrls, + provider: 'all', + }; } - } - - async getCacheStatus(_url: string): Promise<{ - cached: boolean; - age?: number; - expires?: Date; - lastModified?: Date; - }> { - // Check if URL is cached in edge - // This would typically involve HEAD requests to CDN endpoints - return { - cached: false, // Mock implementation - age: undefined, - expires: undefined, - lastModified: undefined, - }; - } - - async setCacheRules(rules: CacheRule[]): Promise { - // Apply caching rules to CDN configuration - for (const rule of rules) { - await this.applyCacheRule(rule); + async purgeByTags(tags: string[]): Promise { + this.logger.log(`Purging content by tags: ${tags.join(', ')}`); + // Get URLs associated with tags + const urls = await this.getUrlsByTags(tags); + return this.purgeUrls(urls); + } + async purgeByPattern(pattern: string): Promise { + this.logger.log(`Purging content by pattern: ${pattern}`); + // Get URLs matching pattern + const urls = await this.getUrlsByPattern(pattern); + return this.purgeUrls(urls); + } + async warmCache(urls: string[]): Promise { + this.logger.log(`Warming cache for ${urls.length} URLs`); + // Prefetch content to edge locations + for (const url of urls) { + try { + await this.prefetchToEdge(url); + } + catch (error) { + this.logger.error(`Failed to prefetch ${url}:`, error); + } + } + } + async getCacheStatus(_url: string): Promise<{ + cached: boolean; + age?: number; + expires?: Date; + lastModified?: Date; + }> { + // Check if URL is cached in edge + // This would typically involve HEAD requests to CDN endpoints + return { + cached: false, // Mock implementation + age: undefined, + expires: undefined, + lastModified: undefined, + }; + } + async setCacheRules(rules: CacheRule[]): Promise { + // Apply caching rules to CDN configuration + for (const rule of rules) { + await this.applyCacheRule(rule); + } + } + private async getContentUrls(contentId: string): Promise { + // Implementation would fetch all URLs associated with content ID + // Including original, optimized versions, etc. + return [ + `https://cdn.example.com/${contentId}`, + `https://cdn.example.com/${contentId}_optimized.webp`, + `https://cdn.example.com/${contentId}_w640.webp`, + ]; + } + private async getUrlsByTags(_tags: string[]): Promise { + // Implementation would query database for URLs with specific tags + return []; + } + private async getUrlsByPattern(_pattern: string): Promise { + // Implementation would find URLs matching pattern + return []; + } + private async prefetchToEdge(_url: string): Promise { + // Implementation would make requests to warm the cache + // This might involve calling CDN APIs or making HTTP requests + } + private async applyCacheRule(rule: CacheRule): Promise { + // Implementation would update CDN configuration + this.logger.log(`Applying cache rule: ${rule.pattern} -> TTL: ${rule.ttl}`); } - } - - private async getContentUrls(contentId: string): Promise { - // Implementation would fetch all URLs associated with content ID - // Including original, optimized versions, etc. - return [ - `https://cdn.example.com/${contentId}`, - `https://cdn.example.com/${contentId}_optimized.webp`, - `https://cdn.example.com/${contentId}_w640.webp`, - ]; - } - - private async getUrlsByTags(_tags: string[]): Promise { - // Implementation would query database for URLs with specific tags - return []; - } - - private async getUrlsByPattern(_pattern: string): Promise { - // Implementation would find URLs matching pattern - return []; - } - - private async prefetchToEdge(_url: string): Promise { - // Implementation would make requests to warm the cache - // This might involve calling CDN APIs or making HTTP requests - } - - private async applyCacheRule(rule: CacheRule): Promise { - // Implementation would update CDN configuration - this.logger.log(`Applying cache rule: ${rule.pattern} -> TTL: ${rule.ttl}`); - } } - export interface CacheRule { - pattern: string; - ttl: number; - headers?: Record; - queryString?: boolean; + pattern: string; + ttl: number; + headers?: Record; + queryString?: boolean; } diff --git a/src/cdn/cdn.controller.ts b/src/cdn/cdn.controller.ts index 9a496135..86ebe906 100644 --- a/src/cdn/cdn.controller.ts +++ b/src/cdn/cdn.controller.ts @@ -1,416 +1,373 @@ -import { - Controller, - Post, - Get, - Delete, - Param, - Query, - Body, - UploadedFile, - UseInterceptors, - HttpException, - HttpStatus, - Logger, - BadRequestException, -} from '@nestjs/common'; +import { Controller, Post, Get, Delete, Param, Query, Body, UploadedFile, UseInterceptors, HttpException, HttpStatus, Logger, BadRequestException, } from '@nestjs/common'; import { FileInterceptor } from '@nestjs/platform-express'; -import { - ApiTags, - ApiOperation, - ApiResponse, - ApiConsumes, - ApiBody, - ApiParam, - ApiQuery, -} from '@nestjs/swagger'; +import { ApiTags, ApiOperation, ApiResponse, ApiConsumes, ApiBody, ApiParam, ApiQuery, } from '@nestjs/swagger'; import { UploadedFile as FileUpload } from '../common/types/file.types'; import { CdnService } from './cdn.service'; import { UploadContentDto } from './dto/upload-content.dto'; import { ContentMetadata } from './entities/content-metadata.entity'; -import { - FileValidationService, - FileValidationResult, -} from '../media/validation/file-validation.service'; +import { FileValidationService, FileValidationResult, } from '../media/validation/file-validation.service'; import { MalwareScanningService } from '../media/validation/malware-scanning.service'; import { ImageProcessingService } from '../media/processing/image-processing.service'; -import { - ALLOWED_FILE_TYPES, - FILE_SIZE_LIMITS, -} from '../media/validation/file-validation.constants'; - +import { ALLOWED_FILE_TYPES, FILE_SIZE_LIMITS, } from '../media/validation/file-validation.constants'; @ApiTags('CDN') @Controller('cdn') export class CdnController { - private readonly logger = new Logger(CdnController.name); - - constructor( - private readonly cdnService: CdnService, - private readonly fileValidation: FileValidationService, - private readonly malwareScanning: MalwareScanningService, - private readonly imageProcessing: ImageProcessingService, - ) {} - - @Post('upload') - @UseInterceptors(FileInterceptor('file')) - @ApiOperation({ summary: 'Upload content to CDN with full validation' }) - @ApiConsumes('multipart/form-data') - @ApiBody({ - description: 'Content upload with validation and optimization options', - type: UploadContentDto, - }) - @ApiResponse({ - status: 201, - description: 'Content uploaded successfully', - type: ContentMetadata, - }) - @ApiResponse({ status: 400, description: 'Validation failed or bad request' }) - @ApiResponse({ status: 403, description: 'Malware detected' }) - @ApiResponse({ status: 413, description: 'File too large' }) - @ApiResponse({ status: 415, description: 'Unsupported media type' }) - @ApiResponse({ status: 500, description: 'Internal server error' }) - async uploadContent( - @UploadedFile() file: FileUpload, - @Body() options: UploadContentDto, - ): Promise { - try { - if (!file) { - throw new HttpException('No file provided', HttpStatus.BAD_REQUEST); - } - - this.logger.log(`Uploading file: ${file.originalname} (${file.size} bytes)`); - - // Step 1: Validate file - const validationResult = await this.fileValidation.validateFile(file); - if (!validationResult.valid) { - this.logger.warn( - `File validation failed for ${file.originalname}:`, - validationResult.errors, - ); - throw new BadRequestException({ - message: 'File validation failed', - errors: validationResult.errors, - warnings: validationResult.warnings, - allowedTypes: Object.values(ALLOWED_FILE_TYPES).flat(), - sizeLimits: FILE_SIZE_LIMITS, - }); - } - - // Step 2: Malware scan - if (this.malwareScanning.isScanningAvailable()) { - this.logger.log(`Scanning file for malware: ${file.originalname}`); - const scanResult = await this.malwareScanning.scanFile(file); - - if (!scanResult.clean) { - const errorMsg = - scanResult.threats.length > 0 - ? `Malware detected: ${scanResult.threats.join(', ')}` - : 'File failed security scan'; - this.logger.error(`Malware detected in ${file.originalname}:`, scanResult.threats); - throw new HttpException(errorMsg, HttpStatus.FORBIDDEN); + private readonly logger = new Logger(CdnController.name); + constructor(private readonly cdnService: CdnService, private readonly fileValidation: FileValidationService, private readonly malwareScanning: MalwareScanningService, private readonly imageProcessing: ImageProcessingService) { } + @Post('upload') + @UseInterceptors(FileInterceptor('file')) + @ApiOperation({ summary: 'Upload content to CDN with full validation' }) + @ApiConsumes('multipart/form-data') + @ApiBody({ + description: 'Content upload with validation and optimization options', + type: UploadContentDto, + }) + @ApiResponse({ + status: 201, + description: 'Content uploaded successfully', + type: ContentMetadata, + }) + @ApiResponse({ status: 400, description: 'Validation failed or bad request' }) + @ApiResponse({ status: 403, description: 'Malware detected' }) + @ApiResponse({ status: 413, description: 'File too large' }) + @ApiResponse({ status: 415, description: 'Unsupported media type' }) + @ApiResponse({ status: 500, description: 'Internal server error' }) + async uploadContent( + @UploadedFile() + file: FileUpload, + @Body() + options: UploadContentDto): Promise { + try { + if (!file) { + throw new HttpException('No file provided', HttpStatus.BAD_REQUEST); + } + this.logger.log(`Uploading file: ${file.originalname} (${file.size} bytes)`); + // Step 1: Validate file + const validationResult = await this.fileValidation.validateFile(file); + if (!validationResult.valid) { + this.logger.warn(`File validation failed for ${file.originalname}:`, validationResult.errors); + throw new BadRequestException({ + message: 'File validation failed', + errors: validationResult.errors, + warnings: validationResult.warnings, + allowedTypes: Object.values(ALLOWED_FILE_TYPES).flat(), + sizeLimits: FILE_SIZE_LIMITS, + }); + } + // Step 2: Malware scan + if (this.malwareScanning.isScanningAvailable()) { + this.logger.log(`Scanning file for malware: ${file.originalname}`); + const scanResult = await this.malwareScanning.scanFile(file); + if (!scanResult.clean) { + const errorMsg = scanResult.threats.length > 0 + ? `Malware detected: ${scanResult.threats.join(', ')}` + : 'File failed security scan'; + this.logger.error(`Malware detected in ${file.originalname}:`, scanResult.threats); + throw new HttpException(errorMsg, HttpStatus.FORBIDDEN); + } + } + // Step 3: Process and upload + const result = await this.cdnService.uploadContent(file, options); + this.logger.log(`Successfully uploaded content: ${result.contentId}`); + return result; + } + catch (error) { + this.logger.error('Upload failed:', error); + if (error instanceof HttpException) { + throw error; + } + throw new HttpException(`Upload failed: ${error instanceof Error ? error.message : 'Unknown error'}`, HttpStatus.INTERNAL_SERVER_ERROR); } - } - - // Step 3: Process and upload - const result = await this.cdnService.uploadContent(file, options); - - this.logger.log(`Successfully uploaded content: ${result.contentId}`); - return result; - } catch (error) { - this.logger.error('Upload failed:', error); - - if (error instanceof HttpException) { - throw error; - } - - throw new HttpException( - `Upload failed: ${error instanceof Error ? error.message : 'Unknown error'}`, - HttpStatus.INTERNAL_SERVER_ERROR, - ); } - } - - @Post('validate') - @UseInterceptors(FileInterceptor('file')) - @ApiOperation({ summary: 'Validate file without uploading' }) - @ApiConsumes('multipart/form-data') - @ApiResponse({ - status: 200, - description: 'File validation result', - schema: { - type: 'object', - properties: { - valid: { type: 'boolean' }, - mimeType: { type: 'string' }, - fileType: { type: 'string' }, - size: { type: 'number' }, - maxSize: { type: 'number' }, - errors: { type: 'array', items: { type: 'string' } }, - warnings: { type: 'array', items: { type: 'string' } }, - metadata: { - type: 'object', - properties: { - width: { type: 'number' }, - height: { type: 'number' }, - format: { type: 'string' }, - hasAlpha: { type: 'boolean' }, - }, + @Post('validate') + @UseInterceptors(FileInterceptor('file')) + @ApiOperation({ summary: 'Validate file without uploading' }) + @ApiConsumes('multipart/form-data') + @ApiResponse({ + status: 200, + description: 'File validation result', + schema: { + type: 'object', + properties: { + valid: { type: 'boolean' }, + mimeType: { type: 'string' }, + fileType: { type: 'string' }, + size: { type: 'number' }, + maxSize: { type: 'number' }, + errors: { type: 'array', items: { type: 'string' } }, + warnings: { type: 'array', items: { type: 'string' } }, + metadata: { + type: 'object', + properties: { + width: { type: 'number' }, + height: { type: 'number' }, + format: { type: 'string' }, + hasAlpha: { type: 'boolean' }, + }, + }, + }, }, - }, - }, - }) - @ApiResponse({ status: 400, description: 'No file provided' }) - async validateFile(@UploadedFile() file: FileUpload): Promise { - if (!file) { - throw new BadRequestException('No file provided'); - } - - this.logger.log(`Validating file: ${file.originalname}`); - return this.fileValidation.validateFile(file); - } - - @Post('scan') - @UseInterceptors(FileInterceptor('file')) - @ApiOperation({ summary: 'Scan file for malware without uploading' }) - @ApiConsumes('multipart/form-data') - @ApiResponse({ - status: 200, - description: 'Malware scan result', - schema: { - type: 'object', - properties: { - clean: { type: 'boolean' }, - threats: { type: 'array', items: { type: 'string' } }, - scanTime: { type: 'number' }, - scannerVersion: { type: 'string' }, - error: { type: 'string' }, - }, - }, - }) - @ApiResponse({ status: 400, description: 'No file provided' }) - @ApiResponse({ status: 503, description: 'Scanning service not available' }) - async scanFile(@UploadedFile() file: FileUpload) { - if (!file) { - throw new BadRequestException('No file provided'); + }) + @ApiResponse({ status: 400, description: 'No file provided' }) + async validateFile( + @UploadedFile() + file: FileUpload): Promise { + if (!file) { + throw new BadRequestException('No file provided'); + } + this.logger.log(`Validating file: ${file.originalname}`); + return this.fileValidation.validateFile(file); } - - if (!this.malwareScanning.isScanningAvailable()) { - throw new HttpException( - 'Malware scanning service not available', - HttpStatus.SERVICE_UNAVAILABLE, - ); + @Post('scan') + @UseInterceptors(FileInterceptor('file')) + @ApiOperation({ summary: 'Scan file for malware without uploading' }) + @ApiConsumes('multipart/form-data') + @ApiResponse({ + status: 200, + description: 'Malware scan result', + schema: { + type: 'object', + properties: { + clean: { type: 'boolean' }, + threats: { type: 'array', items: { type: 'string' } }, + scanTime: { type: 'number' }, + scannerVersion: { type: 'string' }, + error: { type: 'string' }, + }, + }, + }) + @ApiResponse({ status: 400, description: 'No file provided' }) + @ApiResponse({ status: 503, description: 'Scanning service not available' }) + async scanFile( + @UploadedFile() + file: FileUpload) { + if (!file) { + throw new BadRequestException('No file provided'); + } + if (!this.malwareScanning.isScanningAvailable()) { + throw new HttpException('Malware scanning service not available', HttpStatus.SERVICE_UNAVAILABLE); + } + this.logger.log(`Scanning file: ${file.originalname}`); + return this.malwareScanning.scanFile(file); } - - this.logger.log(`Scanning file: ${file.originalname}`); - return this.malwareScanning.scanFile(file); - } - - @Get('allowed-types') - @ApiOperation({ summary: 'Get allowed file types and size limits' }) - @ApiResponse({ - status: 200, - description: 'Allowed file types and limits', - schema: { - type: 'object', - properties: { - allowedTypes: { type: 'object' }, - sizeLimits: { type: 'object' }, - dimensionLimits: { type: 'object' }, - }, - }, - }) - getAllowedTypes() { - return { - allowedTypes: ALLOWED_FILE_TYPES, - sizeLimits: { - image: this.formatBytes(FILE_SIZE_LIMITS.IMAGE_MAX_SIZE), - video: this.formatBytes(FILE_SIZE_LIMITS.VIDEO_MAX_SIZE), - document: this.formatBytes(FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE), - audio: this.formatBytes(FILE_SIZE_LIMITS.AUDIO_MAX_SIZE), - archive: this.formatBytes(FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE), - default: this.formatBytes(FILE_SIZE_LIMITS.DEFAULT_MAX_SIZE), - }, - dimensionLimits: { - minWidth: 1, - minHeight: 1, - maxWidth: 16384, - maxHeight: 16384, - maxPixels: 100_000_000, - }, - }; - } - - @Post('compress-preview') - @UseInterceptors(FileInterceptor('file')) - @ApiOperation({ summary: 'Preview image compression without saving' }) - @ApiConsumes('multipart/form-data') - @ApiResponse({ - status: 200, - description: 'Compression preview result', - schema: { - type: 'object', - properties: { - originalSize: { type: 'number' }, - compressedSize: { type: 'number' }, - compressionRatio: { type: 'number' }, - width: { type: 'number' }, - height: { type: 'number' }, - format: { type: 'string' }, - }, - }, - }) - @ApiResponse({ status: 400, description: 'Invalid file or not an image' }) - async compressPreview(@UploadedFile() file: FileUpload) { - if (!file) { - throw new BadRequestException('No file provided'); + @Get('allowed-types') + @ApiOperation({ summary: 'Get allowed file types and size limits' }) + @ApiResponse({ + status: 200, + description: 'Allowed file types and limits', + schema: { + type: 'object', + properties: { + allowedTypes: { type: 'object' }, + sizeLimits: { type: 'object' }, + dimensionLimits: { type: 'object' }, + }, + }, + }) + getAllowedTypes() { + return { + allowedTypes: ALLOWED_FILE_TYPES, + sizeLimits: { + image: this.formatBytes(FILE_SIZE_LIMITS.IMAGE_MAX_SIZE), + video: this.formatBytes(FILE_SIZE_LIMITS.VIDEO_MAX_SIZE), + document: this.formatBytes(FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE), + audio: this.formatBytes(FILE_SIZE_LIMITS.AUDIO_MAX_SIZE), + archive: this.formatBytes(FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE), + default: this.formatBytes(FILE_SIZE_LIMITS.DEFAULT_MAX_SIZE), + }, + dimensionLimits: { + minWidth: 1, + minHeight: 1, + maxWidth: 16384, + maxHeight: 16384, + maxPixels: 100000000, + }, + }; } - - if (!file.mimetype.startsWith('image/')) { - throw new BadRequestException('File is not an image'); + @Post('compress-preview') + @UseInterceptors(FileInterceptor('file')) + @ApiOperation({ summary: 'Preview image compression without saving' }) + @ApiConsumes('multipart/form-data') + @ApiResponse({ + status: 200, + description: 'Compression preview result', + schema: { + type: 'object', + properties: { + originalSize: { type: 'number' }, + compressedSize: { type: 'number' }, + compressionRatio: { type: 'number' }, + width: { type: 'number' }, + height: { type: 'number' }, + format: { type: 'string' }, + }, + }, + }) + @ApiResponse({ status: 400, description: 'Invalid file or not an image' }) + async compressPreview( + @UploadedFile() + file: FileUpload) { + if (!file) { + throw new BadRequestException('No file provided'); + } + if (!file.mimetype.startsWith('image/')) { + throw new BadRequestException('File is not an image'); + } + try { + const result = await this.imageProcessing.compressImage(file.buffer); + return { + originalSize: result.originalSize, + compressedSize: result.size, + compressionRatio: result.compressionRatio, + width: result.width, + height: result.height, + format: result.format, + }; + } + catch (error) { + this.logger.error('Compression preview failed:', error); + throw new BadRequestException('Failed to compress image'); + } } - - try { - const result = await this.imageProcessing.compressImage(file.buffer); - return { - originalSize: result.originalSize, - compressedSize: result.size, - compressionRatio: result.compressionRatio, - width: result.width, - height: result.height, - format: result.format, - }; - } catch (error) { - this.logger.error('Compression preview failed:', error); - throw new BadRequestException('Failed to compress image'); + @Get('content/:contentId') + @ApiOperation({ summary: 'Get optimized content URL' }) + @ApiParam({ name: 'contentId', description: 'Content identifier' }) + @ApiQuery({ name: 'optimize', required: false, type: Boolean }) + @ApiQuery({ name: 'width', required: false, type: Number }) + @ApiQuery({ name: 'height', required: false, type: Number }) + @ApiQuery({ name: 'quality', required: false, type: Number }) + @ApiQuery({ name: 'format', required: false, enum: ['webp', 'jpeg', 'png'] }) + @ApiQuery({ name: 'userLocation', required: false, type: String }) + @ApiQuery({ name: 'bandwidth', required: false, type: Number }) + @ApiResponse({ status: 200, description: 'Content URL retrieved successfully' }) + @ApiResponse({ status: 404, description: 'Content not found' }) + async getContentUrl( + @Param('contentId') + contentId: string, + @Query('optimize') + optimize?: string, + @Query('width') + width?: string, + @Query('height') + height?: string, + @Query('quality') + quality?: string, + @Query('format') + format?: 'webp' | 'jpeg' | 'png', + @Query('userLocation') + userLocation?: string, + @Query('bandwidth') + bandwidth?: string): Promise<{ + url: string; + metadata?: unknown; + }> { + try { + const options = { + optimize: optimize === 'true', + width: width ? parseInt(width) : undefined, + height: height ? parseInt(height) : undefined, + quality: quality ? parseInt(quality) : undefined, + format, + userLocation, + bandwidth: bandwidth ? parseFloat(bandwidth) : undefined, + }; + const url = await this.cdnService.deliverContent(contentId, options); + return { url }; + } + catch (error) { + this.logger.error(`Failed to get content URL for ${contentId}:`, error); + if (error.message.includes('not found')) { + throw new HttpException('Content not found', HttpStatus.NOT_FOUND); + } + throw new HttpException('Failed to retrieve content', HttpStatus.INTERNAL_SERVER_ERROR); + } } - } - - @Get('content/:contentId') - @ApiOperation({ summary: 'Get optimized content URL' }) - @ApiParam({ name: 'contentId', description: 'Content identifier' }) - @ApiQuery({ name: 'optimize', required: false, type: Boolean }) - @ApiQuery({ name: 'width', required: false, type: Number }) - @ApiQuery({ name: 'height', required: false, type: Number }) - @ApiQuery({ name: 'quality', required: false, type: Number }) - @ApiQuery({ name: 'format', required: false, enum: ['webp', 'jpeg', 'png'] }) - @ApiQuery({ name: 'userLocation', required: false, type: String }) - @ApiQuery({ name: 'bandwidth', required: false, type: Number }) - @ApiResponse({ status: 200, description: 'Content URL retrieved successfully' }) - @ApiResponse({ status: 404, description: 'Content not found' }) - async getContentUrl( - @Param('contentId') contentId: string, - @Query('optimize') optimize?: string, - @Query('width') width?: string, - @Query('height') height?: string, - @Query('quality') quality?: string, - @Query('format') format?: 'webp' | 'jpeg' | 'png', - @Query('userLocation') userLocation?: string, - @Query('bandwidth') bandwidth?: string, - ): Promise<{ url: string; metadata?: any }> { - try { - const options = { - optimize: optimize === 'true', - width: width ? parseInt(width) : undefined, - height: height ? parseInt(height) : undefined, - quality: quality ? parseInt(quality) : undefined, - format, - userLocation, - bandwidth: bandwidth ? parseFloat(bandwidth) : undefined, - }; - - const url = await this.cdnService.deliverContent(contentId, options); - - return { url }; - } catch (error) { - this.logger.error(`Failed to get content URL for ${contentId}:`, error); - if (error.message.includes('not found')) { - throw new HttpException('Content not found', HttpStatus.NOT_FOUND); - } - throw new HttpException('Failed to retrieve content', HttpStatus.INTERNAL_SERVER_ERROR); + @Delete('content/:contentId') + @ApiOperation({ summary: 'Invalidate content cache' }) + @ApiParam({ name: 'contentId', description: 'Content identifier' }) + @ApiResponse({ status: 200, description: 'Content cache invalidated successfully' }) + @ApiResponse({ status: 404, description: 'Content not found' }) + async invalidateContent( + @Param('contentId') + contentId: string): Promise<{ + success: boolean; + }> { + try { + await this.cdnService.invalidateContent(contentId); + this.logger.log(`Invalidated cache for content: ${contentId}`); + return { success: true }; + } + catch (error) { + this.logger.error(`Failed to invalidate content ${contentId}:`, error); + throw new HttpException('Failed to invalidate content', HttpStatus.INTERNAL_SERVER_ERROR); + } } - } - - @Delete('content/:contentId') - @ApiOperation({ summary: 'Invalidate content cache' }) - @ApiParam({ name: 'contentId', description: 'Content identifier' }) - @ApiResponse({ status: 200, description: 'Content cache invalidated successfully' }) - @ApiResponse({ status: 404, description: 'Content not found' }) - async invalidateContent(@Param('contentId') contentId: string): Promise<{ success: boolean }> { - try { - await this.cdnService.invalidateContent(contentId); - this.logger.log(`Invalidated cache for content: ${contentId}`); - return { success: true }; - } catch (error) { - this.logger.error(`Failed to invalidate content ${contentId}:`, error); - throw new HttpException('Failed to invalidate content', HttpStatus.INTERNAL_SERVER_ERROR); + @Get('health') + @ApiOperation({ summary: 'Check CDN health status' }) + @ApiResponse({ status: 200, description: 'CDN health status' }) + async getHealth(): Promise<{ + status: string; + providers: Record; + timestamp: string; + }> { + try { + // In a real implementation, check actual provider connectivity + const providers = { + cloudflare: true, // Mock health check + aws: true, + }; + return { + status: 'healthy', + providers, + timestamp: new Date().toISOString(), + }; + } + catch (_error) { + console.error('health check failed'); + throw new HttpException('Health check failed', HttpStatus.INTERNAL_SERVER_ERROR); + } } - } - - @Get('health') - @ApiOperation({ summary: 'Check CDN health status' }) - @ApiResponse({ status: 200, description: 'CDN health status' }) - async getHealth(): Promise<{ - status: string; - providers: Record; - timestamp: string; - }> { - try { - // In a real implementation, check actual provider connectivity - const providers = { - cloudflare: true, // Mock health check - aws: true, - }; - - return { - status: 'healthy', - providers, - timestamp: new Date().toISOString(), - }; - } catch (_error) { - console.error('health check failed'); - throw new HttpException('Health check failed', HttpStatus.INTERNAL_SERVER_ERROR); + @Get('analytics') + @ApiOperation({ summary: 'Get CDN analytics' }) + @ApiQuery({ name: 'startDate', required: false }) + @ApiQuery({ name: 'endDate', required: false }) + @ApiResponse({ status: 200, description: 'CDN analytics data' }) + async getAnalytics( + @Query('startDate') + startDate?: string, + @Query('endDate') + endDate?: string): Promise { + try { + const start = startDate + ? new Date(startDate) + : new Date(Date.now() - 30 * 24 * 60 * 60 * 1000); + const end = endDate ? new Date(endDate) : new Date(); + // In a real implementation, aggregate analytics from providers + return { + totalRequests: 0, + totalBandwidth: 0, + cacheHitRate: 0, + topContent: [], + period: { + start: start.toISOString(), + end: end.toISOString(), + }, + }; + } + catch (_error) { + console.error('failed to retrieve'); + throw new HttpException('Failed to retrieve analytics', HttpStatus.INTERNAL_SERVER_ERROR); + } } - } - - @Get('analytics') - @ApiOperation({ summary: 'Get CDN analytics' }) - @ApiQuery({ name: 'startDate', required: false }) - @ApiQuery({ name: 'endDate', required: false }) - @ApiResponse({ status: 200, description: 'CDN analytics data' }) - async getAnalytics( - @Query('startDate') startDate?: string, - @Query('endDate') endDate?: string, - ): Promise { - try { - const start = startDate - ? new Date(startDate) - : new Date(Date.now() - 30 * 24 * 60 * 60 * 1000); - const end = endDate ? new Date(endDate) : new Date(); - - // In a real implementation, aggregate analytics from providers - return { - totalRequests: 0, - totalBandwidth: 0, - cacheHitRate: 0, - topContent: [], - period: { - start: start.toISOString(), - end: end.toISOString(), - }, - }; - } catch (_error) { - console.error('failed to retrieve'); - throw new HttpException('Failed to retrieve analytics', HttpStatus.INTERNAL_SERVER_ERROR); + /** + * Format bytes to human readable string + */ + private formatBytes(bytes: number): string { + if (bytes === 0) + return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`; } - } - - /** - * Format bytes to human readable string - */ - private formatBytes(bytes: number): string { - if (bytes === 0) return '0 Bytes'; - const k = 1024; - const sizes = ['Bytes', 'KB', 'MB', 'GB']; - const i = Math.floor(Math.log(bytes) / Math.log(k)); - return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`; - } } diff --git a/src/cdn/cdn.module.ts b/src/cdn/cdn.module.ts index a17f1b3c..01571e89 100644 --- a/src/cdn/cdn.module.ts +++ b/src/cdn/cdn.module.ts @@ -14,40 +14,40 @@ import { ContentMetadata } from './entities/content-metadata.entity'; import { FileValidationService } from '../media/validation/file-validation.service'; import { MalwareScanningService } from '../media/validation/malware-scanning.service'; import { ImageProcessingService } from '../media/processing/image-processing.service'; - @Module({ - imports: [ - ConfigModule, - CacheModule.register(), - TypeOrmModule.forFeature([ContentMetadata]), - MulterModule.register({ - dest: './uploads', - limits: { - fileSize: 500 * 1024 * 1024, // 500MB limit (largest for videos) - }, - }), - ], - controllers: [CdnController], - providers: [ - CdnService, - AssetOptimizationService, - EdgeCachingService, - GeoLocationService, - CloudflareService, - AWSCloudFrontService, - FileValidationService, - MalwareScanningService, - ImageProcessingService, - ], - exports: [ - CdnService, - AssetOptimizationService, - EdgeCachingService, - GeoLocationService, - CloudflareService, - AWSCloudFrontService, - FileValidationService, - ImageProcessingService, - ], + imports: [ + ConfigModule, + CacheModule.register(), + TypeOrmModule.forFeature([ContentMetadata]), + MulterModule.register({ + dest: './uploads', + limits: { + fileSize: 500 * 1024 * 1024, // 500MB limit (largest for videos) + }, + }), + ], + controllers: [CdnController], + providers: [ + CdnService, + AssetOptimizationService, + EdgeCachingService, + GeoLocationService, + CloudflareService, + AWSCloudFrontService, + FileValidationService, + MalwareScanningService, + ImageProcessingService, + ], + exports: [ + CdnService, + AssetOptimizationService, + EdgeCachingService, + GeoLocationService, + CloudflareService, + AWSCloudFrontService, + FileValidationService, + ImageProcessingService, + ], }) -export class CdnModule {} +export class CdnModule { +} diff --git a/src/cdn/cdn.service.ts b/src/cdn/cdn.service.ts index 7334ed0c..48fba994 100644 --- a/src/cdn/cdn.service.ts +++ b/src/cdn/cdn.service.ts @@ -9,241 +9,200 @@ import { GeoLocationService } from './geo/geo-location.service'; import { CloudflareService } from './providers/cloudflare.service'; import { ContentMetadata, ContentType, ContentStatus } from './entities/content-metadata.entity'; import { UploadedFile } from '../common/types/file.types'; - export interface ContentDeliveryOptions { - optimize?: boolean; - quality?: number; - format?: 'webp' | 'jpeg' | 'png'; - width?: number; - height?: number; - userLocation?: string; - bandwidth?: number; - responsive?: boolean; + optimize?: boolean; + quality?: number; + format?: 'webp' | 'jpeg' | 'png'; + width?: number; + height?: number; + userLocation?: string; + bandwidth?: number; + responsive?: boolean; } - @Injectable() export class CdnService { - private readonly logger = new Logger(CdnService.name); - - constructor( - @Inject(CACHE_MANAGER) private cacheManager: Cache, + private readonly logger = new Logger(CdnService.name); + constructor( + @Inject(CACHE_MANAGER) + private cacheManager: Cache, @InjectRepository(ContentMetadata) - private contentMetadataRepository: Repository, - private assetOptimizationService: AssetOptimizationService, - private edgeCachingService: EdgeCachingService, - private geoLocationService: GeoLocationService, - private cloudflareService: CloudflareService, - ) {} - - async deliverContent(contentId: string, options: ContentDeliveryOptions = {}): Promise { - const cacheKey = `cdn:${contentId}:${JSON.stringify(options)}`; - - // Check cache first - const cachedUrl = await this.cacheManager.get(cacheKey); - if (cachedUrl) { - return cachedUrl; + private contentMetadataRepository: Repository, private assetOptimizationService: AssetOptimizationService, private edgeCachingService: EdgeCachingService, private geoLocationService: GeoLocationService, private cloudflareService: CloudflareService) { } + async deliverContent(contentId: string, options: ContentDeliveryOptions = {}): Promise { + const cacheKey = `cdn:${contentId}:${JSON.stringify(options)}`; + // Check cache first + const cachedUrl = await this.cacheManager.get(cacheKey); + if (cachedUrl) { + return cachedUrl; + } + // Get content metadata + const metadata = await this.getContentMetadata(contentId); + if (!metadata) { + throw new Error(`Content not found: ${contentId}`); + } + // Update access statistics + await this.updateAccessStats(metadata); + // Determine optimal delivery strategy + const optimalLocation = await this.geoLocationService.getOptimalLocation(options.userLocation); + // Optimize content if needed + let deliveryUrl = metadata.cdnUrl || metadata.originalUrl; + if (options.optimize && metadata.contentType === ContentType.IMAGE) { + deliveryUrl = await this.assetOptimizationService.optimizeImage(deliveryUrl, options); + } + // Apply bandwidth optimization + if (options.bandwidth) { + deliveryUrl = await this.optimizeForBandwidth(deliveryUrl, options.bandwidth); + } + // Get edge-cached URL + const edgeUrl = await this.edgeCachingService.getEdgeUrl(deliveryUrl, optimalLocation); + // Cache the result + await this.cacheManager.set(cacheKey, edgeUrl, 3600000); // 1 hour + return edgeUrl; } - - // Get content metadata - const metadata = await this.getContentMetadata(contentId); - if (!metadata) { - throw new Error(`Content not found: ${contentId}`); + async invalidateContent(contentId: string): Promise { + // Purge from edge caches + await this.edgeCachingService.purgeContent(contentId); + // Clear local cache - simplified approach + // In a real implementation, you might need to track cache keys separately + // or use a cache store that supports key pattern deletion + this.logger.warn(`Cache invalidation for ${contentId} - manual cleanup may be required`); } - - // Update access statistics - await this.updateAccessStats(metadata); - - // Determine optimal delivery strategy - const optimalLocation = await this.geoLocationService.getOptimalLocation(options.userLocation); - - // Optimize content if needed - let deliveryUrl = metadata.cdnUrl || metadata.originalUrl; - if (options.optimize && metadata.contentType === ContentType.IMAGE) { - deliveryUrl = await this.assetOptimizationService.optimizeImage(deliveryUrl, options); + async uploadContent(file: UploadedFile, options: ContentDeliveryOptions = {}): Promise { + try { + // Upload to primary CDN provider with failover + const uploadResult = await this.uploadWithFailover(file); + // Create metadata entity + const contentId = this.generateContentId(); + const metadata = this.contentMetadataRepository.create({ + contentId, + originalUrl: uploadResult.url, + cdnUrl: uploadResult.url, + contentType: this.mapContentType(file.mimetype), + fileName: file.originalname, + mimeType: file.mimetype, + fileSize: file.size, + status: ContentStatus.READY, + etag: uploadResult.etag, + provider: uploadResult.provider, + optimizationSettings: options.optimize + ? { + width: options.width, + height: options.height, + quality: options.quality, + format: options.format, + responsive: options.responsive, + } + : undefined, + }); + // Store metadata + await this.contentMetadataRepository.save(metadata); + // Optimize asynchronously if needed + if (options.optimize && this.isImageFile(file)) { + setImmediate(async () => { + try { + await this.optimizeContentAsync(metadata, options); + } + catch (error) { + this.logger.error(`Async optimization failed for ${contentId}:`, error); + } + }); + } + return metadata; + } + catch (error) { + this.logger.error('Upload failed:', error); + throw error; + } + } + private async getContentMetadata(contentId: string): Promise { + return this.contentMetadataRepository.findOne({ + where: { contentId }, + }); } - - // Apply bandwidth optimization - if (options.bandwidth) { - deliveryUrl = await this.optimizeForBandwidth(deliveryUrl, options.bandwidth); + private async storeContentMetadata(metadata: ContentMetadata): Promise { + await this.contentMetadataRepository.save(metadata); } - - // Get edge-cached URL - const edgeUrl = await this.edgeCachingService.getEdgeUrl(deliveryUrl, optimalLocation); - - // Cache the result - await this.cacheManager.set(cacheKey, edgeUrl, 3600000); // 1 hour - - return edgeUrl; - } - - async invalidateContent(contentId: string): Promise { - // Purge from edge caches - await this.edgeCachingService.purgeContent(contentId); - - // Clear local cache - simplified approach - // In a real implementation, you might need to track cache keys separately - // or use a cache store that supports key pattern deletion - this.logger.warn(`Cache invalidation for ${contentId} - manual cleanup may be required`); - } - - async uploadContent( - file: UploadedFile, - options: ContentDeliveryOptions = {}, - ): Promise { - try { - // Upload to primary CDN provider with failover - const uploadResult = await this.uploadWithFailover(file); - - // Create metadata entity - const contentId = this.generateContentId(); - const metadata = this.contentMetadataRepository.create({ - contentId, - originalUrl: uploadResult.url, - cdnUrl: uploadResult.url, - contentType: this.mapContentType(file.mimetype), - fileName: file.originalname, - mimeType: file.mimetype, - fileSize: file.size, - status: ContentStatus.READY, - etag: uploadResult.etag, - provider: uploadResult.provider, - optimizationSettings: options.optimize - ? { - width: options.width, - height: options.height, - quality: options.quality, - format: options.format, - responsive: options.responsive, + private async updateAccessStats(metadata: ContentMetadata): Promise { + metadata.accessCount += 1; + metadata.lastAccessedAt = new Date(); + await this.contentMetadataRepository.save(metadata); + } + private async uploadWithFailover(file: UploadedFile): Promise<{ + url: string; + etag?: string; + provider: string; + }> { + // Try primary provider (Cloudflare) + try { + const result = await this.cloudflareService.uploadFile(file); + return { ...result, provider: 'cloudflare' }; + } + catch (error) { + this.logger.warn('Primary provider failed, trying fallback:', error); + // Try fallback provider (AWS CloudFront) + try { + // Note: AWS service would need to be injected + // For now, return mock fallback + throw new Error('AWS provider not implemented in this context'); } - : undefined, - }); - - // Store metadata - await this.contentMetadataRepository.save(metadata); - - // Optimize asynchronously if needed - if (options.optimize && this.isImageFile(file)) { - setImmediate(async () => { - try { - await this.optimizeContentAsync(metadata, options); - } catch (error) { - this.logger.error(`Async optimization failed for ${contentId}:`, error); - } - }); - } - - return metadata; - } catch (error) { - this.logger.error('Upload failed:', error); - throw error; + catch (fallbackError) { + this.logger.error('All providers failed:', fallbackError); + throw new Error('All CDN providers failed to upload file'); + } + } + } + private async optimizeContentAsync(metadata: ContentMetadata, options: ContentDeliveryOptions): Promise { + try { + metadata.status = ContentStatus.PROCESSING; + await this.contentMetadataRepository.save(metadata); + const _optimizedUrl = await this.assetOptimizationService.optimizeImage(metadata.cdnUrl, options); + // Generate responsive variants if requested + let variants = []; + if (options.responsive) { + variants = await this.assetOptimizationService.generateResponsiveImages(metadata.cdnUrl); + } + metadata.status = ContentStatus.OPTIMIZED; + metadata.optimizedSize = variants.reduce((total, variant) => total + variant.optimizedSize, 0); + metadata.variants = variants.map((v) => ({ + name: v.url.split('/').pop(), + url: v.url, + width: options.width || 0, + height: options.height || 0, + size: v.optimizedSize, + })); + await this.contentMetadataRepository.save(metadata); + } + catch (error) { + metadata.status = ContentStatus.FAILED; + metadata.errorMessage = error.message; + await this.contentMetadataRepository.save(metadata); + throw error; + } + } + private async optimizeForBandwidth(url: string, _bandwidth: number): Promise { + // Implementation would adjust quality/format based on bandwidth + // For now, return original URL + return url; + } + private isImageFile(file: UploadedFile): boolean { + return file.mimetype.startsWith('image/'); + } + private getContentType(file: UploadedFile): 'image' | 'video' | 'document' { + if (file.mimetype.startsWith('image/')) + return 'image'; + if (file.mimetype.startsWith('video/')) + return 'video'; + return 'document'; } - } - - private async getContentMetadata(contentId: string): Promise { - return this.contentMetadataRepository.findOne({ - where: { contentId }, - }); - } - - private async storeContentMetadata(metadata: ContentMetadata): Promise { - await this.contentMetadataRepository.save(metadata); - } - - private async updateAccessStats(metadata: ContentMetadata): Promise { - metadata.accessCount += 1; - metadata.lastAccessedAt = new Date(); - await this.contentMetadataRepository.save(metadata); - } - - private async uploadWithFailover(file: UploadedFile): Promise<{ - url: string; - etag?: string; - provider: string; - }> { - // Try primary provider (Cloudflare) - try { - const result = await this.cloudflareService.uploadFile(file); - return { ...result, provider: 'cloudflare' }; - } catch (error) { - this.logger.warn('Primary provider failed, trying fallback:', error); - - // Try fallback provider (AWS CloudFront) - try { - // Note: AWS service would need to be injected - // For now, return mock fallback - throw new Error('AWS provider not implemented in this context'); - } catch (fallbackError) { - this.logger.error('All providers failed:', fallbackError); - throw new Error('All CDN providers failed to upload file'); - } + private generateContentId(): string { + return `cdn_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; } - } - - private async optimizeContentAsync( - metadata: ContentMetadata, - options: ContentDeliveryOptions, - ): Promise { - try { - metadata.status = ContentStatus.PROCESSING; - await this.contentMetadataRepository.save(metadata); - - const _optimizedUrl = await this.assetOptimizationService.optimizeImage( - metadata.cdnUrl, - options, - ); - - // Generate responsive variants if requested - let variants = []; - if (options.responsive) { - variants = await this.assetOptimizationService.generateResponsiveImages(metadata.cdnUrl); - } - - metadata.status = ContentStatus.OPTIMIZED; - metadata.optimizedSize = variants.reduce( - (total, variant) => total + variant.optimizedSize, - 0, - ); - metadata.variants = variants.map((v) => ({ - name: v.url.split('/').pop(), - url: v.url, - width: options.width || 0, - height: options.height || 0, - size: v.optimizedSize, - })); - - await this.contentMetadataRepository.save(metadata); - } catch (error) { - metadata.status = ContentStatus.FAILED; - metadata.errorMessage = error.message; - await this.contentMetadataRepository.save(metadata); - throw error; + private mapContentType(mimeType: string): ContentType { + if (mimeType.startsWith('image/')) + return ContentType.IMAGE; + if (mimeType.startsWith('video/')) + return ContentType.VIDEO; + if (mimeType.startsWith('audio/')) + return ContentType.AUDIO; + return ContentType.DOCUMENT; } - } - - private async optimizeForBandwidth(url: string, _bandwidth: number): Promise { - // Implementation would adjust quality/format based on bandwidth - // For now, return original URL - return url; - } - - private isImageFile(file: UploadedFile): boolean { - return file.mimetype.startsWith('image/'); - } - - private getContentType(file: UploadedFile): 'image' | 'video' | 'document' { - if (file.mimetype.startsWith('image/')) return 'image'; - if (file.mimetype.startsWith('video/')) return 'video'; - return 'document'; - } - - private generateContentId(): string { - return `cdn_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; - } - - private mapContentType(mimeType: string): ContentType { - if (mimeType.startsWith('image/')) return ContentType.IMAGE; - if (mimeType.startsWith('video/')) return ContentType.VIDEO; - if (mimeType.startsWith('audio/')) return ContentType.AUDIO; - return ContentType.DOCUMENT; - } } diff --git a/src/cdn/dto/upload-content.dto.ts b/src/cdn/dto/upload-content.dto.ts index 459932f9..55ee894d 100644 --- a/src/cdn/dto/upload-content.dto.ts +++ b/src/cdn/dto/upload-content.dto.ts @@ -1,76 +1,68 @@ import { IsOptional, IsString, IsNumber, IsBoolean, Min, Max } from 'class-validator'; import { ApiPropertyOptional } from '@nestjs/swagger'; - export class UploadContentDto { - @ApiPropertyOptional({ - description: 'Whether to optimize the content automatically', - default: true, - }) - @IsOptional() - @IsBoolean() - optimize?: boolean = true; - - @ApiPropertyOptional({ - description: 'Target width for image optimization', - minimum: 1, - maximum: 4096, - }) - @IsOptional() - @IsNumber() - @Min(1) - @Max(4096) - width?: number; - - @ApiPropertyOptional({ - description: 'Target height for image optimization', - minimum: 1, - maximum: 4096, - }) - @IsOptional() - @IsNumber() - @Min(1) - @Max(4096) - height?: number; - - @ApiPropertyOptional({ - description: 'Image quality (1-100)', - minimum: 1, - maximum: 100, - }) - @IsOptional() - @IsNumber() - @Min(1) - @Max(100) - quality?: number; - - @ApiPropertyOptional({ - description: 'Output format', - enum: ['webp', 'jpeg', 'png'], - }) - @IsOptional() - @IsString() - format?: 'webp' | 'jpeg' | 'png'; - - @ApiPropertyOptional({ - description: 'User location for geo-optimization', - }) - @IsOptional() - @IsString() - userLocation?: string; - - @ApiPropertyOptional({ - description: 'User bandwidth in Mbps for optimization', - minimum: 0, - }) - @IsOptional() - @IsNumber() - @Min(0) - bandwidth?: number; - - @ApiPropertyOptional({ - description: 'Generate responsive image variants', - }) - @IsOptional() - @IsBoolean() - responsive?: boolean; + @ApiPropertyOptional({ + description: 'Whether to optimize the content automatically', + default: true, + }) + @IsOptional() + @IsBoolean() + optimize?: boolean = true; + @ApiPropertyOptional({ + description: 'Target width for image optimization', + minimum: 1, + maximum: 4096, + }) + @IsOptional() + @IsNumber() + @Min(1) + @Max(4096) + width?: number; + @ApiPropertyOptional({ + description: 'Target height for image optimization', + minimum: 1, + maximum: 4096, + }) + @IsOptional() + @IsNumber() + @Min(1) + @Max(4096) + height?: number; + @ApiPropertyOptional({ + description: 'Image quality (1-100)', + minimum: 1, + maximum: 100, + }) + @IsOptional() + @IsNumber() + @Min(1) + @Max(100) + quality?: number; + @ApiPropertyOptional({ + description: 'Output format', + enum: ['webp', 'jpeg', 'png'], + }) + @IsOptional() + @IsString() + format?: 'webp' | 'jpeg' | 'png'; + @ApiPropertyOptional({ + description: 'User location for geo-optimization', + }) + @IsOptional() + @IsString() + userLocation?: string; + @ApiPropertyOptional({ + description: 'User bandwidth in Mbps for optimization', + minimum: 0, + }) + @IsOptional() + @IsNumber() + @Min(0) + bandwidth?: number; + @ApiPropertyOptional({ + description: 'Generate responsive image variants', + }) + @IsOptional() + @IsBoolean() + responsive?: boolean; } diff --git a/src/cdn/entities/content-metadata.entity.ts b/src/cdn/entities/content-metadata.entity.ts index a0d42b00..8712a725 100644 --- a/src/cdn/entities/content-metadata.entity.ts +++ b/src/cdn/entities/content-metadata.entity.ts @@ -1,125 +1,93 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, - Index, -} from 'typeorm'; - +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, Index, } from 'typeorm'; export enum ContentType { - IMAGE = 'image', - VIDEO = 'video', - DOCUMENT = 'document', - AUDIO = 'audio', + IMAGE = 'image', + VIDEO = 'video', + DOCUMENT = 'document', + AUDIO = 'audio' } - export enum ContentStatus { - UPLOADING = 'uploading', - PROCESSING = 'processing', - OPTIMIZED = 'optimized', - READY = 'ready', - FAILED = 'failed', + UPLOADING = 'uploading', + PROCESSING = 'processing', + OPTIMIZED = 'optimized', + READY = 'ready', + FAILED = 'failed' } - @Entity('content_metadata') @Index(['contentId'], { unique: true }) @Index(['status']) @Index(['contentType']) export class ContentMetadata { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ name: 'content_id', unique: true }) - contentId: string; - - @Column({ name: 'original_url' }) - originalUrl: string; - - @Column({ name: 'cdn_url', nullable: true }) - cdnUrl: string; - - @Column({ - name: 'content_type', - type: 'enum', - enum: ContentType, - }) - contentType: ContentType; - - @Column({ name: 'file_name' }) - fileName: string; - - @Column({ name: 'mime_type' }) - mimeType: string; - - @Column({ name: 'file_size' }) - fileSize: number; - - @Column({ name: 'optimized_size', nullable: true }) - optimizedSize: number; - - @Column({ - name: 'status', - type: 'enum', - enum: ContentStatus, - default: ContentStatus.UPLOADING, - }) - status: ContentStatus; - - @Column({ name: 'etag', nullable: true }) - etag: string; - - @Column({ name: 'provider', default: 'cloudflare' }) - provider: string; - - @Column({ name: 'optimization_settings', type: 'json', nullable: true }) - optimizationSettings: { - width?: number; - height?: number; - quality?: number; - format?: string; - responsive?: boolean; - }; - - @Column({ name: 'variants', type: 'json', nullable: true }) - variants: Array<{ - name: string; - url: string; - width: number; - height: number; - size: number; - }>; - - @Column({ name: 'metadata', type: 'json', nullable: true }) - metadata: { - width?: number; - height?: number; - duration?: number; // for video/audio - bitrate?: number; - codec?: string; - }; - - @Column({ name: 'owner_id', nullable: true }) - ownerId?: string; - - @Column({ name: 'tenant_id', nullable: true }) - tenantId?: string; - - @Column({ name: 'error_message', nullable: true }) - errorMessage: string; - - @Column({ name: 'retry_count', default: 0 }) - retryCount: number; - - @Column({ name: 'last_accessed_at', nullable: true }) - lastAccessedAt: Date; - - @Column({ name: 'access_count', default: 0 }) - accessCount: number; - - @CreateDateColumn({ name: 'created_at' }) - createdAt: Date; - - @UpdateDateColumn({ name: 'updated_at' }) - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ name: 'content_id', unique: true }) + contentId: string; + @Column({ name: 'original_url' }) + originalUrl: string; + @Column({ name: 'cdn_url', nullable: true }) + cdnUrl: string; + @Column({ + name: 'content_type', + type: 'enum', + enum: ContentType, + }) + contentType: ContentType; + @Column({ name: 'file_name' }) + fileName: string; + @Column({ name: 'mime_type' }) + mimeType: string; + @Column({ name: 'file_size' }) + fileSize: number; + @Column({ name: 'optimized_size', nullable: true }) + optimizedSize: number; + @Column({ + name: 'status', + type: 'enum', + enum: ContentStatus, + default: ContentStatus.UPLOADING, + }) + status: ContentStatus; + @Column({ name: 'etag', nullable: true }) + etag: string; + @Column({ name: 'provider', default: 'cloudflare' }) + provider: string; + @Column({ name: 'optimization_settings', type: 'json', nullable: true }) + optimizationSettings: { + width?: number; + height?: number; + quality?: number; + format?: string; + responsive?: boolean; + }; + @Column({ name: 'variants', type: 'json', nullable: true }) + variants: Array<{ + name: string; + url: string; + width: number; + height: number; + size: number; + }>; + @Column({ name: 'metadata', type: 'json', nullable: true }) + metadata: { + width?: number; + height?: number; + duration?: number; // for video/audio + bitrate?: number; + codec?: string; + }; + @Column({ name: 'owner_id', nullable: true }) + ownerId?: string; + @Column({ name: 'tenant_id', nullable: true }) + tenantId?: string; + @Column({ name: 'error_message', nullable: true }) + errorMessage: string; + @Column({ name: 'retry_count', default: 0 }) + retryCount: number; + @Column({ name: 'last_accessed_at', nullable: true }) + lastAccessedAt: Date; + @Column({ name: 'access_count', default: 0 }) + accessCount: number; + @CreateDateColumn({ name: 'created_at' }) + createdAt: Date; + @UpdateDateColumn({ name: 'updated_at' }) + updatedAt: Date; } diff --git a/src/cdn/geo/geo-location.service.ts b/src/cdn/geo/geo-location.service.ts index abb256e9..d549ab31 100644 --- a/src/cdn/geo/geo-location.service.ts +++ b/src/cdn/geo/geo-location.service.ts @@ -1,258 +1,217 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; - export interface LocationInfo { - country: string; - region: string; - city: string; - latitude: number; - longitude: number; - timezone: string; - isp?: string; - connectionType?: string; + country: string; + region: string; + city: string; + latitude: number; + longitude: number; + timezone: string; + isp?: string; + connectionType?: string; } - export interface EdgeLocation { - id: string; - name: string; - country: string; - latitude: number; - longitude: number; - provider: string; - priority: number; + id: string; + name: string; + country: string; + latitude: number; + longitude: number; + provider: string; + priority: number; } - @Injectable() export class GeoLocationService { - private readonly logger = new Logger(GeoLocationService.name); - private edgeLocations: EdgeLocation[] = [ - { - id: 'us-east-1', - name: 'Virginia', - country: 'US', - latitude: 39.0438, - longitude: -77.4874, - provider: 'cloudflare', - priority: 1, - }, - { - id: 'us-west-1', - name: 'California', - country: 'US', - latitude: 37.7749, - longitude: -122.4194, - provider: 'cloudflare', - priority: 2, - }, - { - id: 'eu-west-1', - name: 'Ireland', - country: 'IE', - latitude: 53.1424, - longitude: -7.6921, - provider: 'cloudflare', - priority: 1, - }, - { - id: 'eu-central-1', - name: 'Germany', - country: 'DE', - latitude: 50.1109, - longitude: 8.6821, - provider: 'cloudflare', - priority: 2, - }, - { - id: 'ap-southeast-1', - name: 'Singapore', - country: 'SG', - latitude: 1.3521, - longitude: 103.8198, - provider: 'cloudflare', - priority: 1, - }, - { - id: 'ap-northeast-1', - name: 'Japan', - country: 'JP', - latitude: 35.6762, - longitude: 139.6503, - provider: 'cloudflare', - priority: 2, - }, - ]; - - constructor(private configService: ConfigService) {} - - async getLocationInfo(ipAddress: string): Promise { - try { - // In real implementation, use a geolocation service like MaxMind or IP-API - // For now, return mock data based on IP - return this.mockGeolocation(ipAddress); - } catch (error) { - this.logger.error(`Failed to get location for IP ${ipAddress}:`, error); - return null; + private readonly logger = new Logger(GeoLocationService.name); + private edgeLocations: EdgeLocation[] = [ + { + id: 'us-east-1', + name: 'Virginia', + country: 'US', + latitude: 39.0438, + longitude: -77.4874, + provider: 'cloudflare', + priority: 1, + }, + { + id: 'us-west-1', + name: 'California', + country: 'US', + latitude: 37.7749, + longitude: -122.4194, + provider: 'cloudflare', + priority: 2, + }, + { + id: 'eu-west-1', + name: 'Ireland', + country: 'IE', + latitude: 53.1424, + longitude: -7.6921, + provider: 'cloudflare', + priority: 1, + }, + { + id: 'eu-central-1', + name: 'Germany', + country: 'DE', + latitude: 50.1109, + longitude: 8.6821, + provider: 'cloudflare', + priority: 2, + }, + { + id: 'ap-southeast-1', + name: 'Singapore', + country: 'SG', + latitude: 1.3521, + longitude: 103.8198, + provider: 'cloudflare', + priority: 1, + }, + { + id: 'ap-northeast-1', + name: 'Japan', + country: 'JP', + latitude: 35.6762, + longitude: 139.6503, + provider: 'cloudflare', + priority: 2, + }, + ]; + constructor(private configService: ConfigService) { } + async getLocationInfo(ipAddress: string): Promise { + try { + // In real implementation, use a geolocation service like MaxMind or IP-API + // For now, return mock data based on IP + return this.mockGeolocation(ipAddress); + } + catch (error) { + this.logger.error(`Failed to get location for IP ${ipAddress}:`, error); + return null; + } } - } - - async getOptimalLocation(userLocation?: string): Promise { - if (!userLocation) { - // Default to primary edge location - return this.edgeLocations[0].id; + async getOptimalLocation(userLocation?: string): Promise { + if (!userLocation) { + // Default to primary edge location + return this.edgeLocations[0].id; + } + const userCoords = await this.getCoordinates(userLocation); + if (!userCoords) { + return this.edgeLocations[0].id; + } + let optimalLocation = this.edgeLocations[0]; + let minDistance = this.calculateDistance(userCoords.latitude, userCoords.longitude, optimalLocation.latitude, optimalLocation.longitude); + for (const location of this.edgeLocations.slice(1)) { + const distance = this.calculateDistance(userCoords.latitude, userCoords.longitude, location.latitude, location.longitude); + if (distance < minDistance) { + minDistance = distance; + optimalLocation = location; + } + } + return optimalLocation.id; } - - const userCoords = await this.getCoordinates(userLocation); - if (!userCoords) { - return this.edgeLocations[0].id; + async getNearestEdgeLocations(userLocation: string, limit: number = 3): Promise { + const userCoords = await this.getCoordinates(userLocation); + if (!userCoords) { + return this.edgeLocations.slice(0, limit); + } + const sortedLocations = this.edgeLocations + .map((location) => ({ + ...location, + distance: this.calculateDistance(userCoords.latitude, userCoords.longitude, location.latitude, location.longitude), + })) + .sort((a, b) => a.distance - b.distance); + return sortedLocations.slice(0, limit); } - - let optimalLocation = this.edgeLocations[0]; - let minDistance = this.calculateDistance( - userCoords.latitude, - userCoords.longitude, - optimalLocation.latitude, - optimalLocation.longitude, - ); - - for (const location of this.edgeLocations.slice(1)) { - const distance = this.calculateDistance( - userCoords.latitude, - userCoords.longitude, - location.latitude, - location.longitude, - ); - - if (distance < minDistance) { - minDistance = distance; - optimalLocation = location; - } + async optimizeRouteForConnection(userLocation: string, connectionType: string): Promise { + const locations = await this.getNearestEdgeLocations(userLocation, 5); + // Adjust based on connection type + switch (connectionType.toLowerCase()) { + case 'mobile': + case '3g': + case '4g': + // Prefer locations with better mobile optimization + return locations[0].id; + case 'satellite': + // Prefer locations with lower latency for satellite + return locations[0].id; + default: + return locations[0].id; + } } - - return optimalLocation.id; - } - - async getNearestEdgeLocations(userLocation: string, limit: number = 3): Promise { - const userCoords = await this.getCoordinates(userLocation); - if (!userCoords) { - return this.edgeLocations.slice(0, limit); + async getLatencyEstimates(userLocation: string): Promise> { + const userCoords = await this.getCoordinates(userLocation); + const estimates: Record = {}; + for (const location of this.edgeLocations) { + const distance = this.calculateDistance(userCoords.latitude, userCoords.longitude, location.latitude, location.longitude); + // Rough estimate: 1ms per 100km + estimates[location.id] = Math.round(distance / 100); + } + return estimates; } - - const sortedLocations = this.edgeLocations - .map((location) => ({ - ...location, - distance: this.calculateDistance( - userCoords.latitude, - userCoords.longitude, - location.latitude, - location.longitude, - ), - })) - .sort((a, b) => a.distance - b.distance); - - return sortedLocations.slice(0, limit); - } - - async optimizeRouteForConnection(userLocation: string, connectionType: string): Promise { - const locations = await this.getNearestEdgeLocations(userLocation, 5); - - // Adjust based on connection type - switch (connectionType.toLowerCase()) { - case 'mobile': - case '3g': - case '4g': - // Prefer locations with better mobile optimization - return locations[0].id; - case 'satellite': - // Prefer locations with lower latency for satellite - return locations[0].id; - default: - return locations[0].id; + private async getCoordinates(location: string): Promise<{ + latitude: number; + longitude: number; + } | null> { + // In real implementation, use geocoding service + // For now, return mock coordinates + const mockCoords: Record = { + 'new york': { latitude: 40.7128, longitude: -74.006 }, + london: { latitude: 51.5074, longitude: -0.1278 }, + tokyo: { latitude: 35.6762, longitude: 139.6503 }, + sydney: { latitude: -33.8688, longitude: 151.2093 }, + lagos: { latitude: 6.5244, longitude: 3.3792 }, + }; + return mockCoords[location.toLowerCase()] || null; } - } - - async getLatencyEstimates(userLocation: string): Promise> { - const userCoords = await this.getCoordinates(userLocation); - const estimates: Record = {}; - - for (const location of this.edgeLocations) { - const distance = this.calculateDistance( - userCoords.latitude, - userCoords.longitude, - location.latitude, - location.longitude, - ); - - // Rough estimate: 1ms per 100km - estimates[location.id] = Math.round(distance / 100); + private calculateDistance(lat1: number, lon1: number, lat2: number, lon2: number): number { + const R = 6371; // Earth's radius in kilometers + const dLat = this.toRadians(lat2 - lat1); + const dLon = this.toRadians(lon2 - lon1); + const a = Math.sin(dLat / 2) * Math.sin(dLat / 2) + + Math.cos(this.toRadians(lat1)) * + Math.cos(this.toRadians(lat2)) * + Math.sin(dLon / 2) * + Math.sin(dLon / 2); + const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); + return R * c; + } + private toRadians(degrees: number): number { + return degrees * (Math.PI / 180); + } + private mockGeolocation(_ipAddress: string): LocationInfo { + // Mock implementation - in real app, use actual geolocation service + return { + country: 'US', + region: 'CA', + city: 'San Francisco', + latitude: 37.7749, + longitude: -122.4194, + timezone: 'America/Los_Angeles', + isp: 'Mock ISP', + connectionType: 'fiber', + }; + } + async getGeoStats(): Promise<{ + totalRequests: number; + topCountries: Array<{ + country: string; + count: number; + }>; + averageLatency: number; + }> { + // Implementation would aggregate geolocation analytics + return { + totalRequests: 100000, + topCountries: [ + { country: 'US', count: 45000 }, + { country: 'UK', count: 15000 }, + { country: 'DE', count: 12000 }, + ], + averageLatency: 45, + }; } - - return estimates; - } - - private async getCoordinates( - location: string, - ): Promise<{ latitude: number; longitude: number } | null> { - // In real implementation, use geocoding service - // For now, return mock coordinates - const mockCoords: Record = { - 'new york': { latitude: 40.7128, longitude: -74.006 }, - london: { latitude: 51.5074, longitude: -0.1278 }, - tokyo: { latitude: 35.6762, longitude: 139.6503 }, - sydney: { latitude: -33.8688, longitude: 151.2093 }, - lagos: { latitude: 6.5244, longitude: 3.3792 }, - }; - - return mockCoords[location.toLowerCase()] || null; - } - - private calculateDistance(lat1: number, lon1: number, lat2: number, lon2: number): number { - const R = 6371; // Earth's radius in kilometers - const dLat = this.toRadians(lat2 - lat1); - const dLon = this.toRadians(lon2 - lon1); - - const a = - Math.sin(dLat / 2) * Math.sin(dLat / 2) + - Math.cos(this.toRadians(lat1)) * - Math.cos(this.toRadians(lat2)) * - Math.sin(dLon / 2) * - Math.sin(dLon / 2); - - const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); - return R * c; - } - - private toRadians(degrees: number): number { - return degrees * (Math.PI / 180); - } - - private mockGeolocation(_ipAddress: string): LocationInfo { - // Mock implementation - in real app, use actual geolocation service - return { - country: 'US', - region: 'CA', - city: 'San Francisco', - latitude: 37.7749, - longitude: -122.4194, - timezone: 'America/Los_Angeles', - isp: 'Mock ISP', - connectionType: 'fiber', - }; - } - - async getGeoStats(): Promise<{ - totalRequests: number; - topCountries: Array<{ country: string; count: number }>; - averageLatency: number; - }> { - // Implementation would aggregate geolocation analytics - return { - totalRequests: 100000, - topCountries: [ - { country: 'US', count: 45000 }, - { country: 'UK', count: 15000 }, - { country: 'DE', count: 12000 }, - ], - averageLatency: 45, - }; - } } diff --git a/src/cdn/optimization/asset-optimization.service.ts b/src/cdn/optimization/asset-optimization.service.ts index a1464b89..c5da00f1 100644 --- a/src/cdn/optimization/asset-optimization.service.ts +++ b/src/cdn/optimization/asset-optimization.service.ts @@ -1,140 +1,122 @@ import { Injectable } from '@nestjs/common'; import sharp from 'sharp'; import { ContentDeliveryOptions } from '../cdn.service'; - export interface OptimizationResult { - url: string; - originalSize: number; - optimizedSize: number; - format: string; + url: string; + originalSize: number; + optimizedSize: number; + format: string; } - @Injectable() export class AssetOptimizationService { - async optimizeImage(contentId: string, _options: any): Promise { - try { - // Download image (in real implementation, you'd fetch from storage) - // For now, assume we have the buffer - const buffer = await this.downloadImage(contentId); - - let sharpInstance = sharp(buffer); - - // Apply optimizations - if (_options.width || _options.height) { - sharpInstance = sharpInstance.resize({ - width: _options.width, - height: _options.height, - fit: 'cover', - withoutEnlargement: true, - }); - } - - if (_options.quality) { - sharpInstance = sharpInstance.jpeg({ quality: _options.quality }); - } - - if (_options.format) { - switch (_options.format) { - case 'webp': - sharpInstance = sharpInstance.webp({ quality: _options.quality || 80 }); - break; - case 'png': - sharpInstance = sharpInstance.png({ quality: _options.quality || 80 }); - break; - case 'jpeg': - default: - sharpInstance = sharpInstance.jpeg({ quality: _options.quality || 80 }); - break; + async optimizeImage(contentId: string, _options: unknown): Promise { + try { + // Download image (in real implementation, you'd fetch from storage) + // For now, assume we have the buffer + const buffer = await this.downloadImage(contentId); + let sharpInstance = sharp(buffer); + // Apply optimizations + if (_options.width || _options.height) { + sharpInstance = sharpInstance.resize({ + width: _options.width, + height: _options.height, + fit: 'cover', + withoutEnlargement: true, + }); + } + if (_options.quality) { + sharpInstance = sharpInstance.jpeg({ quality: _options.quality }); + } + if (_options.format) { + switch (_options.format) { + case 'webp': + sharpInstance = sharpInstance.webp({ quality: _options.quality || 80 }); + break; + case 'png': + sharpInstance = sharpInstance.png({ quality: _options.quality || 80 }); + break; + case 'jpeg': + default: + sharpInstance = sharpInstance.jpeg({ quality: _options.quality || 80 }); + break; + } + } + const optimizedBuffer = await sharpInstance.toBuffer(); + const optimizedUrl = await this.uploadOptimizedImage(optimizedBuffer, contentId, _options); + return optimizedUrl; + } + catch (error) { + console.error('Image optimization failed:', error); + return contentId; // Return original if optimization fails } - } - - const optimizedBuffer = await sharpInstance.toBuffer(); - const optimizedUrl = await this.uploadOptimizedImage(optimizedBuffer, contentId, _options); - - return optimizedUrl; - } catch (error) { - console.error('Image optimization failed:', error); - return contentId; // Return original if optimization fails } - } - - async optimizeVideo(contentId: string, _options: any): Promise { - // Implementation for video optimization using ffmpeg - // For now, return original - return contentId; - } - - async generateResponsiveImages(contentId: string): Promise { - const results: OptimizationResult[] = []; - const sizes = [ - { width: 320, suffix: 'sm' }, - { width: 640, suffix: 'md' }, - { width: 1024, suffix: 'lg' }, - { width: 1920, suffix: 'xl' }, - ]; - - const buffer = await this.downloadImage(contentId); - - for (const size of sizes) { - const optimized = await sharp(buffer) - .resize(size.width, null, { withoutEnlargement: true }) - .webp({ quality: 80 }) - .toBuffer(); - - const url = await this.uploadOptimizedImage(optimized, contentId, { - width: size.width, - format: 'webp', - }); - - results.push({ - url, - originalSize: buffer.length, - optimizedSize: optimized.length, - format: 'webp', - }); + async optimizeVideo(contentId: string, _options: unknown): Promise { + // Implementation for video optimization using ffmpeg + // For now, return original + return contentId; + } + async generateResponsiveImages(contentId: string): Promise { + const results: OptimizationResult[] = []; + const sizes = [ + { width: 320, suffix: 'sm' }, + { width: 640, suffix: 'md' }, + { width: 1024, suffix: 'lg' }, + { width: 1920, suffix: 'xl' }, + ]; + const buffer = await this.downloadImage(contentId); + for (const size of sizes) { + const optimized = await sharp(buffer) + .resize(size.width, null, { withoutEnlargement: true }) + .webp({ quality: 80 }) + .toBuffer(); + const url = await this.uploadOptimizedImage(optimized, contentId, { + width: size.width, + format: 'webp', + }); + results.push({ + url, + originalSize: buffer.length, + optimizedSize: optimized.length, + format: 'webp', + }); + } + return results; + } + private async downloadImage(_url: string): Promise { + // In real implementation, download from storage/CDN + // For now, return empty buffer + throw new Error('Download implementation needed'); + } + private async uploadOptimizedImage(buffer: Buffer, originalUrl: string, options: ContentDeliveryOptions): Promise { + // In real implementation, upload to storage and return new URL + // For now, return modified URL + const suffix = this.generateSuffix(options); + return originalUrl.replace(/(\.[^.]+)$/, `_${suffix}$1`); + } + private generateSuffix(options: ContentDeliveryOptions): string { + const parts = []; + if (options.width) + parts.push(`w${options.width}`); + if (options.height) + parts.push(`h${options.height}`); + if (options.quality) + parts.push(`q${options.quality}`); + if (options.format) + parts.push(options.format); + return parts.join('_'); + } + async getOptimizationStats(_contentId: string): Promise<{ + originalSize: number; + optimizedSize: number; + savingsPercentage: number; + formats: string[]; + }> { + // Implementation would fetch stats from database/cache + return { + originalSize: 2048000, + optimizedSize: 512000, + savingsPercentage: 75, + formats: ['webp', 'jpeg'], + }; } - - return results; - } - - private async downloadImage(_url: string): Promise { - // In real implementation, download from storage/CDN - // For now, return empty buffer - throw new Error('Download implementation needed'); - } - - private async uploadOptimizedImage( - buffer: Buffer, - originalUrl: string, - options: ContentDeliveryOptions, - ): Promise { - // In real implementation, upload to storage and return new URL - // For now, return modified URL - const suffix = this.generateSuffix(options); - return originalUrl.replace(/(\.[^.]+)$/, `_${suffix}$1`); - } - - private generateSuffix(options: ContentDeliveryOptions): string { - const parts = []; - if (options.width) parts.push(`w${options.width}`); - if (options.height) parts.push(`h${options.height}`); - if (options.quality) parts.push(`q${options.quality}`); - if (options.format) parts.push(options.format); - return parts.join('_'); - } - - async getOptimizationStats(_contentId: string): Promise<{ - originalSize: number; - optimizedSize: number; - savingsPercentage: number; - formats: string[]; - }> { - // Implementation would fetch stats from database/cache - return { - originalSize: 2048000, - optimizedSize: 512000, - savingsPercentage: 75, - formats: ['webp', 'jpeg'], - }; - } } diff --git a/src/cdn/providers/aws-cloudfront.service.ts b/src/cdn/providers/aws-cloudfront.service.ts index 6e5db618..c8eb9e92 100644 --- a/src/cdn/providers/aws-cloudfront.service.ts +++ b/src/cdn/providers/aws-cloudfront.service.ts @@ -1,282 +1,244 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; -import { - CloudFrontClient, - CreateInvalidationCommand, - GetInvalidationCommand, -} from '@aws-sdk/client-cloudfront'; +import { CloudFrontClient, CreateInvalidationCommand, GetInvalidationCommand, } from '@aws-sdk/client-cloudfront'; import { S3Client, PutObjectCommand, DeleteObjectCommand } from '@aws-sdk/client-s3'; - export interface FileUpload { - originalname: string; - buffer: Buffer; - mimetype: string; - size: number; + originalname: string; + buffer: Buffer; + mimetype: string; + size: number; } - export interface AWSCloudFrontConfig { - accessKeyId: string; - secretAccessKey: string; - region: string; - distributionId: string; - bucketName?: string; + accessKeyId: string; + secretAccessKey: string; + region: string; + distributionId: string; + bucketName?: string; } - export interface UploadResult { - id: string; - url: string; - etag?: string; - size: number; + id: string; + url: string; + etag?: string; + size: number; } - export interface PurgeResult { - success: boolean; - purgedUrls: string[]; - failedUrls: string[]; - invalidationId?: string; + success: boolean; + purgedUrls: string[]; + failedUrls: string[]; + invalidationId?: string; } - @Injectable() export class AWSCloudFrontService { - private readonly logger = new Logger(AWSCloudFrontService.name); - private readonly cloudfrontClient: CloudFrontClient; - private readonly s3Client: S3Client; - private readonly config: AWSCloudFrontConfig; - - constructor(private configService: ConfigService) { - this.config = { - accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), - secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), - region: this.configService.get('AWS_REGION', 'us-east-1'), - distributionId: this.configService.get('AWS_CLOUDFRONT_DISTRIBUTION_ID', ''), - bucketName: this.configService.get('AWS_S3_BUCKET_NAME'), - }; - - this.cloudfrontClient = new CloudFrontClient({ - region: this.config.region, - credentials: { - accessKeyId: this.config.accessKeyId, - secretAccessKey: this.config.secretAccessKey, - }, - }); - - this.s3Client = new S3Client({ - region: this.config.region, - credentials: { - accessKeyId: this.config.accessKeyId, - secretAccessKey: this.config.secretAccessKey, - }, - }); - } - - async uploadFile(file: FileUpload): Promise { - try { - this.logger.log(`Uploading file ${file.originalname} to AWS CloudFront/S3`); - - if (!this.config.bucketName) { - throw new Error('S3 bucket name not configured'); - } - - const key = `uploads/${Date.now()}_${file.originalname}`; - - const command = new PutObjectCommand({ - Bucket: this.config.bucketName, - Key: key, - Body: file.buffer, - ContentType: file.mimetype, - ACL: 'public-read', // Make it publicly accessible - }); - - const result = await this.s3Client.send(command); - - const url = `https://${this.config.distributionId}.cloudfront.net/${key}`; - - return { - id: key, - url, - etag: result.ETag, - size: file.size, - }; - } catch (error) { - this.logger.error('AWS CloudFront upload failed:', error); - throw new Error(`Failed to upload file to AWS CloudFront: ${error.message}`); + private readonly logger = new Logger(AWSCloudFrontService.name); + private readonly cloudfrontClient: CloudFrontClient; + private readonly s3Client: S3Client; + private readonly config: AWSCloudFrontConfig; + constructor(private configService: ConfigService) { + this.config = { + accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), + secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), + region: this.configService.get('AWS_REGION', 'us-east-1'), + distributionId: this.configService.get('AWS_CLOUDFRONT_DISTRIBUTION_ID', ''), + bucketName: this.configService.get('AWS_S3_BUCKET_NAME'), + }; + this.cloudfrontClient = new CloudFrontClient({ + region: this.config.region, + credentials: { + accessKeyId: this.config.accessKeyId, + secretAccessKey: this.config.secretAccessKey, + }, + }); + this.s3Client = new S3Client({ + region: this.config.region, + credentials: { + accessKeyId: this.config.accessKeyId, + secretAccessKey: this.config.secretAccessKey, + }, + }); } - } - - async purgeUrls(urls: string[]): Promise { - try { - this.logger.log(`Creating CloudFront invalidation for ${urls.length} URLs`); - - // Convert full URLs to paths relative to distribution - const paths = urls.map((url) => { + async uploadFile(file: FileUpload): Promise { try { - const urlObj = new URL(url); - return urlObj.pathname; - } catch { - // If not a full URL, assume it's already a path - return url.startsWith('/') ? url : `/${url}`; + this.logger.log(`Uploading file ${file.originalname} to AWS CloudFront/S3`); + if (!this.config.bucketName) { + throw new Error('S3 bucket name not configured'); + } + const key = `uploads/${Date.now()}_${file.originalname}`; + const command = new PutObjectCommand({ + Bucket: this.config.bucketName, + Key: key, + Body: file.buffer, + ContentType: file.mimetype, + ACL: 'public-read', // Make it publicly accessible + }); + const result = await this.s3Client.send(command); + const url = `https://${this.config.distributionId}.cloudfront.net/${key}`; + return { + id: key, + url, + etag: result.ETag, + size: file.size, + }; + } + catch (error) { + this.logger.error('AWS CloudFront upload failed:', error); + throw new Error(`Failed to upload file to AWS CloudFront: ${error.message}`); } - }); - - const command = new CreateInvalidationCommand({ - DistributionId: this.config.distributionId, - InvalidationBatch: { - CallerReference: `cdn-purge-${Date.now()}`, - Paths: { - Quantity: paths.length, - Items: paths, - }, - }, - }); - - const result = await this.cloudfrontClient.send(command); - - // Wait for invalidation to complete - await this.waitForInvalidation(result.Invalidation?.Id || ''); - - return { - success: true, - purgedUrls: urls, - failedUrls: [], - invalidationId: result.Invalidation?.Id, - }; - } catch (error) { - this.logger.error('CloudFront invalidation failed:', error); - return { - success: false, - purgedUrls: [], - failedUrls: urls, - }; } - } - - async purgeEverything(): Promise { - try { - this.logger.log('Creating CloudFront invalidation for all content'); - - const command = new CreateInvalidationCommand({ - DistributionId: this.config.distributionId, - InvalidationBatch: { - CallerReference: `cdn-purge-all-${Date.now()}`, - Paths: { - Quantity: 1, - Items: ['/*'], // Invalidate all paths - }, - }, - }); - - const result = await this.cloudfrontClient.send(command); - - await this.waitForInvalidation(result.Invalidation?.Id || ''); - - return { - success: true, - purgedUrls: ['/*'], - failedUrls: [], - invalidationId: result.Invalidation?.Id, - }; - } catch (error) { - this.logger.error('CloudFront purge everything failed:', error); - return { - success: false, - purgedUrls: [], - failedUrls: ['/*'], - }; + async purgeUrls(urls: string[]): Promise { + try { + this.logger.log(`Creating CloudFront invalidation for ${urls.length} URLs`); + // Convert full URLs to paths relative to distribution + const paths = urls.map((url) => { + try { + const urlObj = new URL(url); + return urlObj.pathname; + } + catch { + // If not a full URL, assume it's already a path + return url.startsWith('/') ? url : `/${url}`; + } + }); + const command = new CreateInvalidationCommand({ + DistributionId: this.config.distributionId, + InvalidationBatch: { + CallerReference: `cdn-purge-${Date.now()}`, + Paths: { + Quantity: paths.length, + Items: paths, + }, + }, + }); + const result = await this.cloudfrontClient.send(command); + // Wait for invalidation to complete + await this.waitForInvalidation(result.Invalidation?.Id || ''); + return { + success: true, + purgedUrls: urls, + failedUrls: [], + invalidationId: result.Invalidation?.Id, + }; + } + catch (error) { + this.logger.error('CloudFront invalidation failed:', error); + return { + success: false, + purgedUrls: [], + failedUrls: urls, + }; + } } - } - - async getUsageStatistics(_startDate: Date, _endDate: Date): Promise { - // AWS CloudFront doesn't have direct metrics API in SDK - // Would need to use CloudWatch or external monitoring - // For now, return mock data - return { - requests: 100000, - bytesDownloaded: 5000000000, - errorRate: 0.01, - topUrls: [ - { url: '/index.html', requests: 50000 }, - { url: '/main.js', requests: 30000 }, - ], - }; - } - - async updateDistributionSettings(_settings: any): Promise { - try { - // Get current distribution config - // This would require additional API calls to get and update distribution - // For now, return success - this.logger.log('Updating CloudFront distribution settings'); - } catch (error) { - this.logger.error('Failed to update distribution settings:', error); + async purgeEverything(): Promise { + try { + this.logger.log('Creating CloudFront invalidation for all content'); + const command = new CreateInvalidationCommand({ + DistributionId: this.config.distributionId, + InvalidationBatch: { + CallerReference: `cdn-purge-all-${Date.now()}`, + Paths: { + Quantity: 1, + Items: ['/*'], // Invalidate all paths + }, + }, + }); + const result = await this.cloudfrontClient.send(command); + await this.waitForInvalidation(result.Invalidation?.Id || ''); + return { + success: true, + purgedUrls: ['/*'], + failedUrls: [], + invalidationId: result.Invalidation?.Id, + }; + } + catch (error) { + this.logger.error('CloudFront purge everything failed:', error); + return { + success: false, + purgedUrls: [], + failedUrls: ['/*'], + }; + } } - } - - async createOriginAccessIdentity(): Promise { - // Implementation would create CloudFront Origin Access Identity - // for secure S3 access - const identityId = `origin-access-identity-${Date.now()}`; - this.logger.log(`Created Origin Access Identity: ${identityId}`); - return identityId; - } - - private async waitForInvalidation(invalidationId: string): Promise { - const maxAttempts = 30; // 5 minutes with 10s intervals - let attempts = 0; - - while (attempts < maxAttempts) { - try { - const command = new GetInvalidationCommand({ - DistributionId: this.config.distributionId, - Id: invalidationId, - }); - - const result = await this.cloudfrontClient.send(command); - - if (result.Invalidation?.Status === 'Completed') { - this.logger.log(`Invalidation ${invalidationId} completed`); - return; + async getUsageStatistics(_startDate: Date, _endDate: Date): Promise { + // AWS CloudFront doesn't have direct metrics API in SDK + // Would need to use CloudWatch or external monitoring + // For now, return mock data + return { + requests: 100000, + bytesDownloaded: 5000000000, + errorRate: 0.01, + topUrls: [ + { url: '/index.html', requests: 50000 }, + { url: '/main.js', requests: 30000 }, + ], + }; + } + async updateDistributionSettings(_settings: unknown): Promise { + try { + // Get current distribution config + // This would require additional API calls to get and update distribution + // For now, return success + this.logger.log('Updating CloudFront distribution settings'); + } + catch (error) { + this.logger.error('Failed to update distribution settings:', error); + } + } + async createOriginAccessIdentity(): Promise { + // Implementation would create CloudFront Origin Access Identity + // for secure S3 access + const identityId = `origin-access-identity-${Date.now()}`; + this.logger.log(`Created Origin Access Identity: ${identityId}`); + return identityId; + } + private async waitForInvalidation(invalidationId: string): Promise { + const maxAttempts = 30; // 5 minutes with 10s intervals + let attempts = 0; + while (attempts < maxAttempts) { + try { + const command = new GetInvalidationCommand({ + DistributionId: this.config.distributionId, + Id: invalidationId, + }); + const result = await this.cloudfrontClient.send(command); + if (result.Invalidation?.Status === 'Completed') { + this.logger.log(`Invalidation ${invalidationId} completed`); + return; + } + await new Promise((resolve) => setTimeout(resolve, 10000)); // Wait 10 seconds + attempts++; + } + catch (error) { + this.logger.error('Error checking invalidation status:', error); + attempts++; + } + } + throw new Error(`Invalidation ${invalidationId} did not complete within timeout`); + } + async deleteFile(key: string): Promise { + try { + if (!this.config.bucketName) { + throw new Error('S3 bucket name not configured'); + } + const command = new DeleteObjectCommand({ + Bucket: this.config.bucketName, + Key: key, + }); + await this.s3Client.send(command); + // Invalidate the deleted file + await this.purgeUrls([`https://${this.config.distributionId}.cloudfront.net/${key}`]); + return true; + } + catch (error) { + this.logger.error(`Failed to delete file ${key}:`, error); + return false; } - - await new Promise((resolve) => setTimeout(resolve, 10000)); // Wait 10 seconds - attempts++; - } catch (error) { - this.logger.error('Error checking invalidation status:', error); - attempts++; - } } - - throw new Error(`Invalidation ${invalidationId} did not complete within timeout`); - } - - async deleteFile(key: string): Promise { - try { - if (!this.config.bucketName) { - throw new Error('S3 bucket name not configured'); - } - - const command = new DeleteObjectCommand({ - Bucket: this.config.bucketName, - Key: key, - }); - - await this.s3Client.send(command); - - // Invalidate the deleted file - await this.purgeUrls([`https://${this.config.distributionId}.cloudfront.net/${key}`]); - - return true; - } catch (error) { - this.logger.error(`Failed to delete file ${key}:`, error); - return false; + async getFileMetadata(_key: string): Promise { + // Implementation would get object metadata from S3 + return { + size: 0, + lastModified: new Date(), + contentType: 'application/octet-stream', + }; } - } - - async getFileMetadata(_key: string): Promise { - // Implementation would get object metadata from S3 - return { - size: 0, - lastModified: new Date(), - contentType: 'application/octet-stream', - }; - } } diff --git a/src/cdn/providers/cloudflare.service.ts b/src/cdn/providers/cloudflare.service.ts index 355a9369..135a0f2a 100644 --- a/src/cdn/providers/cloudflare.service.ts +++ b/src/cdn/providers/cloudflare.service.ts @@ -2,225 +2,192 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import axios, { AxiosInstance } from 'axios'; import { UploadedFile } from '../../common/types/file.types'; - export interface CloudflareConfig { - apiToken: string; - accountId: string; - zoneId: string; - baseUrl?: string; + apiToken: string; + accountId: string; + zoneId: string; + baseUrl?: string; } - export interface UploadResult { - id: string; - url: string; - etag?: string; - size: number; + id: string; + url: string; + etag?: string; + size: number; } - export interface PurgeResult { - success: boolean; - purgedUrls: string[]; - failedUrls: string[]; + success: boolean; + purgedUrls: string[]; + failedUrls: string[]; } - @Injectable() export class CloudflareService { - private readonly logger = new Logger(CloudflareService.name); - private readonly httpClient: AxiosInstance; - private readonly config: CloudflareConfig; - - constructor(private configService: ConfigService) { - this.config = { - apiToken: this.configService.get('CLOUDFLARE_API_TOKEN', ''), - accountId: this.configService.get('CLOUDFLARE_ACCOUNT_ID', ''), - zoneId: this.configService.get('CLOUDFLARE_ZONE_ID', ''), - baseUrl: 'https://api.cloudflare.com/client/v4', - }; - - this.httpClient = axios.create({ - baseURL: this.config.baseUrl, - headers: { - Authorization: `Bearer ${this.config.apiToken}`, - 'Content-Type': 'application/json', - }, - }); - } - - async uploadFile(file: UploadedFile): Promise { - try { - this.logger.log(`Uploading file ${file.originalname} to Cloudflare`); - - // For images, use Cloudflare Images API - if (file.mimetype.startsWith('image/')) { - return this.uploadImage(file); - } - - // For other files, use R2 or Stream - return this.uploadToR2(file); - } catch (error) { - this.logger.error('Cloudflare upload failed:', error); - throw new Error(`Failed to upload file to Cloudflare: ${error.message}`); - } - } - - async purgeUrls(urls: string[]): Promise { - try { - this.logger.log(`Purging ${urls.length} URLs from Cloudflare`); - - const response = await this.httpClient.post(`/zones/${this.config.zoneId}/purge_cache`, { - files: urls, - }); - - if (response.data.success) { - return { - success: true, - purgedUrls: urls, - failedUrls: [], - }; - } else { - this.logger.error('Cloudflare purge failed:', response.data.errors); - return { - success: false, - purgedUrls: [], - failedUrls: urls, + private readonly logger = new Logger(CloudflareService.name); + private readonly httpClient: AxiosInstance; + private readonly config: CloudflareConfig; + constructor(private configService: ConfigService) { + this.config = { + apiToken: this.configService.get('CLOUDFLARE_API_TOKEN', ''), + accountId: this.configService.get('CLOUDFLARE_ACCOUNT_ID', ''), + zoneId: this.configService.get('CLOUDFLARE_ZONE_ID', ''), + baseUrl: 'https://api.cloudflare.com/client/v4', }; - } - } catch (error) { - this.logger.error('Cloudflare purge error:', error); - return { - success: false, - purgedUrls: [], - failedUrls: urls, - }; + this.httpClient = axios.create({ + baseURL: this.config.baseUrl, + headers: { + Authorization: `Bearer ${this.config.apiToken}`, + 'Content-Type': 'application/json', + }, + }); + } + async uploadFile(file: UploadedFile): Promise { + try { + this.logger.log(`Uploading file ${file.originalname} to Cloudflare`); + // For images, use Cloudflare Images API + if (file.mimetype.startsWith('image/')) { + return this.uploadImage(file); + } + // For other files, use R2 or Stream + return this.uploadToR2(file); + } + catch (error) { + this.logger.error('Cloudflare upload failed:', error); + throw new Error(`Failed to upload file to Cloudflare: ${error.message}`); + } + } + async purgeUrls(urls: string[]): Promise { + try { + this.logger.log(`Purging ${urls.length} URLs from Cloudflare`); + const response = await this.httpClient.post(`/zones/${this.config.zoneId}/purge_cache`, { + files: urls, + }); + if (response.data.success) { + return { + success: true, + purgedUrls: urls, + failedUrls: [], + }; + } + else { + this.logger.error('Cloudflare purge failed:', response.data.errors); + return { + success: false, + purgedUrls: [], + failedUrls: urls, + }; + } + } + catch (error) { + this.logger.error('Cloudflare purge error:', error); + return { + success: false, + purgedUrls: [], + failedUrls: urls, + }; + } } - } - - async purgeEverything(): Promise { - try { - const response = await this.httpClient.post(`/zones/${this.config.zoneId}/purge_cache`, { - purge_everything: true, - }); - - return response.data.success; - } catch (error) { - this.logger.error('Cloudflare purge everything failed:', error); - return false; + async purgeEverything(): Promise { + try { + const response = await this.httpClient.post(`/zones/${this.config.zoneId}/purge_cache`, { + purge_everything: true, + }); + return response.data.success; + } + catch (error) { + this.logger.error('Cloudflare purge everything failed:', error); + return false; + } } - } - - async getAnalytics(startDate: Date, endDate: Date): Promise { - try { - const response = await this.httpClient.get( - `/zones/${this.config.zoneId}/analytics/dashboard`, - { - params: { - since: startDate.toISOString(), - until: endDate.toISOString(), - }, - }, - ); - - return response.data.result; - } catch (error) { - this.logger.error('Failed to get Cloudflare analytics:', error); - return null; + async getAnalytics(startDate: Date, endDate: Date): Promise { + try { + const response = await this.httpClient.get(`/zones/${this.config.zoneId}/analytics/dashboard`, { + params: { + since: startDate.toISOString(), + until: endDate.toISOString(), + }, + }); + return response.data.result; + } + catch (error) { + this.logger.error('Failed to get Cloudflare analytics:', error); + return null; + } } - } - - async createCustomDomain(domain: string): Promise { - try { - const response = await this.httpClient.post( - `/zones/${this.config.zoneId}/custom_certificates`, - { - certificate: '', // Would need actual certificate - private_key: '', // Would need actual private key - bundle_method: 'ubiquitous', - }, - ); - - return response.data.success; - } catch (error) { - this.logger.error(`Failed to create custom domain ${domain}:`, error); - return false; + async createCustomDomain(domain: string): Promise { + try { + const response = await this.httpClient.post(`/zones/${this.config.zoneId}/custom_certificates`, { + certificate: '', // Would need actual certificate + private_key: '', // Would need actual private key + bundle_method: 'ubiquitous', + }); + return response.data.success; + } + catch (error) { + this.logger.error(`Failed to create custom domain ${domain}:`, error); + return false; + } } - } - - async getZoneSettings(): Promise { - try { - const response = await this.httpClient.get(`/zones/${this.config.zoneId}/settings`); - - return response.data.result; - } catch (error) { - this.logger.error('Failed to get zone settings:', error); - return null; + async getZoneSettings(): Promise { + try { + const response = await this.httpClient.get(`/zones/${this.config.zoneId}/settings`); + return response.data.result; + } + catch (error) { + this.logger.error('Failed to get zone settings:', error); + return null; + } } - } - - async updateCacheSettings(settings: any): Promise { - try { - const response = await this.httpClient.patch( - `/zones/${this.config.zoneId}/settings/cache_level`, - { - value: settings.cacheLevel || 'aggressive', - }, - ); - - return response.data.success; - } catch (error) { - this.logger.error('Failed to update cache settings:', error); - return false; + async updateCacheSettings(settings: unknown): Promise { + try { + const response = await this.httpClient.patch(`/zones/${this.config.zoneId}/settings/cache_level`, { + value: settings.cacheLevel || 'aggressive', + }); + return response.data.success; + } + catch (error) { + this.logger.error('Failed to update cache settings:', error); + return false; + } + } + private async uploadImage(file: UploadedFile): Promise { + // Use Cloudflare Images API + // In real implementation, would use proper multipart/form-data + // For now, return mock result + const mockId = `cf_img_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; + return { + id: mockId, + url: `https://imagedelivery.net/${mockId}/${file.originalname}`, + size: file.size, + }; + } + private async uploadToR2(file: UploadedFile): Promise { + // Use Cloudflare R2 for non-image files + // This would require R2 bucket configuration + // For now, return mock result + const mockId = `cf_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; + return { + id: mockId, + url: `https://r2.example.com/${mockId}/${file.originalname}`, + size: file.size, + }; } - } - - private async uploadImage(file: UploadedFile): Promise { - // Use Cloudflare Images API - // In real implementation, would use proper multipart/form-data - // For now, return mock result - const mockId = `cf_img_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; - - return { - id: mockId, - url: `https://imagedelivery.net/${mockId}/${file.originalname}`, - size: file.size, - }; - } - - private async uploadToR2(file: UploadedFile): Promise { - // Use Cloudflare R2 for non-image files - // This would require R2 bucket configuration - // For now, return mock result - const mockId = `cf_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; - - return { - id: mockId, - url: `https://r2.example.com/${mockId}/${file.originalname}`, - size: file.size, - }; - } - - async getImageVariants(imageId: string): Promise { - try { - const response = await this.httpClient.get( - `/accounts/${this.config.accountId}/images/v1/${imageId}/variants`, - ); - - return response.data.result?.variants || []; - } catch (error) { - this.logger.error(`Failed to get variants for image ${imageId}:`, error); - return []; + async getImageVariants(imageId: string): Promise { + try { + const response = await this.httpClient.get(`/accounts/${this.config.accountId}/images/v1/${imageId}/variants`); + return response.data.result?.variants || []; + } + catch (error) { + this.logger.error(`Failed to get variants for image ${imageId}:`, error); + return []; + } } - } - - async deleteImage(imageId: string): Promise { - try { - const response = await this.httpClient.delete( - `/accounts/${this.config.accountId}/images/v1/${imageId}`, - ); - - return response.data.success; - } catch (error) { - this.logger.error(`Failed to delete image ${imageId}:`, error); - return false; + async deleteImage(imageId: string): Promise { + try { + const response = await this.httpClient.delete(`/accounts/${this.config.accountId}/images/v1/${imageId}`); + return response.data.success; + } + catch (error) { + this.logger.error(`Failed to delete image ${imageId}:`, error); + return false; + } } - } } diff --git a/src/collaboration/collaboration.controller.ts b/src/collaboration/collaboration.controller.ts index 921da3d2..729a5232 100644 --- a/src/collaboration/collaboration.controller.ts +++ b/src/collaboration/collaboration.controller.ts @@ -1,351 +1,276 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Body, - Param, - UseGuards, - Request, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Body, Param, UseGuards, Request, } from '@nestjs/common'; import { CollaborationService } from './collaboration.service'; import { SharedDocumentService } from './documents/shared-document.service'; import { WhiteboardService } from './whiteboard/whiteboard.service'; import { VersionControlService } from './versioning/version-control.service'; -import { - CollaborationPermissionsService, - PermissionLevel, -} from './permissions/collaboration-permissions.service'; +import { CollaborationPermissionsService, PermissionLevel, } from './permissions/collaboration-permissions.service'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RolesGuard } from '../auth/guards/roles.guard'; - @Controller('collaboration') @UseGuards(JwtAuthGuard, RolesGuard) export class CollaborationController { - constructor( - private readonly collaborationService: CollaborationService, - private readonly sharedDocumentService: SharedDocumentService, - private readonly whiteboardService: WhiteboardService, - private readonly versionControlService: VersionControlService, - private readonly permissionsService: CollaborationPermissionsService, - ) {} - - /** - * Initialize a new collaborative session - */ - @Post('session') - async createSession( - @Request() req, - @Body() body: { sessionId: string; resourceType: 'document' | 'whiteboard' }, - ): Promise { - const { sessionId, resourceType } = body; - const userId = req.user.id; - - const session = await this.collaborationService.initializeSession( - sessionId, - userId, - resourceType, - ); - - return { - success: true, - sessionId, - resourceType, - session, - }; - } - - /** - * Get collaborative document - */ - @Get('document/:id') - async getDocument(@Param('id') documentId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - documentId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access document' }; + constructor(private readonly collaborationService: CollaborationService, private readonly sharedDocumentService: SharedDocumentService, private readonly whiteboardService: WhiteboardService, private readonly versionControlService: VersionControlService, private readonly permissionsService: CollaborationPermissionsService) { } + /** + * Initialize a new collaborative session + */ + @Post('session') + async createSession( + @Request() + req, + @Body() + body: { + sessionId: string; + resourceType: 'document' | 'whiteboard'; + }): Promise { + const { sessionId, resourceType } = body; + const userId = req.user.id; + const session = await this.collaborationService.initializeSession(sessionId, userId, resourceType); + return { + success: true, + sessionId, + resourceType, + session, + }; } - - const document = await this.sharedDocumentService.getDocument(documentId); - - return { - success: true, - document, - }; - } - - /** - * Get collaborative whiteboard - */ - @Get('whiteboard/:id') - async getWhiteboard(@Param('id') whiteboardId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - whiteboardId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access whiteboard' }; + /** + * Get collaborative document + */ + @Get('document/:id') + async getDocument( + @Param('id') + documentId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(documentId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access document' }; + } + const document = await this.sharedDocumentService.getDocument(documentId); + return { + success: true, + document, + }; } - - const whiteboard = await this.whiteboardService.getWhiteboard(whiteboardId); - - return { - success: true, - whiteboard, - }; - } - - /** - * Update collaborative document - */ - @Put('document/:id') - async updateDocument( - @Param('id') documentId: string, - @Request() req, - @Body() body: { operation: any }, - ): Promise { - const { operation } = body; - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - documentId, - userId, - PermissionLevel.WRITE, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to modify document' }; + /** + * Get collaborative whiteboard + */ + @Get('whiteboard/:id') + async getWhiteboard( + @Param('id') + whiteboardId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(whiteboardId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access whiteboard' }; + } + const whiteboard = await this.whiteboardService.getWhiteboard(whiteboardId); + return { + success: true, + whiteboard, + }; } - - const document = await this.sharedDocumentService.applyOperation(documentId, userId, operation); - - return { - success: true, - document, - }; - } - - /** - * Update collaborative whiteboard - */ - @Put('whiteboard/:id') - async updateWhiteboard( - @Param('id') whiteboardId: string, - @Request() req, - @Body() body: { operation: any }, - ): Promise { - const { operation } = body; - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - whiteboardId, - userId, - PermissionLevel.WRITE, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to modify whiteboard' }; + /** + * Update collaborative document + */ + @Put('document/:id') + async updateDocument( + @Param('id') + documentId: string, + @Request() + req, + @Body() + body: { + operation: unknown; + }): Promise { + const { operation } = body; + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(documentId, userId, PermissionLevel.WRITE); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to modify document' }; + } + const document = await this.sharedDocumentService.applyOperation(documentId, userId, operation); + return { + success: true, + document, + }; } - - const whiteboard = await this.whiteboardService.applyOperation(whiteboardId, userId, operation); - - return { - success: true, - whiteboard, - }; - } - - /** - * Get document history - */ - @Get('document/:id/history') - async getDocumentHistory(@Param('id') documentId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - documentId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access history' }; + /** + * Update collaborative whiteboard + */ + @Put('whiteboard/:id') + async updateWhiteboard( + @Param('id') + whiteboardId: string, + @Request() + req, + @Body() + body: { + operation: unknown; + }): Promise { + const { operation } = body; + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(whiteboardId, userId, PermissionLevel.WRITE); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to modify whiteboard' }; + } + const whiteboard = await this.whiteboardService.applyOperation(whiteboardId, userId, operation); + return { + success: true, + whiteboard, + }; } - - const history = await this.sharedDocumentService.getDocumentHistory(documentId); - - return { - success: true, - history, - }; - } - - /** - * Get whiteboard history - */ - @Get('whiteboard/:id/history') - async getWhiteboardHistory(@Param('id') whiteboardId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - whiteboardId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access history' }; + /** + * Get document history + */ + @Get('document/:id/history') + async getDocumentHistory( + @Param('id') + documentId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(documentId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access history' }; + } + const history = await this.sharedDocumentService.getDocumentHistory(documentId); + return { + success: true, + history, + }; } - - const history = await this.whiteboardService.getWhiteboardHistory(whiteboardId); - - return { - success: true, - history, - }; - } - - /** - * Get version history for a session - */ - @Get('version-history/:sessionId') - async getVersionHistory(@Param('sessionId') sessionId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - sessionId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access version history' }; + /** + * Get whiteboard history + */ + @Get('whiteboard/:id/history') + async getWhiteboardHistory( + @Param('id') + whiteboardId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(whiteboardId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access history' }; + } + const history = await this.whiteboardService.getWhiteboardHistory(whiteboardId); + return { + success: true, + history, + }; } - - const history = await this.versionControlService.getVersionHistory(sessionId); - - return { - success: true, - history, - }; - } - - /** - * Get current version of a session - */ - @Get('version-current/:sessionId') - async getCurrentVersion(@Param('sessionId') sessionId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - sessionId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to access current version' }; + /** + * Get version history for a session + */ + @Get('version-history/:sessionId') + async getVersionHistory( + @Param('sessionId') + sessionId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access version history' }; + } + const history = await this.versionControlService.getVersionHistory(sessionId); + return { + success: true, + history, + }; } - - const currentVersion = await this.versionControlService.getCurrentVersion(sessionId); - - return { - success: true, - currentVersion, - }; - } - - /** - * Grant permissions to a user - */ - @Post('permission/:resourceId/user/:userId') - async grantPermission( - @Param('resourceId') resourceId: string, - @Param('userId') userId: string, - @Request() req, - @Body() body: { permission: PermissionLevel }, - ): Promise { - const adminUserId = req.user.id; - const { permission } = body; - - // Check if the requesting user has admin permissions - const isAdmin = await this.permissionsService.hasAccess( - resourceId, - adminUserId, - PermissionLevel.ADMIN, - ); - - if (!isAdmin) { - return { success: false, message: 'Insufficient permissions to grant permissions' }; + /** + * Get current version of a session + */ + @Get('version-current/:sessionId') + async getCurrentVersion( + @Param('sessionId') + sessionId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to access current version' }; + } + const currentVersion = await this.versionControlService.getCurrentVersion(sessionId); + return { + success: true, + currentVersion, + }; } - - const grantedPermission = await this.permissionsService.grantAccess( - resourceId, - userId, - permission, - adminUserId, - ); - - return { - success: true, - permission: grantedPermission, - }; - } - - /** - * Revoke permissions from a user - */ - @Delete('permission/:resourceId/user/:userId') - async revokePermission( - @Param('resourceId') resourceId: string, - @Param('userId') userId: string, - @Request() req, - ): Promise { - const adminUserId = req.user.id; - - // Check if the requesting user has admin permissions - const isAdmin = await this.permissionsService.hasAccess( - resourceId, - adminUserId, - PermissionLevel.ADMIN, - ); - - if (!isAdmin) { - return { success: false, message: 'Insufficient permissions to revoke permissions' }; + /** + * Grant permissions to a user + */ + @Post('permission/:resourceId/user/:userId') + async grantPermission( + @Param('resourceId') + resourceId: string, + @Param('userId') + userId: string, + @Request() + req, + @Body() + body: { + permission: PermissionLevel; + }): Promise { + const adminUserId = req.user.id; + const { permission } = body; + // Check if the requesting user has admin permissions + const isAdmin = await this.permissionsService.hasAccess(resourceId, adminUserId, PermissionLevel.ADMIN); + if (!isAdmin) { + return { success: false, message: 'Insufficient permissions to grant permissions' }; + } + const grantedPermission = await this.permissionsService.grantAccess(resourceId, userId, permission, adminUserId); + return { + success: true, + permission: grantedPermission, + }; } - - const revoked = await this.permissionsService.revokeAccess(resourceId, userId); - - return { - success: true, - revoked, - }; - } - - /** - * Get users with access to a resource - */ - @Get('users/:resourceId') - async getUsersForResource(@Param('resourceId') resourceId: string, @Request() req): Promise { - const userId = req.user.id; - const hasPermission = await this.permissionsService.hasAccess( - resourceId, - userId, - PermissionLevel.READ, - ); - - if (!hasPermission) { - return { success: false, message: 'Insufficient permissions to view resource users' }; + /** + * Revoke permissions from a user + */ + @Delete('permission/:resourceId/user/:userId') + async revokePermission( + @Param('resourceId') + resourceId: string, + @Param('userId') + userId: string, + @Request() + req): Promise { + const adminUserId = req.user.id; + // Check if the requesting user has admin permissions + const isAdmin = await this.permissionsService.hasAccess(resourceId, adminUserId, PermissionLevel.ADMIN); + if (!isAdmin) { + return { success: false, message: 'Insufficient permissions to revoke permissions' }; + } + const revoked = await this.permissionsService.revokeAccess(resourceId, userId); + return { + success: true, + revoked, + }; + } + /** + * Get users with access to a resource + */ + @Get('users/:resourceId') + async getUsersForResource( + @Param('resourceId') + resourceId: string, + @Request() + req): Promise { + const userId = req.user.id; + const hasPermission = await this.permissionsService.hasAccess(resourceId, userId, PermissionLevel.READ); + if (!hasPermission) { + return { success: false, message: 'Insufficient permissions to view resource users' }; + } + const users = await this.permissionsService.getUsersForResource(resourceId); + return { + success: true, + users, + }; } - - const users = await this.permissionsService.getUsersForResource(resourceId); - - return { - success: true, - users, - }; - } } diff --git a/src/collaboration/collaboration.module.ts b/src/collaboration/collaboration.module.ts index 48915c73..f0c319f3 100644 --- a/src/collaboration/collaboration.module.ts +++ b/src/collaboration/collaboration.module.ts @@ -6,24 +6,24 @@ import { CollaborationPermissionsService } from './permissions/collaboration-per import { CollaborationService } from './collaboration.service'; import { CollaborationGateway } from './gateway/collaboration.gateway'; import { CollaborationController } from './collaboration.controller'; - @Module({ - imports: [], - controllers: [CollaborationController], - providers: [ - CollaborationService, - SharedDocumentService, - WhiteboardService, - VersionControlService, - CollaborationPermissionsService, - CollaborationGateway, - ], - exports: [ - CollaborationService, - SharedDocumentService, - WhiteboardService, - VersionControlService, - CollaborationPermissionsService, - ], + imports: [], + controllers: [CollaborationController], + providers: [ + CollaborationService, + SharedDocumentService, + WhiteboardService, + VersionControlService, + CollaborationPermissionsService, + CollaborationGateway, + ], + exports: [ + CollaborationService, + SharedDocumentService, + WhiteboardService, + VersionControlService, + CollaborationPermissionsService, + ], }) -export class CollaborationModule {} +export class CollaborationModule { +} diff --git a/src/collaboration/collaboration.service.ts b/src/collaboration/collaboration.service.ts index 2c7c5acd..66d1582a 100644 --- a/src/collaboration/collaboration.service.ts +++ b/src/collaboration/collaboration.service.ts @@ -3,64 +3,44 @@ import { SharedDocumentService } from './documents/shared-document.service'; import { WhiteboardService } from './whiteboard/whiteboard.service'; import { VersionControlService } from './versioning/version-control.service'; import { CollaborationPermissionsService } from './permissions/collaboration-permissions.service'; - @Injectable() export class CollaborationService { - constructor( - private readonly sharedDocumentService: SharedDocumentService, - private readonly whiteboardService: WhiteboardService, - private readonly versionControlService: VersionControlService, - private readonly permissionsService: CollaborationPermissionsService, - ) {} - - /** - * Initialize a new collaborative session - */ - async initializeSession( - sessionId: string, - userId: string, - resourceType: 'document' | 'whiteboard', - ): Promise { - // Set up initial permissions and session tracking - await this.permissionsService.grantAccess(sessionId, userId); - - if (resourceType === 'document') { - return await this.sharedDocumentService.initializeDocument(sessionId); - } else if (resourceType === 'whiteboard') { - return await this.whiteboardService.initializeWhiteboard(sessionId); + constructor(private readonly sharedDocumentService: SharedDocumentService, private readonly whiteboardService: WhiteboardService, private readonly versionControlService: VersionControlService, private readonly permissionsService: CollaborationPermissionsService) { } + /** + * Initialize a new collaborative session + */ + async initializeSession(sessionId: string, userId: string, resourceType: 'document' | 'whiteboard'): Promise { + // Set up initial permissions and session tracking + await this.permissionsService.grantAccess(sessionId, userId); + if (resourceType === 'document') { + return await this.sharedDocumentService.initializeDocument(sessionId); + } + else if (resourceType === 'whiteboard') { + return await this.whiteboardService.initializeWhiteboard(sessionId); + } + throw new Error(`Unsupported resource type: ${resourceType}`); } - - throw new Error(`Unsupported resource type: ${resourceType}`); - } - - /** - * Handle incoming collaborative changes - */ - async handleCollaborativeChange( - sessionId: string, - userId: string, - operation: any, - resourceType: 'document' | 'whiteboard', - ): Promise { - // Check permissions - const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); - if (!hasPermission) { - throw new Error('User does not have permission to modify this resource'); + /** + * Handle incoming collaborative changes + */ + async handleCollaborativeChange(sessionId: string, userId: string, operation: unknown, resourceType: 'document' | 'whiteboard'): Promise { + // Check permissions + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); + if (!hasPermission) { + throw new Error('User does not have permission to modify this resource'); + } + if (resourceType === 'document') { + return await this.sharedDocumentService.applyOperation(sessionId, userId, operation); + } + else if (resourceType === 'whiteboard') { + return await this.whiteboardService.applyOperation(sessionId, userId, operation); + } + throw new Error(`Unsupported resource type: ${resourceType}`); } - - if (resourceType === 'document') { - return await this.sharedDocumentService.applyOperation(sessionId, userId, operation); - } else if (resourceType === 'whiteboard') { - return await this.whiteboardService.applyOperation(sessionId, userId, operation); + /** + * Track changes for version control + */ + async trackChange(sessionId: string, userId: string, change: unknown): Promise { + return await this.versionControlService.recordChange(sessionId, userId, change); } - - throw new Error(`Unsupported resource type: ${resourceType}`); - } - - /** - * Track changes for version control - */ - async trackChange(sessionId: string, userId: string, change: any): Promise { - return await this.versionControlService.recordChange(sessionId, userId, change); - } } diff --git a/src/collaboration/constants/collaboration-events.constants.ts b/src/collaboration/constants/collaboration-events.constants.ts index 99586236..ed4a1ce4 100644 --- a/src/collaboration/constants/collaboration-events.constants.ts +++ b/src/collaboration/constants/collaboration-events.constants.ts @@ -1,25 +1,22 @@ export const COLLABORATION_EVENTS = { - // Inbound WebSocket messages - JOIN_SESSION: 'join-session', - COLLABORATIVE_OPERATION: 'collaborative-operation', - REQUEST_SYNC: 'request-sync', - RESOLVE_CONFLICT: 'resolve-conflict', - - // Outbound WebSocket messages - USER_JOINED: 'user-joined', - SESSION_STATE: 'session-state', - OPERATION_APPLIED: 'operation-applied', - FULL_SYNC: 'full-sync', - CONFLICT_RESOLVED: 'conflict-resolved', + // Inbound WebSocket messages + JOIN_SESSION: 'join-session', + COLLABORATIVE_OPERATION: 'collaborative-operation', + REQUEST_SYNC: 'request-sync', + RESOLVE_CONFLICT: 'resolve-conflict', + // Outbound WebSocket messages + USER_JOINED: 'user-joined', + SESSION_STATE: 'session-state', + OPERATION_APPLIED: 'operation-applied', + FULL_SYNC: 'full-sync', + CONFLICT_RESOLVED: 'conflict-resolved', } as const; - export const NOTIFICATION_GATEWAY_EVENTS = { - SUBSCRIBE: 'subscribe', - NOTIFICATION: 'notification', - BROADCAST_NOTIFICATION: 'broadcast_notification', - SUBSCRIBE_NOTIFICATIONS: 'subscribe_notifications', + SUBSCRIBE: 'subscribe', + NOTIFICATION: 'notification', + BROADCAST_NOTIFICATION: 'broadcast_notification', + SUBSCRIBE_NOTIFICATIONS: 'subscribe_notifications', } as const; - export const MESSAGING_GATEWAY_EVENTS = { - SEND_MESSAGE: 'send_message', + SEND_MESSAGE: 'send_message', } as const; diff --git a/src/collaboration/documents/shared-document.service.ts b/src/collaboration/documents/shared-document.service.ts index d81665c8..839bc52a 100644 --- a/src/collaboration/documents/shared-document.service.ts +++ b/src/collaboration/documents/shared-document.service.ts @@ -1,255 +1,196 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface DocumentOperation { - id: string; - type: 'insert' | 'delete' | 'update'; - position: number; - content?: string; - length?: number; - timestamp: number; - userId: string; + id: string; + type: 'insert' | 'delete' | 'update'; + position: number; + content?: string; + length?: number; + timestamp: number; + userId: string; } - export interface CollaborativeDocument { - id: string; - content: string; - operations: DocumentOperation[]; - collaborators: string[]; - createdAt: Date; - updatedAt: Date; + id: string; + content: string; + operations: DocumentOperation[]; + collaborators: string[]; + createdAt: Date; + updatedAt: Date; } - @Injectable() export class SharedDocumentService { - private readonly logger = Logger; - private documents: Map = new Map(); - - /** - * Initialize a new collaborative document - */ - async initializeDocument(documentId: string): Promise { - const document: CollaborativeDocument = { - id: documentId, - content: '', - operations: [], - collaborators: [], - createdAt: new Date(), - updatedAt: new Date(), - }; - - this.documents.set(documentId, document); - this.logger.log(`Initialized document ${documentId}`); - - return document; - } - - /** - * Get a collaborative document - */ - async getDocument(documentId: string): Promise { - return this.documents.get(documentId) || null; - } - - /** - * Apply an operation to a document using operational transformation - */ - async applyOperation( - documentId: string, - userId: string, - operation: Omit, - ): Promise { - const document = this.documents.get(documentId); - if (!document) { - throw new Error(`Document ${documentId} not found`); + private readonly logger = Logger; + private documents: Map = new Map(); + /** + * Initialize a new collaborative document + */ + async initializeDocument(documentId: string): Promise { + const document: CollaborativeDocument = { + id: documentId, + content: '', + operations: [], + collaborators: [], + createdAt: new Date(), + updatedAt: new Date(), + }; + this.documents.set(documentId, document); + this.logger.log(`Initialized document ${documentId}`); + return document; } - - // Add metadata to the operation - const opWithMetadata: DocumentOperation = { - ...operation, - id: uuidv4(), - timestamp: Date.now(), - userId, - }; - - // Transform the operation against concurrent operations - const transformedOp = this.transformOperation(opWithMetadata, document.operations); - - // Apply the transformed operation to the document content - document.content = this.applyOperationToContent(document.content, transformedOp); - - // Add the operation to the document's operation history - document.operations.push(transformedOp); - document.updatedAt = new Date(); - - // Add user to collaborators if not already present - if (!document.collaborators.includes(userId)) { - document.collaborators.push(userId); + /** + * Get a collaborative document + */ + async getDocument(documentId: string): Promise { + return this.documents.get(documentId) || null; } - - this.logger.log(`Applied operation ${transformedOp.id} to document ${documentId}`); - - return document; - } - - /** - * Transform an operation against a list of concurrent operations - */ - private transformOperation( - operation: DocumentOperation, - concurrentOperations: DocumentOperation[], - ): DocumentOperation { - let transformedOp = { ...operation }; - - for (const concurrentOp of concurrentOperations) { - // Only transform if operations affect overlapping positions - if (this.operationsOverlap(transformedOp, concurrentOp)) { - transformedOp = this.transformSingleOperation(transformedOp, concurrentOp); - } - } - - return transformedOp; - } - - /** - * Check if two operations overlap in their effect on document content - */ - private operationsOverlap(op1: DocumentOperation, op2: DocumentOperation): boolean { - // Operations don't overlap if one happens before the other ends - if (op1.type === 'insert' && op2.type === 'insert') { - // Two inserts at the same position need transformation - return op1.position === op2.position; - } - - if (op1.type === 'insert') { - // Insert and another operation overlap if insert position is within or adjacent to other operation - return ( - op2.position <= op1.position && - op1.position <= op2.position + (op2.length || op2.content?.length || 0) - ); - } - - if (op2.type === 'insert') { - // Same as above but reversed - return ( - op1.position <= op2.position && - op2.position <= op1.position + (op1.length || op1.content?.length || 0) - ); + /** + * Apply an operation to a document using operational transformation + */ + async applyOperation(documentId: string, userId: string, operation: Omit): Promise { + const document = this.documents.get(documentId); + if (!document) { + throw new Error(`Document ${documentId} not found`); + } + // Add metadata to the operation + const opWithMetadata: DocumentOperation = { + ...operation, + id: uuidv4(), + timestamp: Date.now(), + userId, + }; + // Transform the operation against concurrent operations + const transformedOp = this.transformOperation(opWithMetadata, document.operations); + // Apply the transformed operation to the document content + document.content = this.applyOperationToContent(document.content, transformedOp); + // Add the operation to the document's operation history + document.operations.push(transformedOp); + document.updatedAt = new Date(); + // Add user to collaborators if not already present + if (!document.collaborators.includes(userId)) { + document.collaborators.push(userId); + } + this.logger.log(`Applied operation ${transformedOp.id} to document ${documentId}`); + return document; } - - // Both are delete/update operations - overlap if ranges intersect - const op1End = op1.position + (op1.length || op1.content?.length || 0); - const op2End = op2.position + (op2.length || op2.content?.length || 0); - return !(op1.position >= op2End || op2.position >= op1End); - } - - /** - * Transform a single operation against a concurrent operation - */ - private transformSingleOperation( - operation: DocumentOperation, - concurrentOp: DocumentOperation, - ): DocumentOperation { - const transformedOp = { ...operation }; - - // Adjust positions based on concurrent operations - if (concurrentOp.type === 'insert' && operation.position >= concurrentOp.position) { - // If concurrent operation inserted text before our operation, adjust position - transformedOp.position += concurrentOp.content ? concurrentOp.content.length : 0; - } else if (concurrentOp.type === 'delete') { - const concurrentEnd = concurrentOp.position + concurrentOp.length; - - if (operation.position >= concurrentEnd) { - // Operation is after the deleted range, adjust position - transformedOp.position -= concurrentOp.length; - } else if (operation.position > concurrentOp.position) { - // Operation starts within the deleted range, clamp to start of deletion - transformedOp.position = concurrentOp.position; - } - // If operation starts before the deletion, position stays the same + /** + * Transform an operation against a list of concurrent operations + */ + private transformOperation(operation: DocumentOperation, concurrentOperations: DocumentOperation[]): DocumentOperation { + let transformedOp = { ...operation }; + for (const concurrentOp of concurrentOperations) { + // Only transform if operations affect overlapping positions + if (this.operationsOverlap(transformedOp, concurrentOp)) { + transformedOp = this.transformSingleOperation(transformedOp, concurrentOp); + } + } + return transformedOp; } - - return transformedOp; - } - - /** - * Apply an operation to document content - */ - private applyOperationToContent(content: string, operation: DocumentOperation): string { - switch (operation.type) { - case 'insert': - if (operation.content !== undefined) { - return ( - content.slice(0, operation.position) + - operation.content + - content.slice(operation.position) - ); + /** + * Check if two operations overlap in their effect on document content + */ + private operationsOverlap(op1: DocumentOperation, op2: DocumentOperation): boolean { + // Operations don't overlap if one happens before the other ends + if (op1.type === 'insert' && op2.type === 'insert') { + // Two inserts at the same position need transformation + return op1.position === op2.position; } - return content; - - case 'delete': - if (operation.length !== undefined) { - return ( - content.slice(0, operation.position) + - content.slice(operation.position + operation.length) - ); + if (op1.type === 'insert') { + // Insert and another operation overlap if insert position is within or adjacent to other operation + return (op2.position <= op1.position && + op1.position <= op2.position + (op2.length || op2.content?.length || 0)); } - return content; - - case 'update': - if (operation.content !== undefined && operation.length !== undefined) { - return ( - content.slice(0, operation.position) + - operation.content + - content.slice(operation.position + operation.length) - ); + if (op2.type === 'insert') { + // Same as above but reversed + return (op1.position <= op2.position && + op2.position <= op1.position + (op1.length || op1.content?.length || 0)); } - return content; - - default: - return content; + // Both are delete/update operations - overlap if ranges intersect + const op1End = op1.position + (op1.length || op1.content?.length || 0); + const op2End = op2.position + (op2.length || op2.content?.length || 0); + return !(op1.position >= op2End || op2.position >= op1End); } - } - - /** - * Resolve conflicts between simultaneous edits - */ - async resolveConflicts( - documentId: string, - operations: DocumentOperation[], - ): Promise { - const document = this.documents.get(documentId); - if (!document) { - throw new Error(`Document ${documentId} not found`); + /** + * Transform a single operation against a concurrent operation + */ + private transformSingleOperation(operation: DocumentOperation, concurrentOp: DocumentOperation): DocumentOperation { + const transformedOp = { ...operation }; + // Adjust positions based on concurrent operations + if (concurrentOp.type === 'insert' && operation.position >= concurrentOp.position) { + // If concurrent operation inserted text before our operation, adjust position + transformedOp.position += concurrentOp.content ? concurrentOp.content.length : 0; + } + else if (concurrentOp.type === 'delete') { + const concurrentEnd = concurrentOp.position + concurrentOp.length; + if (operation.position >= concurrentEnd) { + // Operation is after the deleted range, adjust position + transformedOp.position -= concurrentOp.length; + } + else if (operation.position > concurrentOp.position) { + // Operation starts within the deleted range, clamp to start of deletion + transformedOp.position = concurrentOp.position; + } + // If operation starts before the deletion, position stays the same + } + return transformedOp; + } + /** + * Apply an operation to document content + */ + private applyOperationToContent(content: string, operation: DocumentOperation): string { + switch (operation.type) { + case 'insert': + if (operation.content !== undefined) { + return (content.slice(0, operation.position) + + operation.content + + content.slice(operation.position)); + } + return content; + case 'delete': + if (operation.length !== undefined) { + return (content.slice(0, operation.position) + + content.slice(operation.position + operation.length)); + } + return content; + case 'update': + if (operation.content !== undefined && operation.length !== undefined) { + return (content.slice(0, operation.position) + + operation.content + + content.slice(operation.position + operation.length)); + } + return content; + default: + return content; + } } - - // Sort operations by timestamp to process in chronological order - const sortedOps = [...operations].sort((a, b) => a.timestamp - b.timestamp); - - // Clear existing operations and reapply in order to resolve conflicts - document.operations = []; - document.content = ''; - - // Rebuild document content from the sorted operations - for (const operation of sortedOps) { - document.content = this.applyOperationToContent(document.content, operation); - document.operations.push(operation); + /** + * Resolve conflicts between simultaneous edits + */ + async resolveConflicts(documentId: string, operations: DocumentOperation[]): Promise { + const document = this.documents.get(documentId); + if (!document) { + throw new Error(`Document ${documentId} not found`); + } + // Sort operations by timestamp to process in chronological order + const sortedOps = [...operations].sort((a, b) => a.timestamp - b.timestamp); + // Clear existing operations and reapply in order to resolve conflicts + document.operations = []; + document.content = ''; + // Rebuild document content from the sorted operations + for (const operation of sortedOps) { + document.content = this.applyOperationToContent(document.content, operation); + document.operations.push(operation); + } + document.updatedAt = new Date(); + return document; } - - document.updatedAt = new Date(); - - return document; - } - - /** - * Get document history - */ - async getDocumentHistory(documentId: string): Promise { - const document = this.documents.get(documentId); - if (!document) { - throw new Error(`Document ${documentId} not found`); + /** + * Get document history + */ + async getDocumentHistory(documentId: string): Promise { + const document = this.documents.get(documentId); + if (!document) { + throw new Error(`Document ${documentId} not found`); + } + return [...document.operations]; } - - return [...document.operations]; - } } diff --git a/src/collaboration/gateway/collaboration.gateway.ts b/src/collaboration/gateway/collaboration.gateway.ts index d99580b8..e583d7a3 100644 --- a/src/collaboration/gateway/collaboration.gateway.ts +++ b/src/collaboration/gateway/collaboration.gateway.ts @@ -1,13 +1,4 @@ -import { - WebSocketGateway, - WebSocketServer, - SubscribeMessage, - OnGatewayInit, - OnGatewayConnection, - OnGatewayDisconnect, - MessageBody, - ConnectedSocket, -} from '@nestjs/websockets'; +import { WebSocketGateway, WebSocketServer, SubscribeMessage, OnGatewayInit, OnGatewayConnection, OnGatewayDisconnect, MessageBody, ConnectedSocket, } from '@nestjs/websockets'; import { COLLABORATION_EVENTS } from '../constants/collaboration-events.constants'; import { Server, Socket } from 'socket.io'; import { Logger, UseGuards } from '@nestjs/common'; @@ -16,253 +7,220 @@ import { WsThrottlerGuard } from '../../common/guards/ws-throttler.guard'; import { SharedDocumentService } from '../documents/shared-document.service'; import { WhiteboardService } from '../whiteboard/whiteboard.service'; import { VersionControlService } from '../versioning/version-control.service'; -import { - CollaborationPermissionsService, - PermissionLevel, -} from '../permissions/collaboration-permissions.service'; +import { CollaborationPermissionsService, PermissionLevel, } from '../permissions/collaboration-permissions.service'; import { wsManager } from '../../common/utils/websocket.utils'; - export interface CollaborativeOperation { - sessionId: string; - userId: string; - resourceType: 'document' | 'whiteboard'; - operation: any; - timestamp: number; + sessionId: string; + userId: string; + resourceType: 'document' | 'whiteboard'; + operation: unknown; + timestamp: number; } - @WebSocketGateway({ - cors: { - origin: '*', - methods: ['GET', 'POST'], - credentials: true, - }, + cors: { + origin: '*', + methods: ['GET', 'POST'], + credentials: true, + }, }) @UseGuards(WsThrottlerGuard) -export class CollaborationGateway - implements OnGatewayInit, OnGatewayConnection, OnGatewayDisconnect -{ - @WebSocketServer() server: Server; - private logger: Logger = new Logger('CollaborationGateway'); - - constructor( - private readonly collaborationService: CollaborationService, - private readonly sharedDocumentService: SharedDocumentService, - private readonly whiteboardService: WhiteboardService, - private readonly versionControlService: VersionControlService, - private readonly permissionsService: CollaborationPermissionsService, - ) {} - - afterInit(_server: Server): void { - this.logger.log('Collaboration Gateway initialized'); - } - - async handleConnection(_server: any, @ConnectedSocket() client: Socket): Promise { - if (wsManager.getTotalConnections() >= 5000) { - client.emit('error', { message: 'Server is at maximum capacity' }); - client.disconnect(true); - return; +export class CollaborationGateway implements OnGatewayInit, OnGatewayConnection, OnGatewayDisconnect { + @WebSocketServer() + server: Server; + private logger: Logger = new Logger('CollaborationGateway'); + constructor(private readonly collaborationService: CollaborationService, private readonly sharedDocumentService: SharedDocumentService, private readonly whiteboardService: WhiteboardService, private readonly versionControlService: VersionControlService, private readonly permissionsService: CollaborationPermissionsService) { } + afterInit(_server: Server): void { + this.logger.log('Collaboration Gateway initialized'); } - this.logger.log(`Client connected: ${client.id}`); - - // Optionally authenticate the user here based on token - // const token = client.handshake.auth.token; - // const user = await this.authService.validateToken(token); - } - - async handleDisconnect(@ConnectedSocket() client: Socket): Promise { - wsManager.cleanupSocket(client); - this.logger.log(`Client disconnected: ${client.id}`); - - // Clean up any session associations for this client - // Remove client from any active sessions - } - - @SubscribeMessage(COLLABORATION_EVENTS.JOIN_SESSION) - async handleJoinSession( - @MessageBody() data: { sessionId: string; userId: string; resourceType: string }, - @ConnectedSocket() client: Socket, - ): Promise { - const { sessionId, userId, resourceType } = data; - - try { - // Check permissions - const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); - if (!hasPermission) { - client.emit('error', { message: 'Insufficient permissions to join session' }); - return; - } - - const registered = wsManager.registerConnection(userId, client); - if (!registered) { - client.emit('error', { message: 'Connection limit reached' }); - return; - } - - // Join the room - client.join(sessionId); - - // Initialize or get the resource - let resource: any; - if (resourceType === 'document') { - resource = await this.sharedDocumentService.getDocument(sessionId); - if (!resource) { - resource = await this.collaborationService.initializeSession( - sessionId, - userId, - 'document' as any, - ); + async handleConnection(_server: unknown, + @ConnectedSocket() + client: Socket): Promise { + if (wsManager.getTotalConnections() >= 5000) { + client.emit('error', { message: 'Server is at maximum capacity' }); + client.disconnect(true); + return; } - } else if (resourceType === 'whiteboard') { - resource = await this.whiteboardService.getWhiteboard(sessionId); - if (!resource) { - resource = await this.collaborationService.initializeSession( - sessionId, - userId, - 'whiteboard' as any, - ); + this.logger.log(`Client connected: ${client.id}`); + // Optionally authenticate the user here based on token + // const token = client.handshake.auth.token; + // const user = await this.authService.validateToken(token); + } + async handleDisconnect( + @ConnectedSocket() + client: Socket): Promise { + wsManager.cleanupSocket(client); + this.logger.log(`Client disconnected: ${client.id}`); + // Clean up any session associations for this client + // Remove client from any active sessions + } + @SubscribeMessage(COLLABORATION_EVENTS.JOIN_SESSION) + async handleJoinSession( + @MessageBody() + data: { + sessionId: string; + userId: string; + resourceType: string; + }, + @ConnectedSocket() + client: Socket): Promise { + const { sessionId, userId, resourceType } = data; + try { + // Check permissions + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); + if (!hasPermission) { + client.emit('error', { message: 'Insufficient permissions to join session' }); + return; + } + const registered = wsManager.registerConnection(userId, client); + if (!registered) { + client.emit('error', { message: 'Connection limit reached' }); + return; + } + // Join the room + client.join(sessionId); + // Initialize or get the resource + let resource: unknown; + if (resourceType === 'document') { + resource = await this.sharedDocumentService.getDocument(sessionId); + if (!resource) { + resource = await this.collaborationService.initializeSession(sessionId, userId, 'document' as unknown); + } + } + else if (resourceType === 'whiteboard') { + resource = await this.whiteboardService.getWhiteboard(sessionId); + if (!resource) { + resource = await this.collaborationService.initializeSession(sessionId, userId, 'whiteboard' as unknown); + } + } + // Notify other users in the session + client.to(sessionId).emit(COLLABORATION_EVENTS.USER_JOINED, { userId, sessionId }); + // Send current resource state to the joining user + client.emit(COLLABORATION_EVENTS.SESSION_STATE, { + sessionId, + resourceType, + resource, + collaborators: await this.permissionsService.getUsersForResource(sessionId), + }); + this.logger.log(`User ${userId} joined session ${sessionId}`); + } + catch (error) { + this.logger.error(`Error joining session: ${error.message}`); + client.emit('error', { message: error.message }); } - } - - // Notify other users in the session - client.to(sessionId).emit(COLLABORATION_EVENTS.USER_JOINED, { userId, sessionId }); - - // Send current resource state to the joining user - client.emit(COLLABORATION_EVENTS.SESSION_STATE, { - sessionId, - resourceType, - resource, - collaborators: await this.permissionsService.getUsersForResource(sessionId), - }); - - this.logger.log(`User ${userId} joined session ${sessionId}`); - } catch (error) { - this.logger.error(`Error joining session: ${error.message}`); - client.emit('error', { message: error.message }); } - } - - @SubscribeMessage(COLLABORATION_EVENTS.COLLABORATIVE_OPERATION) - async handleCollaborativeOperation( - @MessageBody() operation: CollaborativeOperation, - @ConnectedSocket() client: Socket, - ): Promise { - const { sessionId, userId, resourceType, operation: opData } = operation; - - try { - // Validate permissions - const hasPermission = await this.permissionsService.hasAccess( - sessionId, - userId, - resourceType === 'document' ? PermissionLevel.WRITE : PermissionLevel.WRITE, - ); - - if (!hasPermission) { - client.emit('error', { message: 'Insufficient permissions to perform operation' }); - return; - } - - // Process the operation based on resource type - let result: any; - if (resourceType === 'document') { - result = await this.sharedDocumentService.applyOperation(sessionId, userId, opData); - } else if (resourceType === 'whiteboard') { - result = await this.whiteboardService.applyOperation(sessionId, userId, opData); - } - - // Record the change for version control - await this.collaborationService.trackChange(sessionId, userId, { - operation: opData, - resourceType, - result, - }); - - // Broadcast the operation to all other clients in the session - client.to(sessionId).emit(COLLABORATION_EVENTS.OPERATION_APPLIED, { - operation: opData, - userId, - timestamp: Date.now(), - result, - }); - - this.logger.log(`Operation applied in session ${sessionId} by user ${userId}`); - } catch (error) { - this.logger.error(`Error applying operation: ${error.message}`); - client.emit('error', { message: error.message }); + @SubscribeMessage(COLLABORATION_EVENTS.COLLABORATIVE_OPERATION) + async handleCollaborativeOperation( + @MessageBody() + operation: CollaborativeOperation, + @ConnectedSocket() + client: Socket): Promise { + const { sessionId, userId, resourceType, operation: opData } = operation; + try { + // Validate permissions + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId, resourceType === 'document' ? PermissionLevel.WRITE : PermissionLevel.WRITE); + if (!hasPermission) { + client.emit('error', { message: 'Insufficient permissions to perform operation' }); + return; + } + // Process the operation based on resource type + let result: unknown; + if (resourceType === 'document') { + result = await this.sharedDocumentService.applyOperation(sessionId, userId, opData); + } + else if (resourceType === 'whiteboard') { + result = await this.whiteboardService.applyOperation(sessionId, userId, opData); + } + // Record the change for version control + await this.collaborationService.trackChange(sessionId, userId, { + operation: opData, + resourceType, + result, + }); + // Broadcast the operation to all other clients in the session + client.to(sessionId).emit(COLLABORATION_EVENTS.OPERATION_APPLIED, { + operation: opData, + userId, + timestamp: Date.now(), + result, + }); + this.logger.log(`Operation applied in session ${sessionId} by user ${userId}`); + } + catch (error) { + this.logger.error(`Error applying operation: ${error.message}`); + client.emit('error', { message: error.message }); + } } - } - - @SubscribeMessage(COLLABORATION_EVENTS.REQUEST_SYNC) - async handleSyncRequest( - @MessageBody() data: { sessionId: string; userId: string }, - @ConnectedSocket() client: Socket, - ): Promise { - const { sessionId, userId } = data; - - try { - const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); - if (!hasPermission) { - client.emit('error', { message: 'Insufficient permissions to sync' }); - return; - } - - // Send current state to requesting client - const document = await this.sharedDocumentService.getDocument(sessionId); - const whiteboard = await this.whiteboardService.getWhiteboard(sessionId); - - client.emit(COLLABORATION_EVENTS.FULL_SYNC, { - sessionId, - document: document || null, - whiteboard: whiteboard || null, - }); - } catch (error) { - this.logger.error(`Error syncing state: ${error.message}`); - client.emit('error', { message: error.message }); + @SubscribeMessage(COLLABORATION_EVENTS.REQUEST_SYNC) + async handleSyncRequest( + @MessageBody() + data: { + sessionId: string; + userId: string; + }, + @ConnectedSocket() + client: Socket): Promise { + const { sessionId, userId } = data; + try { + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId); + if (!hasPermission) { + client.emit('error', { message: 'Insufficient permissions to sync' }); + return; + } + // Send current state to requesting client + const document = await this.sharedDocumentService.getDocument(sessionId); + const whiteboard = await this.whiteboardService.getWhiteboard(sessionId); + client.emit(COLLABORATION_EVENTS.FULL_SYNC, { + sessionId, + document: document || null, + whiteboard: whiteboard || null, + }); + } + catch (error) { + this.logger.error(`Error syncing state: ${error.message}`); + client.emit('error', { message: error.message }); + } } - } - - @SubscribeMessage(COLLABORATION_EVENTS.RESOLVE_CONFLICT) - async handleConflictResolution( + @SubscribeMessage(COLLABORATION_EVENTS.RESOLVE_CONFLICT) + async handleConflictResolution( @MessageBody() - data: { sessionId: string; userId: string; resourceType: string; operations: any[] }, - @ConnectedSocket() client: Socket, - ): Promise { - const { sessionId, userId, resourceType, operations } = data; - - try { - // Only admins/owners can resolve conflicts - const hasPermission = await this.permissionsService.hasAccess( - sessionId, - userId, - PermissionLevel.ADMIN, - ); - if (!hasPermission) { - client.emit('error', { message: 'Insufficient permissions to resolve conflicts' }); - return; - } - - let result: any; - if (resourceType === 'document') { - result = await this.sharedDocumentService.resolveConflicts(sessionId, operations); - } else if (resourceType === 'whiteboard') { - result = await this.whiteboardService.resolveConflicts(sessionId, operations); - } - - // Broadcast resolved state to all clients - this.server.to(sessionId).emit(COLLABORATION_EVENTS.CONFLICT_RESOLVED, { - sessionId, - resourceType, - resolvedState: result, - }); - - this.logger.log(`Conflict resolved in session ${sessionId}`); - } catch (error) { - this.logger.error(`Error resolving conflict: ${error.message}`); - client.emit('error', { message: error.message }); + data: { + sessionId: string; + userId: string; + resourceType: string; + operations: unknown[]; + }, + @ConnectedSocket() + client: Socket): Promise { + const { sessionId, userId, resourceType, operations } = data; + try { + // Only admins/owners can resolve conflicts + const hasPermission = await this.permissionsService.hasAccess(sessionId, userId, PermissionLevel.ADMIN); + if (!hasPermission) { + client.emit('error', { message: 'Insufficient permissions to resolve conflicts' }); + return; + } + let result: unknown; + if (resourceType === 'document') { + result = await this.sharedDocumentService.resolveConflicts(sessionId, operations); + } + else if (resourceType === 'whiteboard') { + result = await this.whiteboardService.resolveConflicts(sessionId, operations); + } + // Broadcast resolved state to all clients + this.server.to(sessionId).emit(COLLABORATION_EVENTS.CONFLICT_RESOLVED, { + sessionId, + resourceType, + resolvedState: result, + }); + this.logger.log(`Conflict resolved in session ${sessionId}`); + } + catch (error) { + this.logger.error(`Error resolving conflict: ${error.message}`); + client.emit('error', { message: error.message }); + } + } + // Method to broadcast to a specific session + broadcastToSession(sessionId: string, event: string, data: unknown): void { + this.server.to(sessionId).emit(event, data); } - } - - // Method to broadcast to a specific session - broadcastToSession(sessionId: string, event: string, data: any): void { - this.server.to(sessionId).emit(event, data); - } } diff --git a/src/collaboration/permissions/collaboration-permissions.service.ts b/src/collaboration/permissions/collaboration-permissions.service.ts index 53b9f63a..58192f31 100644 --- a/src/collaboration/permissions/collaboration-permissions.service.ts +++ b/src/collaboration/permissions/collaboration-permissions.service.ts @@ -1,298 +1,215 @@ import { Injectable, Logger } from '@nestjs/common'; - export enum PermissionLevel { - READ = 'read', - WRITE = 'write', - ADMIN = 'admin', - OWNER = 'owner', + READ = 'read', + WRITE = 'write', + ADMIN = 'admin', + OWNER = 'owner' } - export interface ResourcePermission { - resourceId: string; - userId: string; - permission: PermissionLevel; - grantedAt: Date; - grantedBy: string; + resourceId: string; + userId: string; + permission: PermissionLevel; + grantedAt: Date; + grantedBy: string; } - export interface CollaborativeResource { - id: string; - type: 'document' | 'whiteboard' | 'project'; - ownerId: string; - permissions: ResourcePermission[]; - createdAt: Date; - updatedAt: Date; + id: string; + type: 'document' | 'whiteboard' | 'project'; + ownerId: string; + permissions: ResourcePermission[]; + createdAt: Date; + updatedAt: Date; } - @Injectable() export class CollaborationPermissionsService { - private readonly logger = Logger; - private resources: Map = new Map(); - private defaultPermission: PermissionLevel = PermissionLevel.READ; - - /** - * Grant permission to a user for a resource - */ - async grantAccess( - resourceId: string, - userId: string, - permission: PermissionLevel = PermissionLevel.WRITE, - grantedBy: string = 'system', - ): Promise { - let resource = this.resources.get(resourceId); - - if (!resource) { - resource = { - id: resourceId, - type: this.getResourceType(resourceId), - ownerId: userId, - permissions: [], - createdAt: new Date(), - updatedAt: new Date(), - }; - this.resources.set(resourceId, resource); + private readonly logger = Logger; + private resources: Map = new Map(); + private defaultPermission: PermissionLevel = PermissionLevel.READ; + /** + * Grant permission to a user for a resource + */ + async grantAccess(resourceId: string, userId: string, permission: PermissionLevel = PermissionLevel.WRITE, grantedBy: string = 'system'): Promise { + let resource = this.resources.get(resourceId); + if (!resource) { + resource = { + id: resourceId, + type: this.getResourceType(resourceId), + ownerId: userId, + permissions: [], + createdAt: new Date(), + updatedAt: new Date(), + }; + this.resources.set(resourceId, resource); + } + // Check if user already has permission + const existingPermission = resource.permissions.find((p) => p.userId === userId); + if (existingPermission) { + // Update existing permission + existingPermission.permission = permission; + existingPermission.grantedAt = new Date(); + existingPermission.grantedBy = grantedBy; + } + else { + // Add new permission + const newPermission: ResourcePermission = { + resourceId, + userId, + permission, + grantedAt: new Date(), + grantedBy, + }; + resource.permissions.push(newPermission); + } + resource.updatedAt = new Date(); + this.logger.log(`Granted ${permission} permission to user ${userId} for resource ${resourceId}`); + return this.getUserPermission(resourceId, userId) ?? null; } - - // Check if user already has permission - const existingPermission = resource.permissions.find((p) => p.userId === userId); - - if (existingPermission) { - // Update existing permission - existingPermission.permission = permission; - existingPermission.grantedAt = new Date(); - existingPermission.grantedBy = grantedBy; - } else { - // Add new permission - const newPermission: ResourcePermission = { - resourceId, - userId, - permission, - grantedAt: new Date(), - grantedBy, - }; - resource.permissions.push(newPermission); + /** + * Revoke permission from a user for a resource + */ + async revokeAccess(resourceId: string, userId: string): Promise { + const resource = this.resources.get(resourceId); + if (!resource) { + return false; + } + const initialLength = resource.permissions.length; + resource.permissions = resource.permissions.filter((p) => p.userId !== userId); + if (resource.permissions.length !== initialLength) { + resource.updatedAt = new Date(); + this.logger.log(`Revoked permission for user ${userId} from resource ${resourceId}`); + return true; + } + return false; } - - resource.updatedAt = new Date(); - - this.logger.log( - `Granted ${permission} permission to user ${userId} for resource ${resourceId}`, - ); - - return this.getUserPermission(resourceId, userId) ?? null; - } - - /** - * Revoke permission from a user for a resource - */ - async revokeAccess(resourceId: string, userId: string): Promise { - const resource = this.resources.get(resourceId); - if (!resource) { - return false; + /** + * Check if a user has access to a resource + */ + async hasAccess(resourceId: string, userId: string, requiredPermission: PermissionLevel = PermissionLevel.READ): Promise { + const userPermission = this.getUserPermission(resourceId, userId); + if (!userPermission) { + return false; + } + return this.checkPermissionLevel(userPermission.permission, requiredPermission); } - - const initialLength = resource.permissions.length; - resource.permissions = resource.permissions.filter((p) => p.userId !== userId); - - if (resource.permissions.length !== initialLength) { - resource.updatedAt = new Date(); - this.logger.log(`Revoked permission for user ${userId} from resource ${resourceId}`); - return true; + /** + * Get user's permission for a resource + */ + getUserPermission(resourceId: string, userId: string): ResourcePermission | undefined { + const resource = this.resources.get(resourceId); + if (!resource) { + return undefined; + } + return resource.permissions.find((p) => p.userId === userId); } - - return false; - } - - /** - * Check if a user has access to a resource - */ - async hasAccess( - resourceId: string, - userId: string, - requiredPermission: PermissionLevel = PermissionLevel.READ, - ): Promise { - const userPermission = this.getUserPermission(resourceId, userId); - - if (!userPermission) { - return false; + /** + * Get all users with access to a resource + */ + async getUsersForResource(resourceId: string): Promise { + const resource = this.resources.get(resourceId); + if (!resource) { + return []; + } + return [...resource.permissions]; } - - return this.checkPermissionLevel(userPermission.permission, requiredPermission); - } - - /** - * Get user's permission for a resource - */ - getUserPermission(resourceId: string, userId: string): ResourcePermission | undefined { - const resource = this.resources.get(resourceId); - if (!resource) { - return undefined; + /** + * Get all resources a user has access to + */ + async getResourcesForUser(userId: string, minPermission: PermissionLevel = PermissionLevel.READ): Promise { + const userResources: CollaborativeResource[] = []; + for (const resource of this.resources.values()) { + const userPermission = resource.permissions.find((p) => p.userId === userId); + if (userPermission && this.checkPermissionLevel(userPermission.permission, minPermission)) { + userResources.push({ ...resource }); + } + } + return userResources; } - - return resource.permissions.find((p) => p.userId === userId); - } - - /** - * Get all users with access to a resource - */ - async getUsersForResource(resourceId: string): Promise { - const resource = this.resources.get(resourceId); - if (!resource) { - return []; + /** + * Update user's permission level + */ + async updatePermission(resourceId: string, userId: string, newPermission: PermissionLevel, updatedBy: string): Promise { + const resource = this.resources.get(resourceId); + if (!resource) { + return null; + } + const userPermission = resource.permissions.find((p) => p.userId === userId); + if (!userPermission) { + return null; + } + userPermission.permission = newPermission; + userPermission.grantedAt = new Date(); + userPermission.grantedBy = updatedBy; + resource.updatedAt = new Date(); + this.logger.log(`Updated permission to ${newPermission} for user ${userId} on resource ${resourceId}`); + return { ...userPermission }; } - - return [...resource.permissions]; - } - - /** - * Get all resources a user has access to - */ - async getResourcesForUser( - userId: string, - minPermission: PermissionLevel = PermissionLevel.READ, - ): Promise { - const userResources: CollaborativeResource[] = []; - - for (const resource of this.resources.values()) { - const userPermission = resource.permissions.find((p) => p.userId === userId); - - if (userPermission && this.checkPermissionLevel(userPermission.permission, minPermission)) { - userResources.push({ ...resource }); - } + /** + * Invite a user to collaborate on a resource + */ + async inviteUser(resourceId: string, inviterId: string, inviteeId: string, permission: PermissionLevel = PermissionLevel.WRITE): Promise { + // Check if inviter has admin or owner permissions + const inviterPermission = this.getUserPermission(resourceId, inviterId); + if (!inviterPermission || + !this.checkPermissionLevel(inviterPermission.permission, PermissionLevel.ADMIN)) { + throw new Error(`User ${inviterId} does not have permission to invite users to resource ${resourceId}`); + } + return await this.grantAccess(resourceId, inviteeId, permission, inviterId); } - - return userResources; - } - - /** - * Update user's permission level - */ - async updatePermission( - resourceId: string, - userId: string, - newPermission: PermissionLevel, - updatedBy: string, - ): Promise { - const resource = this.resources.get(resourceId); - if (!resource) { - return null; + /** + * Remove a user from a collaboration session + */ + async removeUser(resourceId: string, removerId: string, userIdToRemove: string): Promise { + // Check if remover has admin or owner permissions + const removerPermission = this.getUserPermission(resourceId, removerId); + if (!removerPermission || + !this.checkPermissionLevel(removerPermission.permission, PermissionLevel.ADMIN)) { + throw new Error(`User ${removerId} does not have permission to remove users from resource ${resourceId}`); + } + // Can't remove the owner + const resource = this.resources.get(resourceId); + if (resource && resource.ownerId === userIdToRemove) { + throw new Error(`Cannot remove owner from resource ${resourceId}`); + } + return await this.revokeAccess(resourceId, userIdToRemove); } - - const userPermission = resource.permissions.find((p) => p.userId === userId); - if (!userPermission) { - return null; + /** + * Check if one permission level meets or exceeds another + */ + private checkPermissionLevel(userPermission: PermissionLevel, requiredPermission: PermissionLevel): boolean { + const permissionLevels: PermissionLevel[] = [ + PermissionLevel.READ, + PermissionLevel.WRITE, + PermissionLevel.ADMIN, + PermissionLevel.OWNER, + ]; + const userIndex = permissionLevels.indexOf(userPermission); + const requiredIndex = permissionLevels.indexOf(requiredPermission); + return userIndex >= requiredIndex; } - - userPermission.permission = newPermission; - userPermission.grantedAt = new Date(); - userPermission.grantedBy = updatedBy; - - resource.updatedAt = new Date(); - - this.logger.log( - `Updated permission to ${newPermission} for user ${userId} on resource ${resourceId}`, - ); - - return { ...userPermission }; - } - - /** - * Invite a user to collaborate on a resource - */ - async inviteUser( - resourceId: string, - inviterId: string, - inviteeId: string, - permission: PermissionLevel = PermissionLevel.WRITE, - ): Promise { - // Check if inviter has admin or owner permissions - const inviterPermission = this.getUserPermission(resourceId, inviterId); - - if ( - !inviterPermission || - !this.checkPermissionLevel(inviterPermission.permission, PermissionLevel.ADMIN) - ) { - throw new Error( - `User ${inviterId} does not have permission to invite users to resource ${resourceId}`, - ); + /** + * Determine resource type based on ID pattern + */ + private getResourceType(resourceId: string): 'document' | 'whiteboard' | 'project' { + if (resourceId.startsWith('doc_')) { + return 'document'; + } + else if (resourceId.startsWith('wb_')) { + return 'whiteboard'; + } + else { + return 'project'; + } } - - return await this.grantAccess(resourceId, inviteeId, permission, inviterId); - } - - /** - * Remove a user from a collaboration session - */ - async removeUser( - resourceId: string, - removerId: string, - userIdToRemove: string, - ): Promise { - // Check if remover has admin or owner permissions - const removerPermission = this.getUserPermission(resourceId, removerId); - - if ( - !removerPermission || - !this.checkPermissionLevel(removerPermission.permission, PermissionLevel.ADMIN) - ) { - throw new Error( - `User ${removerId} does not have permission to remove users from resource ${resourceId}`, - ); + /** + * Set default permission level for new resources + */ + setDefaultPermission(permission: PermissionLevel): void { + this.defaultPermission = permission; } - - // Can't remove the owner - const resource = this.resources.get(resourceId); - if (resource && resource.ownerId === userIdToRemove) { - throw new Error(`Cannot remove owner from resource ${resourceId}`); + /** + * Get default permission level + */ + getDefaultPermission(): PermissionLevel { + return this.defaultPermission; } - - return await this.revokeAccess(resourceId, userIdToRemove); - } - - /** - * Check if one permission level meets or exceeds another - */ - private checkPermissionLevel( - userPermission: PermissionLevel, - requiredPermission: PermissionLevel, - ): boolean { - const permissionLevels: PermissionLevel[] = [ - PermissionLevel.READ, - PermissionLevel.WRITE, - PermissionLevel.ADMIN, - PermissionLevel.OWNER, - ]; - - const userIndex = permissionLevels.indexOf(userPermission); - const requiredIndex = permissionLevels.indexOf(requiredPermission); - - return userIndex >= requiredIndex; - } - - /** - * Determine resource type based on ID pattern - */ - private getResourceType(resourceId: string): 'document' | 'whiteboard' | 'project' { - if (resourceId.startsWith('doc_')) { - return 'document'; - } else if (resourceId.startsWith('wb_')) { - return 'whiteboard'; - } else { - return 'project'; - } - } - - /** - * Set default permission level for new resources - */ - setDefaultPermission(permission: PermissionLevel): void { - this.defaultPermission = permission; - } - - /** - * Get default permission level - */ - getDefaultPermission(): PermissionLevel { - return this.defaultPermission; - } } diff --git a/src/collaboration/versioning/version-control.service.ts b/src/collaboration/versioning/version-control.service.ts index f7fc438e..7b3e2c8f 100644 --- a/src/collaboration/versioning/version-control.service.ts +++ b/src/collaboration/versioning/version-control.service.ts @@ -1,257 +1,222 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface ChangeRecord { - id: string; - sessionId: string; - userId: string; - change: any; - operationType: string; - timestamp: number; - previousValue?: any; - newValue?: any; - versionNumber: number; + id: string; + sessionId: string; + userId: string; + change: unknown; + operationType: string; + timestamp: number; + previousValue?: unknown; + newValue?: unknown; + versionNumber: number; } - export interface VersionHistory { - sessionId: string; - versions: ChangeRecord[]; - currentVersion: number; - createdAt: Date; - updatedAt: Date; + sessionId: string; + versions: ChangeRecord[]; + currentVersion: number; + createdAt: Date; + updatedAt: Date; } - @Injectable() export class VersionControlService { - private readonly logger = Logger; - private histories: Map = new Map(); - - /** - * Record a change in the version history - */ - async recordChange(sessionId: string, userId: string, change: any): Promise { - let history = this.histories.get(sessionId); - - if (!history) { - history = { - sessionId, - versions: [], - currentVersion: 0, - createdAt: new Date(), - updatedAt: new Date(), - }; - this.histories.set(sessionId, history); + private readonly logger = Logger; + private histories: Map = new Map(); + /** + * Record a change in the version history + */ + async recordChange(sessionId: string, userId: string, change: unknown): Promise { + let history = this.histories.get(sessionId); + if (!history) { + history = { + sessionId, + versions: [], + currentVersion: 0, + createdAt: new Date(), + updatedAt: new Date(), + }; + this.histories.set(sessionId, history); + } + const versionNumber = history.currentVersion + 1; + const changeRecord: ChangeRecord = { + id: uuidv4(), + sessionId, + userId, + change, + operationType: this.getOperationType(change), + timestamp: Date.now(), + previousValue: this.getPreviousValue(history, change), + newValue: this.getNewValue(change), + versionNumber, + }; + history.versions.push(changeRecord); + history.currentVersion = versionNumber; + history.updatedAt = new Date(); + this.logger.log(`Recorded change ${changeRecord.id} for session ${sessionId}, version ${versionNumber}`); + return changeRecord; } - - const versionNumber = history.currentVersion + 1; - const changeRecord: ChangeRecord = { - id: uuidv4(), - sessionId, - userId, - change, - operationType: this.getOperationType(change), - timestamp: Date.now(), - previousValue: this.getPreviousValue(history, change), - newValue: this.getNewValue(change), - versionNumber, - }; - - history.versions.push(changeRecord); - history.currentVersion = versionNumber; - history.updatedAt = new Date(); - - this.logger.log( - `Recorded change ${changeRecord.id} for session ${sessionId}, version ${versionNumber}`, - ); - - return changeRecord; - } - - /** - * Get the version history for a session - */ - async getVersionHistory(sessionId: string): Promise { - const history = this.histories.get(sessionId); - if (!history) { - return []; + /** + * Get the version history for a session + */ + async getVersionHistory(sessionId: string): Promise { + const history = this.histories.get(sessionId); + if (!history) { + return []; + } + return [...history.versions]; } - - return [...history.versions]; - } - - /** - * Get a specific version of a session - */ - async getVersion(sessionId: string, versionNumber: number): Promise { - const history = this.histories.get(sessionId); - if (!history) { - return null; + /** + * Get a specific version of a session + */ + async getVersion(sessionId: string, versionNumber: number): Promise { + const history = this.histories.get(sessionId); + if (!history) { + return null; + } + const version = history.versions.find((v) => v.versionNumber === versionNumber); + return version || null; } - - const version = history.versions.find((v) => v.versionNumber === versionNumber); - return version || null; - } - - /** - * Get the current version of a session - */ - async getCurrentVersion(sessionId: string): Promise { - const history = this.histories.get(sessionId); - if (!history) { - return null; + /** + * Get the current version of a session + */ + async getCurrentVersion(sessionId: string): Promise { + const history = this.histories.get(sessionId); + if (!history) { + return null; + } + if (history.versions.length === 0) { + return null; + } + return history.versions[history.versions.length - 1]; } - - if (history.versions.length === 0) { - return null; + /** + * Revert to a specific version + */ + async revertToVersion(sessionId: string, versionNumber: number): Promise { + const history = this.histories.get(sessionId); + if (!history) { + throw new Error(`No history found for session ${sessionId}`); + } + const versionIndex = history.versions.findIndex((v) => v.versionNumber === versionNumber); + if (versionIndex === -1) { + throw new Error(`Version ${versionNumber} not found for session ${sessionId}`); + } + // Keep only versions up to the specified version + history.versions = history.versions.slice(0, versionIndex + 1); + history.currentVersion = versionNumber; + history.updatedAt = new Date(); + this.logger.log(`Reverted session ${sessionId} to version ${versionNumber}`); + return [...history.versions]; } - - return history.versions[history.versions.length - 1]; - } - - /** - * Revert to a specific version - */ - async revertToVersion(sessionId: string, versionNumber: number): Promise { - const history = this.histories.get(sessionId); - if (!history) { - throw new Error(`No history found for session ${sessionId}`); + /** + * Compare two versions + */ + async compareVersions(sessionId: string, version1: number, version2: number): Promise<{ + differences: unknown; + version1Data: ChangeRecord | null; + version2Data: ChangeRecord | null; + }> { + const v1 = await this.getVersion(sessionId, version1); + const v2 = await this.getVersion(sessionId, version2); + const differences = this.calculateDifferences(v1?.change, v2?.change); + return { + differences, + version1Data: v1, + version2Data: v2, + }; } - - const versionIndex = history.versions.findIndex((v) => v.versionNumber === versionNumber); - if (versionIndex === -1) { - throw new Error(`Version ${versionNumber} not found for session ${sessionId}`); + /** + * Get change statistics + */ + async getChangeStatistics(sessionId: string): Promise<{ + totalChanges: number; + changesByUser: Map; + changesOverTime: Array<{ + date: Date; + count: number; + }>; + }> { + const history = this.histories.get(sessionId); + if (!history) { + return { + totalChanges: 0, + changesByUser: new Map(), + changesOverTime: [], + }; + } + const changesByUser = new Map(); + const changesOverTime: Array<{ + date: Date; + count: number; + }> = []; + for (const version of history.versions) { + // Count changes by user + const userCount = changesByUser.get(version.userId) || 0; + changesByUser.set(version.userId, userCount + 1); + // Group by day for time series + const _dateStr = new Date(version.timestamp).toDateString(); + const timeEntry = changesOverTime.find((entry) => entry.date.toDateString() === new Date(version.timestamp).toDateString()); + if (timeEntry) { + timeEntry.count++; + } + else { + changesOverTime.push({ + date: new Date(version.timestamp), + count: 1, + }); + } + } + return { + totalChanges: history.versions.length, + changesByUser, + changesOverTime, + }; } - - // Keep only versions up to the specified version - history.versions = history.versions.slice(0, versionIndex + 1); - history.currentVersion = versionNumber; - history.updatedAt = new Date(); - - this.logger.log(`Reverted session ${sessionId} to version ${versionNumber}`); - - return [...history.versions]; - } - - /** - * Compare two versions - */ - async compareVersions( - sessionId: string, - version1: number, - version2: number, - ): Promise<{ - differences: any; - version1Data: ChangeRecord | null; - version2Data: ChangeRecord | null; - }> { - const v1 = await this.getVersion(sessionId, version1); - const v2 = await this.getVersion(sessionId, version2); - - const differences = this.calculateDifferences(v1?.change, v2?.change); - - return { - differences, - version1Data: v1, - version2Data: v2, - }; - } - - /** - * Get change statistics - */ - async getChangeStatistics(sessionId: string): Promise<{ - totalChanges: number; - changesByUser: Map; - changesOverTime: Array<{ date: Date; count: number }>; - }> { - const history = this.histories.get(sessionId); - if (!history) { - return { - totalChanges: 0, - changesByUser: new Map(), - changesOverTime: [], - }; + /** + * Get the operation type from a change object + */ + private getOperationType(change: unknown): string { + if (change.type) { + return change.type; + } + if (change.operation) { + return change.operation; + } + return 'unknown'; } - - const changesByUser = new Map(); - const changesOverTime: Array<{ date: Date; count: number }> = []; - - for (const version of history.versions) { - // Count changes by user - const userCount = changesByUser.get(version.userId) || 0; - changesByUser.set(version.userId, userCount + 1); - - // Group by day for time series - const _dateStr = new Date(version.timestamp).toDateString(); - const timeEntry = changesOverTime.find( - (entry) => entry.date.toDateString() === new Date(version.timestamp).toDateString(), - ); - - if (timeEntry) { - timeEntry.count++; - } else { - changesOverTime.push({ - date: new Date(version.timestamp), - count: 1, - }); - } + /** + * Extract the previous value from history + */ + private getPreviousValue(history: VersionHistory, _change: unknown): unknown { + // In a real implementation, this would retrieve the previous state + // For now, we'll return the last recorded value if available + if (history.versions.length > 0) { + return history.versions[history.versions.length - 1].newValue; + } + return undefined; } - - return { - totalChanges: history.versions.length, - changesByUser, - changesOverTime, - }; - } - - /** - * Get the operation type from a change object - */ - private getOperationType(change: any): string { - if (change.type) { - return change.type; + /** + * Extract the new value from a change + */ + private getNewValue(change: unknown): unknown { + if (change.newValue !== undefined) { + return change.newValue; + } + if (change.data !== undefined) { + return change.data; + } + return change; } - if (change.operation) { - return change.operation; + /** + * Calculate differences between two states + */ + private calculateDifferences(state1: unknown, state2: unknown): unknown { + // This is a simplified difference calculation + // A production implementation would use a more sophisticated diff algorithm + return { + state1, + state2, + hasChanges: JSON.stringify(state1) !== JSON.stringify(state2), + }; } - return 'unknown'; - } - - /** - * Extract the previous value from history - */ - private getPreviousValue(history: VersionHistory, _change: any): any { - // In a real implementation, this would retrieve the previous state - // For now, we'll return the last recorded value if available - if (history.versions.length > 0) { - return history.versions[history.versions.length - 1].newValue; - } - return undefined; - } - - /** - * Extract the new value from a change - */ - private getNewValue(change: any): any { - if (change.newValue !== undefined) { - return change.newValue; - } - if (change.data !== undefined) { - return change.data; - } - return change; - } - - /** - * Calculate differences between two states - */ - private calculateDifferences(state1: any, state2: any): any { - // This is a simplified difference calculation - // A production implementation would use a more sophisticated diff algorithm - return { - state1, - state2, - hasChanges: JSON.stringify(state1) !== JSON.stringify(state2), - }; - } } diff --git a/src/collaboration/whiteboard/whiteboard.service.ts b/src/collaboration/whiteboard/whiteboard.service.ts index 07e7bfb0..ff6e2dc9 100644 --- a/src/collaboration/whiteboard/whiteboard.service.ts +++ b/src/collaboration/whiteboard/whiteboard.service.ts @@ -1,278 +1,222 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface DrawingElement { - id: string; - type: 'line' | 'rectangle' | 'circle' | 'text' | 'freehand'; - x: number; - y: number; - width?: number; - height?: number; - points?: Array<{ x: number; y: number }>; // For freehand drawings - text?: string; // For text elements - color: string; - strokeWidth: number; - userId: string; - timestamp: number; + id: string; + type: 'line' | 'rectangle' | 'circle' | 'text' | 'freehand'; + x: number; + y: number; + width?: number; + height?: number; + points?: Array<{ + x: number; + y: number; + }>; // For freehand drawings + text?: string; // For text elements + color: string; + strokeWidth: number; + userId: string; + timestamp: number; } - export interface WhiteboardOperation { - id: string; - type: 'addElement' | 'removeElement' | 'updateElement' | 'clearBoard'; - element?: DrawingElement; - elementId?: string; - userId: string; - timestamp: number; + id: string; + type: 'addElement' | 'removeElement' | 'updateElement' | 'clearBoard'; + element?: DrawingElement; + elementId?: string; + userId: string; + timestamp: number; } - export interface CollaborativeWhiteboard { - id: string; - elements: DrawingElement[]; - operations: WhiteboardOperation[]; - collaborators: string[]; - createdAt: Date; - updatedAt: Date; + id: string; + elements: DrawingElement[]; + operations: WhiteboardOperation[]; + collaborators: string[]; + createdAt: Date; + updatedAt: Date; } - @Injectable() export class WhiteboardService { - private readonly logger = Logger; - private whiteboards: Map = new Map(); - - /** - * Initialize a new collaborative whiteboard - */ - async initializeWhiteboard(whiteboardId: string): Promise { - const whiteboard: CollaborativeWhiteboard = { - id: whiteboardId, - elements: [], - operations: [], - collaborators: [], - createdAt: new Date(), - updatedAt: new Date(), - }; - - this.whiteboards.set(whiteboardId, whiteboard); - this.logger.log(`Initialized whiteboard ${whiteboardId}`); - - return whiteboard; - } - - /** - * Get a collaborative whiteboard - */ - async getWhiteboard(whiteboardId: string): Promise { - return this.whiteboards.get(whiteboardId) || null; - } - - /** - * Apply an operation to a whiteboard - */ - async applyOperation( - whiteboardId: string, - userId: string, - operation: Omit, - ): Promise { - const whiteboard = this.whiteboards.get(whiteboardId); - if (!whiteboard) { - throw new Error(`Whiteboard ${whiteboardId} not found`); + private readonly logger = Logger; + private whiteboards: Map = new Map(); + /** + * Initialize a new collaborative whiteboard + */ + async initializeWhiteboard(whiteboardId: string): Promise { + const whiteboard: CollaborativeWhiteboard = { + id: whiteboardId, + elements: [], + operations: [], + collaborators: [], + createdAt: new Date(), + updatedAt: new Date(), + }; + this.whiteboards.set(whiteboardId, whiteboard); + this.logger.log(`Initialized whiteboard ${whiteboardId}`); + return whiteboard; } - - // Add metadata to the operation - const opWithMetadata: WhiteboardOperation = { - ...operation, - id: uuidv4(), - timestamp: Date.now(), - userId, - }; - - // Apply the operation to the whiteboard - this.applyOperationToWhiteboard(whiteboard, opWithMetadata); - - // Add the operation to the whiteboard's operation history - whiteboard.operations.push(opWithMetadata); - whiteboard.updatedAt = new Date(); - - // Add user to collaborators if not already present - if (!whiteboard.collaborators.includes(userId)) { - whiteboard.collaborators.push(userId); + /** + * Get a collaborative whiteboard + */ + async getWhiteboard(whiteboardId: string): Promise { + return this.whiteboards.get(whiteboardId) || null; } - - this.logger.log(`Applied operation ${opWithMetadata.id} to whiteboard ${whiteboardId}`); - - return whiteboard; - } - - /** - * Apply an operation to the whiteboard state - */ - private applyOperationToWhiteboard( - whiteboard: CollaborativeWhiteboard, - operation: WhiteboardOperation, - ): void { - switch (operation.type) { - case 'addElement': - if (operation.element) { - whiteboard.elements.push({ ...operation.element }); - } - break; - - case 'removeElement': - if (operation.elementId) { - whiteboard.elements = whiteboard.elements.filter((el) => el.id !== operation.elementId); - } - break; - - case 'updateElement': - if (operation.elementId && operation.element) { - const index = whiteboard.elements.findIndex((el) => el.id === operation.elementId); - if (index !== -1) { - whiteboard.elements[index] = { ...operation.element }; - } + /** + * Apply an operation to a whiteboard + */ + async applyOperation(whiteboardId: string, userId: string, operation: Omit): Promise { + const whiteboard = this.whiteboards.get(whiteboardId); + if (!whiteboard) { + throw new Error(`Whiteboard ${whiteboardId} not found`); } - break; - - case 'clearBoard': - whiteboard.elements = []; - break; - } - } - - /** - * Transform an operation against concurrent operations - */ - private transformOperation( - operation: WhiteboardOperation, - concurrentOperations: WhiteboardOperation[], - ): WhiteboardOperation { - // For whiteboard operations, transformation is simpler than text operations - // We mainly need to handle cases where elements are removed while others try to update them - let transformedOp = { ...operation }; - - for (const concurrentOp of concurrentOperations) { - // If a concurrent operation removes an element that this operation tries to update - if ( - concurrentOp.type === 'removeElement' && - operation.type === 'updateElement' && - concurrentOp.elementId === operation.elementId - ) { - // Convert the update to an add operation since the element was removed - transformedOp = { - ...transformedOp, - type: 'addElement', - elementId: undefined, + // Add metadata to the operation + const opWithMetadata: WhiteboardOperation = { + ...operation, + id: uuidv4(), + timestamp: Date.now(), + userId, }; - } + // Apply the operation to the whiteboard + this.applyOperationToWhiteboard(whiteboard, opWithMetadata); + // Add the operation to the whiteboard's operation history + whiteboard.operations.push(opWithMetadata); + whiteboard.updatedAt = new Date(); + // Add user to collaborators if not already present + if (!whiteboard.collaborators.includes(userId)) { + whiteboard.collaborators.push(userId); + } + this.logger.log(`Applied operation ${opWithMetadata.id} to whiteboard ${whiteboardId}`); + return whiteboard; } - - return transformedOp; - } - - /** - * Resolve conflicts between simultaneous whiteboard edits - */ - async resolveConflicts( - whiteboardId: string, - operations: WhiteboardOperation[], - ): Promise { - const whiteboard = this.whiteboards.get(whiteboardId); - if (!whiteboard) { - throw new Error(`Whiteboard ${whiteboardId} not found`); + /** + * Apply an operation to the whiteboard state + */ + private applyOperationToWhiteboard(whiteboard: CollaborativeWhiteboard, operation: WhiteboardOperation): void { + switch (operation.type) { + case 'addElement': + if (operation.element) { + whiteboard.elements.push({ ...operation.element }); + } + break; + case 'removeElement': + if (operation.elementId) { + whiteboard.elements = whiteboard.elements.filter((el) => el.id !== operation.elementId); + } + break; + case 'updateElement': + if (operation.elementId && operation.element) { + const index = whiteboard.elements.findIndex((el) => el.id === operation.elementId); + if (index !== -1) { + whiteboard.elements[index] = { ...operation.element }; + } + } + break; + case 'clearBoard': + whiteboard.elements = []; + break; + } } - - // Sort operations by timestamp to process in chronological order - const sortedOps = [...operations].sort((a, b) => a.timestamp - b.timestamp); - - // Clear existing operations and reapply in order to resolve conflicts - whiteboard.operations = []; - whiteboard.elements = []; - - // Rebuild whiteboard state from the sorted operations - for (const operation of sortedOps) { - this.applyOperationToWhiteboard(whiteboard, operation); - whiteboard.operations.push(operation); + /** + * Transform an operation against concurrent operations + */ + private transformOperation(operation: WhiteboardOperation, concurrentOperations: WhiteboardOperation[]): WhiteboardOperation { + // For whiteboard operations, transformation is simpler than text operations + // We mainly need to handle cases where elements are removed while others try to update them + let transformedOp = { ...operation }; + for (const concurrentOp of concurrentOperations) { + // If a concurrent operation removes an element that this operation tries to update + if (concurrentOp.type === 'removeElement' && + operation.type === 'updateElement' && + concurrentOp.elementId === operation.elementId) { + // Convert the update to an add operation since the element was removed + transformedOp = { + ...transformedOp, + type: 'addElement', + elementId: undefined, + }; + } + } + return transformedOp; } - - whiteboard.updatedAt = new Date(); - - return whiteboard; - } - - /** - * Get whiteboard history - */ - async getWhiteboardHistory(whiteboardId: string): Promise { - const whiteboard = this.whiteboards.get(whiteboardId); - if (!whiteboard) { - throw new Error(`Whiteboard ${whiteboardId} not found`); + /** + * Resolve conflicts between simultaneous whiteboard edits + */ + async resolveConflicts(whiteboardId: string, operations: WhiteboardOperation[]): Promise { + const whiteboard = this.whiteboards.get(whiteboardId); + if (!whiteboard) { + throw new Error(`Whiteboard ${whiteboardId} not found`); + } + // Sort operations by timestamp to process in chronological order + const sortedOps = [...operations].sort((a, b) => a.timestamp - b.timestamp); + // Clear existing operations and reapply in order to resolve conflicts + whiteboard.operations = []; + whiteboard.elements = []; + // Rebuild whiteboard state from the sorted operations + for (const operation of sortedOps) { + this.applyOperationToWhiteboard(whiteboard, operation); + whiteboard.operations.push(operation); + } + whiteboard.updatedAt = new Date(); + return whiteboard; } - - return [...whiteboard.operations]; - } - - /** - * Add a drawing element to the whiteboard - */ - async addElement( - whiteboardId: string, - element: Omit, - userId: string, - ): Promise { - const whiteboard = this.whiteboards.get(whiteboardId); - if (!whiteboard) { - throw new Error(`Whiteboard ${whiteboardId} not found`); + /** + * Get whiteboard history + */ + async getWhiteboardHistory(whiteboardId: string): Promise { + const whiteboard = this.whiteboards.get(whiteboardId); + if (!whiteboard) { + throw new Error(`Whiteboard ${whiteboardId} not found`); + } + return [...whiteboard.operations]; } - - const newElement: DrawingElement = { - ...element, - id: uuidv4(), - timestamp: Date.now(), - userId, - }; - - whiteboard.elements.push(newElement); - whiteboard.updatedAt = new Date(); - - // Record the operation - const operation: WhiteboardOperation = { - id: uuidv4(), - type: 'addElement', - element: newElement, - userId, - timestamp: Date.now(), - }; - whiteboard.operations.push(operation); - - return newElement; - } - - /** - * Remove a drawing element from the whiteboard - */ - async removeElement(whiteboardId: string, elementId: string, userId: string): Promise { - const whiteboard = this.whiteboards.get(whiteboardId); - if (!whiteboard) { - throw new Error(`Whiteboard ${whiteboardId} not found`); + /** + * Add a drawing element to the whiteboard + */ + async addElement(whiteboardId: string, element: Omit, userId: string): Promise { + const whiteboard = this.whiteboards.get(whiteboardId); + if (!whiteboard) { + throw new Error(`Whiteboard ${whiteboardId} not found`); + } + const newElement: DrawingElement = { + ...element, + id: uuidv4(), + timestamp: Date.now(), + userId, + }; + whiteboard.elements.push(newElement); + whiteboard.updatedAt = new Date(); + // Record the operation + const operation: WhiteboardOperation = { + id: uuidv4(), + type: 'addElement', + element: newElement, + userId, + timestamp: Date.now(), + }; + whiteboard.operations.push(operation); + return newElement; } - - const elementIndex = whiteboard.elements.findIndex((el) => el.id === elementId); - if (elementIndex === -1) { - return false; + /** + * Remove a drawing element from the whiteboard + */ + async removeElement(whiteboardId: string, elementId: string, userId: string): Promise { + const whiteboard = this.whiteboards.get(whiteboardId); + if (!whiteboard) { + throw new Error(`Whiteboard ${whiteboardId} not found`); + } + const elementIndex = whiteboard.elements.findIndex((el) => el.id === elementId); + if (elementIndex === -1) { + return false; + } + whiteboard.elements.splice(elementIndex, 1); + whiteboard.updatedAt = new Date(); + // Record the operation + const operation: WhiteboardOperation = { + id: uuidv4(), + type: 'removeElement', + elementId, + userId, + timestamp: Date.now(), + }; + whiteboard.operations.push(operation); + return true; } - - whiteboard.elements.splice(elementIndex, 1); - whiteboard.updatedAt = new Date(); - - // Record the operation - const operation: WhiteboardOperation = { - id: uuidv4(), - type: 'removeElement', - elementId, - userId, - timestamp: Date.now(), - }; - whiteboard.operations.push(operation); - - return true; - } } diff --git a/src/common/common.module.ts b/src/common/common.module.ts index b1229df2..bc33ed2d 100644 --- a/src/common/common.module.ts +++ b/src/common/common.module.ts @@ -1,8 +1,8 @@ import { Module } from '@nestjs/common'; import { TransactionHelperService } from './database/transaction-helper.service'; - @Module({ - providers: [TransactionHelperService], - exports: [TransactionHelperService], + providers: [TransactionHelperService], + exports: [TransactionHelperService], }) -export class CommonModule {} +export class CommonModule { +} diff --git a/src/common/constants/app.constants.ts b/src/common/constants/app.constants.ts index 07bd4241..9e7eacfe 100644 --- a/src/common/constants/app.constants.ts +++ b/src/common/constants/app.constants.ts @@ -2,32 +2,27 @@ * Application constants * Centralized location for magic values and configuration constants */ - export const APP_CONSTANTS = { - // Security constants - DUMMY_USER_ID: '00000000-0000-0000-0000-000000000000', - - // Pagination defaults - DEFAULT_PAGE_SIZE: 10, - MAX_PAGE_SIZE: 100, // Max items allowed per page for paginated API responses - - // Status values - COURSE_STATUS: { - DRAFT: 'draft', - PUBLISHED: 'published', - ARCHIVED: 'archived', - }, - - ENROLLMENT_STATUS: { - ACTIVE: 'active', - COMPLETED: 'completed', - CANCELLED: 'cancelled', - }, - - // Role constants - USER_ROLES: { - INSTRUCTOR: 'instructor', - STUDENT: 'student', - ADMIN: 'admin', - }, + // Security constants + DUMMY_USER_ID: '00000000-0000-0000-0000-000000000000', + // Pagination defaults + DEFAULT_PAGE_SIZE: 10, + MAX_PAGE_SIZE: 100, // Max items allowed per page for paginated API responses + // Status values + COURSE_STATUS: { + DRAFT: 'draft', + PUBLISHED: 'published', + ARCHIVED: 'archived', + }, + ENROLLMENT_STATUS: { + ACTIVE: 'active', + COMPLETED: 'completed', + CANCELLED: 'cancelled', + }, + // Role constants + USER_ROLES: { + INSTRUCTOR: 'instructor', + STUDENT: 'student', + ADMIN: 'admin', + }, } as const; diff --git a/src/common/constants/auth.constants.ts b/src/common/constants/auth.constants.ts index 5b3dd5df..5c8bb425 100644 --- a/src/common/constants/auth.constants.ts +++ b/src/common/constants/auth.constants.ts @@ -1,3 +1,3 @@ export const AUTH_STRATEGY = { - JWT: 'jwt', + JWT: 'jwt', } as const; diff --git a/src/common/constants/event.constants.ts b/src/common/constants/event.constants.ts index 91b65d2c..7fa18851 100644 --- a/src/common/constants/event.constants.ts +++ b/src/common/constants/event.constants.ts @@ -1,31 +1,24 @@ export const APP_EVENTS = { - // Data sync events - DATA_UPDATED: 'data.updated', - - // Cache events - CACHE_INVALIDATED: 'cache.invalidated', - CACHE_PURGED: 'cache.purged', - - // Data integrity events - DATA_CONSISTENCY_SCHEDULED: 'data.consistency.scheduled', - DATA_INTEGRITY_VIOLATION: 'data.integrity.violation', - - // User events - USER_SIGNUP: 'user.signup', - USER_ADD_TAG: 'user.addTag', - USER_REMOVE_TAG: 'user.removeTag', - - // Course events - COURSE_ENROLLED: 'course.enrolled', - COURSE_COMPLETED: 'course.completed', - - // Payment events - PAYMENT_COMPLETED: 'payment.completed', - - // Segment events - SEGMENT_ADD_USER: 'segment.addUser', - - // Notification events - NOTIFICATION_SEND: 'notification.send', - NOTIFICATION_TEMPLATE_SEND: 'notification.template.send', + // Data sync events + DATA_UPDATED: 'data.updated', + // Cache events + CACHE_INVALIDATED: 'cache.invalidated', + CACHE_PURGED: 'cache.purged', + // Data integrity events + DATA_CONSISTENCY_SCHEDULED: 'data.consistency.scheduled', + DATA_INTEGRITY_VIOLATION: 'data.integrity.violation', + // User events + USER_SIGNUP: 'user.signup', + USER_ADD_TAG: 'user.addTag', + USER_REMOVE_TAG: 'user.removeTag', + // Course events + COURSE_ENROLLED: 'course.enrolled', + COURSE_COMPLETED: 'course.completed', + // Payment events + PAYMENT_COMPLETED: 'payment.completed', + // Segment events + SEGMENT_ADD_USER: 'segment.addUser', + // Notification events + NOTIFICATION_SEND: 'notification.send', + NOTIFICATION_TEMPLATE_SEND: 'notification.template.send', } as const; diff --git a/src/common/constants/queue.constants.ts b/src/common/constants/queue.constants.ts index cf131b2a..24956b0b 100644 --- a/src/common/constants/queue.constants.ts +++ b/src/common/constants/queue.constants.ts @@ -1,42 +1,36 @@ export const QUEUE_NAMES = { - EMAIL: 'email', - EMAIL_MARKETING: 'email-marketing', - SYNC_TASKS: 'sync-tasks', - BACKUP_PROCESSING: 'backup-processing', - MESSAGE_QUEUE: 'message-queue', - MEDIA_PROCESSING: 'media-processing', - DEFAULT: 'default', - USER_DATA_EXPORT: 'user-data-export', - SUBSCRIPTIONS: 'subscriptions', - WEBHOOKS: 'webhooks', + EMAIL: 'email', + EMAIL_MARKETING: 'email-marketing', + SYNC_TASKS: 'sync-tasks', + BACKUP_PROCESSING: 'backup-processing', + MESSAGE_QUEUE: 'message-queue', + MEDIA_PROCESSING: 'media-processing', + DEFAULT: 'default', + USER_DATA_EXPORT: 'user-data-export', + SUBSCRIPTIONS: 'subscriptions', + WEBHOOKS: 'webhooks', } as const; - export const JOB_NAMES = { - // Email queue - SEND_EMAIL: 'send-email', - - // Email marketing queue - SEND_CAMPAIGN: 'send-campaign', - PROCESS_CAMPAIGN: 'process-campaign', - RESUME_CAMPAIGN: 'resume-campaign', - SEND_AUTOMATION_EMAIL: 'send-automation-email', - CONTINUE_AUTOMATION: 'continue-automation', - CALL_WEBHOOK: 'call-webhook', - - // Sync tasks queue - CONSISTENCY_CHECK: 'consistency-check', - REPLICATE_DATA: 'replicate-data', - - // Backup processing queue - CREATE_BACKUP: 'create-backup', - VERIFY_BACKUP: 'verify-backup', - RECOVERY_TEST: 'recovery-test', - DELETE_BACKUP: 'delete-backup', - - // Media processing queue - TRANSCODE_VIDEO: 'transcode-video', - - // Payments queues - PROCESS_SUBSCRIPTION: 'process_subscription', - PROCESS_WEBHOOK: 'process-webhook', + // Email queue + SEND_EMAIL: 'send-email', + // Email marketing queue + SEND_CAMPAIGN: 'send-campaign', + PROCESS_CAMPAIGN: 'process-campaign', + RESUME_CAMPAIGN: 'resume-campaign', + SEND_AUTOMATION_EMAIL: 'send-automation-email', + CONTINUE_AUTOMATION: 'continue-automation', + CALL_WEBHOOK: 'call-webhook', + // Sync tasks queue + CONSISTENCY_CHECK: 'consistency-check', + REPLICATE_DATA: 'replicate-data', + // Backup processing queue + CREATE_BACKUP: 'create-backup', + VERIFY_BACKUP: 'verify-backup', + RECOVERY_TEST: 'recovery-test', + DELETE_BACKUP: 'delete-backup', + // Media processing queue + TRANSCODE_VIDEO: 'transcode-video', + // Payments queues + PROCESS_SUBSCRIPTION: 'process_subscription', + PROCESS_WEBHOOK: 'process-webhook', } as const; diff --git a/src/common/csrf/csrf.controller.ts b/src/common/csrf/csrf.controller.ts index 97ffab44..261921e2 100644 --- a/src/common/csrf/csrf.controller.ts +++ b/src/common/csrf/csrf.controller.ts @@ -3,54 +3,58 @@ import { Request, Response } from 'express'; import { ApiTags, ApiOperation, ApiResponse } from '@nestjs/swagger'; import { JwtAuthGuard } from '../../auth/guards/jwt-auth.guard'; import { CsrfService } from './csrf.service'; - @ApiTags('CSRF') @Controller('csrf') export class CsrfController { - constructor(private readonly csrfService: CsrfService) {} - - @Get('token') - @UseGuards(JwtAuthGuard) - @ApiOperation({ summary: 'Get CSRF token' }) - @ApiResponse({ status: 200, description: 'CSRF token generated successfully' }) - getCsrfToken(@Req() req: Request, @Res() res: Response): void { - const sessionId = this.getSessionId(req); - const token = this.csrfService.generateToken(sessionId); - - res.setHeader('X-CSRF-Token', token); - res.json({ csrfToken: token }); - } - - @Post('validate') - @UseGuards(JwtAuthGuard) - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Validate CSRF token' }) - @ApiResponse({ status: 200, description: 'CSRF token is valid' }) - @ApiResponse({ status: 400, description: 'Invalid CSRF token' }) - validateCsrfToken(@Req() req: Request): { valid: boolean } { - const sessionId = this.getSessionId(req); - const token = req.body?.csrfToken || req.headers['x-csrf-token']; - - const isValid = this.csrfService.validateToken(sessionId, token); - return { valid: isValid }; - } - - @Post('invalidate') - @UseGuards(JwtAuthGuard) - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Invalidate CSRF token' }) - @ApiResponse({ status: 200, description: 'CSRF token invalidated successfully' }) - invalidateCsrfToken(@Req() req: Request): { message: string } { - const sessionId = this.getSessionId(req); - this.csrfService.invalidateToken(sessionId); - - return { message: 'CSRF token invalidated successfully' }; - } - - private getSessionId(req: Request): string { - if ((req as any).session?.id) { - return (req as any).session.id; + constructor(private readonly csrfService: CsrfService) { } + @Get('token') + @UseGuards(JwtAuthGuard) + @ApiOperation({ summary: 'Get CSRF token' }) + @ApiResponse({ status: 200, description: 'CSRF token generated successfully' }) + getCsrfToken( + @Req() + req: Request, + @Res() + res: Response): void { + const sessionId = this.getSessionId(req); + const token = this.csrfService.generateToken(sessionId); + res.setHeader('X-CSRF-Token', token); + res.json({ csrfToken: token }); + } + @Post('validate') + @UseGuards(JwtAuthGuard) + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Validate CSRF token' }) + @ApiResponse({ status: 200, description: 'CSRF token is valid' }) + @ApiResponse({ status: 400, description: 'Invalid CSRF token' }) + validateCsrfToken( + @Req() + req: Request): { + valid: boolean; + } { + const sessionId = this.getSessionId(req); + const token = req.body?.csrfToken || req.headers['x-csrf-token']; + const isValid = this.csrfService.validateToken(sessionId, token); + return { valid: isValid }; + } + @Post('invalidate') + @UseGuards(JwtAuthGuard) + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Invalidate CSRF token' }) + @ApiResponse({ status: 200, description: 'CSRF token invalidated successfully' }) + invalidateCsrfToken( + @Req() + req: Request): { + message: string; + } { + const sessionId = this.getSessionId(req); + this.csrfService.invalidateToken(sessionId); + return { message: 'CSRF token invalidated successfully' }; + } + private getSessionId(req: Request): string { + if ((req as unknown).session?.id) { + return (req as unknown).session.id; + } + return req.ip || req.connection.remoteAddress || 'unknown'; } - return req.ip || req.connection.remoteAddress || 'unknown'; - } } diff --git a/src/common/csrf/csrf.module.ts b/src/common/csrf/csrf.module.ts index 2535efe1..894a5fdf 100644 --- a/src/common/csrf/csrf.module.ts +++ b/src/common/csrf/csrf.module.ts @@ -2,14 +2,13 @@ import { Module, MiddlewareConsumer, RequestMethod } from '@nestjs/common'; import { CsrfMiddleware } from '../middleware/csrf.middleware'; import { CsrfService } from './csrf.service'; import { CsrfController } from './csrf.controller'; - @Module({ - providers: [CsrfService], - controllers: [CsrfController], - exports: [CsrfService], + providers: [CsrfService], + controllers: [CsrfController], + exports: [CsrfService], }) export class CsrfModule { - configure(consumer: MiddlewareConsumer) { - consumer.apply(CsrfMiddleware).forRoutes({ path: '*', method: RequestMethod.ALL }); - } + configure(consumer: MiddlewareConsumer) { + consumer.apply(CsrfMiddleware).forRoutes({ path: '*', method: RequestMethod.ALL }); + } } diff --git a/src/common/csrf/csrf.service.ts b/src/common/csrf/csrf.service.ts index 2f2ef0b4..954ad64a 100644 --- a/src/common/csrf/csrf.service.ts +++ b/src/common/csrf/csrf.service.ts @@ -1,54 +1,45 @@ import { Injectable } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import { v4 as uuidv4 } from 'uuid'; - @Injectable() export class CsrfService { - private readonly csrfTokens = new Map(); - private readonly tokenExpiryTime: number; - - constructor(private configService: ConfigService) { - this.tokenExpiryTime = this.configService.get('CSRF_TOKEN_EXPIRY', 3600000); // 1 hour default - } - - generateToken(sessionId: string): string { - const token = uuidv4(); - const expires = Date.now() + this.tokenExpiryTime; - - this.csrfTokens.set(sessionId, { token, expires }); - return token; - } - - validateToken(sessionId: string, token: string): boolean { - const storedToken = this.csrfTokens.get(sessionId); - - if (!storedToken || storedToken.expires <= Date.now()) { - return false; + private readonly csrfTokens = new Map(); + private readonly tokenExpiryTime: number; + constructor(private configService: ConfigService) { + this.tokenExpiryTime = this.configService.get('CSRF_TOKEN_EXPIRY', 3600000); // 1 hour default } - - return storedToken.token === token; - } - - invalidateToken(sessionId: string): void { - this.csrfTokens.delete(sessionId); - } - - getToken(sessionId: string): string | null { - const storedToken = this.csrfTokens.get(sessionId); - - if (!storedToken || storedToken.expires <= Date.now()) { - return null; + generateToken(sessionId: string): string { + const token = uuidv4(); + const expires = Date.now() + this.tokenExpiryTime; + this.csrfTokens.set(sessionId, { token, expires }); + return token; } - - return storedToken.token; - } - - cleanupExpiredTokens(): void { - const now = Date.now(); - for (const [sessionId, tokenData] of this.csrfTokens.entries()) { - if (tokenData.expires <= now) { + validateToken(sessionId: string, token: string): boolean { + const storedToken = this.csrfTokens.get(sessionId); + if (!storedToken || storedToken.expires <= Date.now()) { + return false; + } + return storedToken.token === token; + } + invalidateToken(sessionId: string): void { this.csrfTokens.delete(sessionId); - } } - } + getToken(sessionId: string): string | null { + const storedToken = this.csrfTokens.get(sessionId); + if (!storedToken || storedToken.expires <= Date.now()) { + return null; + } + return storedToken.token; + } + cleanupExpiredTokens(): void { + const now = Date.now(); + for (const [sessionId, tokenData] of this.csrfTokens.entries()) { + if (tokenData.expires <= now) { + this.csrfTokens.delete(sessionId); + } + } + } } diff --git a/src/common/database/database.module.ts b/src/common/database/database.module.ts index a5ab9a6d..2ac32def 100644 --- a/src/common/database/database.module.ts +++ b/src/common/database/database.module.ts @@ -1,14 +1,14 @@ import { Module, Global } from '@nestjs/common'; import { TransactionService } from './transaction.service'; import { TransactionalInterceptor } from './transactional.interceptor'; - /** * Database Module * Provides transaction management services globally */ @Global() @Module({ - providers: [TransactionService, TransactionalInterceptor], - exports: [TransactionService, TransactionalInterceptor], + providers: [TransactionService, TransactionalInterceptor], + exports: [TransactionService, TransactionalInterceptor], }) -export class DatabaseModule {} +export class DatabaseModule { +} diff --git a/src/common/database/examples/booking-transaction.example.ts b/src/common/database/examples/booking-transaction.example.ts index b3ce5f24..1d5b1059 100644 --- a/src/common/database/examples/booking-transaction.example.ts +++ b/src/common/database/examples/booking-transaction.example.ts @@ -1,194 +1,109 @@ import { Injectable, Logger } from '@nestjs/common'; import { TransactionService } from '../transaction.service'; - /** * Example: Booking Transaction * Demonstrates atomic booking operations (consulting sessions, courses, etc.) */ @Injectable() export class BookingTransactionExample { - private readonly logger = new Logger(BookingTransactionExample.name); - - constructor(private readonly transactionService: TransactionService) {} - - /** - * Book a consulting session - * Ensures slot is reserved, payment is processed, and notification is sent atomically - */ - async bookConsultingSession( - userId: string, - consultantId: string, - slotId: string, - amount: number, - ): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Check and lock the slot - const slot = await manager.query( - 'SELECT * FROM consulting_slots WHERE id = $1 AND status = $2 FOR UPDATE', - [slotId, 'available'], - ); - - if (!slot || slot.length === 0) { - throw new Error('Slot not available'); - } - - // 2. Check user balance - const user = await manager.query('SELECT balance FROM users WHERE id = $1 FOR UPDATE', [ - userId, - ]); - - if (!user || user.length === 0 || user[0].balance < amount) { - throw new Error('Insufficient balance'); - } - - // 3. Deduct payment from user - await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ - amount, - userId, - ]); - - // 4. Add payment to consultant - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - amount, - consultantId, - ]); - - // 5. Mark slot as booked - await manager.query( - 'UPDATE consulting_slots SET status = $1, booked_by = $2, booked_at = NOW() WHERE id = $3', - ['booked', userId, slotId], - ); - - // 6. Create booking record - const booking = await manager.query( - 'INSERT INTO bookings (user_id, consultant_id, slot_id, amount, status) VALUES ($1, $2, $3, $4, $5) RETURNING *', - [userId, consultantId, slotId, amount, 'confirmed'], - ); - - // 7. Create notification - await manager.query( - 'INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', - [userId, 'booking_confirmed', `Your session with consultant ${consultantId} is confirmed`], - ); - - await manager.query( - 'INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', - [consultantId, 'new_booking', `New booking from user ${userId}`], - ); - - this.logger.log(`Booking created: ${booking[0].id}`); - - return booking[0]; - }); - } - - /** - * Cancel booking with refund - */ - async cancelBooking(bookingId: string, refundAmount: number): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Get booking details - const booking = await manager.query( - 'SELECT * FROM bookings WHERE id = $1 AND status = $2 FOR UPDATE', - [bookingId, 'confirmed'], - ); - - if (!booking || booking.length === 0) { - throw new Error('Booking not found or already cancelled'); - } - - const { user_id, consultant_id, slot_id, amount: _amount } = booking[0]; - - // 2. Refund user - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - refundAmount, - user_id, - ]); - - // 3. Deduct from consultant - await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ - refundAmount, - consultant_id, - ]); - - // 4. Free up the slot - await manager.query( - 'UPDATE consulting_slots SET status = $1, booked_by = NULL, booked_at = NULL WHERE id = $2', - ['available', slot_id], - ); - - // 5. Update booking status - await manager.query( - 'UPDATE bookings SET status = $1, cancelled_at = NOW(), refund_amount = $2 WHERE id = $3', - ['cancelled', refundAmount, bookingId], - ); - - // 6. Create notifications - await manager.query( - 'INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', - [user_id, 'booking_cancelled', `Your booking has been cancelled. Refund: $${refundAmount}`], - ); - - this.logger.log(`Booking cancelled: ${bookingId}`); - - return { success: true, bookingId, refundAmount }; - }); - } - - /** - * Reschedule booking - */ - async rescheduleBooking(bookingId: string, newSlotId: string): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Get current booking - const booking = await manager.query( - 'SELECT * FROM bookings WHERE id = $1 AND status = $2 FOR UPDATE', - [bookingId, 'confirmed'], - ); - - if (!booking || booking.length === 0) { - throw new Error('Booking not found'); - } - - const { slot_id: oldSlotId, user_id } = booking[0]; - - // 2. Check new slot availability - const newSlot = await manager.query( - 'SELECT * FROM consulting_slots WHERE id = $1 AND status = $2 FOR UPDATE', - [newSlotId, 'available'], - ); - - if (!newSlot || newSlot.length === 0) { - throw new Error('New slot not available'); - } - - // 3. Free old slot - await manager.query( - 'UPDATE consulting_slots SET status = $1, booked_by = NULL WHERE id = $2', - ['available', oldSlotId], - ); - - // 4. Book new slot - await manager.query( - 'UPDATE consulting_slots SET status = $1, booked_by = $2, booked_at = NOW() WHERE id = $3', - ['booked', user_id, newSlotId], - ); - - // 5. Update booking - await manager.query( - 'UPDATE bookings SET slot_id = $1, rescheduled_at = NOW() WHERE id = $2', - [newSlotId, bookingId], - ); - - // 6. Create notification - await manager.query( - 'INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', - [user_id, 'booking_rescheduled', 'Your booking has been rescheduled'], - ); - - this.logger.log(`Booking rescheduled: ${bookingId}`); - - return { success: true, bookingId, newSlotId }; - }); - } + private readonly logger = new Logger(BookingTransactionExample.name); + constructor(private readonly transactionService: TransactionService) { } + /** + * Book a consulting session + * Ensures slot is reserved, payment is processed, and notification is sent atomically + */ + async bookConsultingSession(userId: string, consultantId: string, slotId: string, amount: number): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Check and lock the slot + const slot = await manager.query('SELECT * FROM consulting_slots WHERE id = $1 AND status = $2 FOR UPDATE', [slotId, 'available']); + if (!slot || slot.length === 0) { + throw new Error('Slot not available'); + } + // 2. Check user balance + const user = await manager.query('SELECT balance FROM users WHERE id = $1 FOR UPDATE', [ + userId, + ]); + if (!user || user.length === 0 || user[0].balance < amount) { + throw new Error('Insufficient balance'); + } + // 3. Deduct payment from user + await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ + amount, + userId, + ]); + // 4. Add payment to consultant + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + amount, + consultantId, + ]); + // 5. Mark slot as booked + await manager.query('UPDATE consulting_slots SET status = $1, booked_by = $2, booked_at = NOW() WHERE id = $3', ['booked', userId, slotId]); + // 6. Create booking record + const booking = await manager.query('INSERT INTO bookings (user_id, consultant_id, slot_id, amount, status) VALUES ($1, $2, $3, $4, $5) RETURNING *', [userId, consultantId, slotId, amount, 'confirmed']); + // 7. Create notification + await manager.query('INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', [userId, 'booking_confirmed', `Your session with consultant ${consultantId} is confirmed`]); + await manager.query('INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', [consultantId, 'new_booking', `New booking from user ${userId}`]); + this.logger.log(`Booking created: ${booking[0].id}`); + return booking[0]; + }); + } + /** + * Cancel booking with refund + */ + async cancelBooking(bookingId: string, refundAmount: number): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Get booking details + const booking = await manager.query('SELECT * FROM bookings WHERE id = $1 AND status = $2 FOR UPDATE', [bookingId, 'confirmed']); + if (!booking || booking.length === 0) { + throw new Error('Booking not found or already cancelled'); + } + const { user_id, consultant_id, slot_id, amount: _amount } = booking[0]; + // 2. Refund user + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + refundAmount, + user_id, + ]); + // 3. Deduct from consultant + await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ + refundAmount, + consultant_id, + ]); + // 4. Free up the slot + await manager.query('UPDATE consulting_slots SET status = $1, booked_by = NULL, booked_at = NULL WHERE id = $2', ['available', slot_id]); + // 5. Update booking status + await manager.query('UPDATE bookings SET status = $1, cancelled_at = NOW(), refund_amount = $2 WHERE id = $3', ['cancelled', refundAmount, bookingId]); + // 6. Create notifications + await manager.query('INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', [user_id, 'booking_cancelled', `Your booking has been cancelled. Refund: $${refundAmount}`]); + this.logger.log(`Booking cancelled: ${bookingId}`); + return { success: true, bookingId, refundAmount }; + }); + } + /** + * Reschedule booking + */ + async rescheduleBooking(bookingId: string, newSlotId: string): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Get current booking + const booking = await manager.query('SELECT * FROM bookings WHERE id = $1 AND status = $2 FOR UPDATE', [bookingId, 'confirmed']); + if (!booking || booking.length === 0) { + throw new Error('Booking not found'); + } + const { slot_id: oldSlotId, user_id } = booking[0]; + // 2. Check new slot availability + const newSlot = await manager.query('SELECT * FROM consulting_slots WHERE id = $1 AND status = $2 FOR UPDATE', [newSlotId, 'available']); + if (!newSlot || newSlot.length === 0) { + throw new Error('New slot not available'); + } + // 3. Free old slot + await manager.query('UPDATE consulting_slots SET status = $1, booked_by = NULL WHERE id = $2', ['available', oldSlotId]); + // 4. Book new slot + await manager.query('UPDATE consulting_slots SET status = $1, booked_by = $2, booked_at = NOW() WHERE id = $3', ['booked', user_id, newSlotId]); + // 5. Update booking + await manager.query('UPDATE bookings SET slot_id = $1, rescheduled_at = NOW() WHERE id = $2', [newSlotId, bookingId]); + // 6. Create notification + await manager.query('INSERT INTO notifications (user_id, type, message) VALUES ($1, $2, $3)', [user_id, 'booking_rescheduled', 'Your booking has been rescheduled']); + this.logger.log(`Booking rescheduled: ${bookingId}`); + return { success: true, bookingId, newSlotId }; + }); + } } diff --git a/src/common/database/examples/payment-transaction.example.ts b/src/common/database/examples/payment-transaction.example.ts index ff96302c..5dfbe67e 100644 --- a/src/common/database/examples/payment-transaction.example.ts +++ b/src/common/database/examples/payment-transaction.example.ts @@ -1,163 +1,103 @@ import { Injectable, Logger } from '@nestjs/common'; import { TransactionService } from '../transaction.service'; // import { Transactional } from '../transactional.decorator'; - /** * Example: Payment Transaction * Demonstrates atomic payment processing with transaction management */ @Injectable() export class PaymentTransactionExample { - private readonly logger = new Logger(PaymentTransactionExample.name); - - constructor( - private readonly transactionService: TransactionService, - // @InjectRepository(Payment) private paymentRepo: Repository, - // @InjectRepository(User) private userRepo: Repository, - // @InjectRepository(Transaction) private transactionRepo: Repository, - ) {} - - /** - * Process payment with transaction - * Ensures all steps succeed or all fail together - */ - async processPayment(userId: string, amount: number, recipientId: string): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Deduct from sender - const sender = await manager.query( - 'UPDATE users SET balance = balance - $1 WHERE id = $2 AND balance >= $1 RETURNING *', - [amount, userId], - ); - - if (!sender || sender.length === 0) { - throw new Error('Insufficient balance'); - } - - // 2. Add to recipient - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - amount, - recipientId, - ]); - - // 3. Create payment record - const payment = await manager.query( - 'INSERT INTO payments (user_id, recipient_id, amount, status) VALUES ($1, $2, $3, $4) RETURNING *', - [userId, recipientId, amount, 'completed'], - ); - - // 4. Create transaction log - await manager.query( - 'INSERT INTO transaction_logs (payment_id, type, amount) VALUES ($1, $2, $3)', - [payment[0].id, 'payment', amount], - ); - - this.logger.log(`Payment processed: ${amount} from ${userId} to ${recipientId}`); - - return payment[0]; - }); - } - - /** - * Process payment with retry on deadlock - */ - async processPaymentWithRetry(userId: string, amount: number, recipientId: string): Promise { - return this.transactionService.runWithRetry( - async (manager) => { - return this.processPaymentLogic(manager, userId, amount, recipientId); - }, - 3, // max retries - 1000, // initial delay - ); - } - - /** - * Process payment with serializable isolation - * Prevents concurrent modifications - */ - async processPaymentSerializable( - userId: string, - amount: number, - recipientId: string, - ): Promise { - return this.transactionService.runWithIsolationLevel('SERIALIZABLE', async (manager) => { - return this.processPaymentLogic(manager, userId, amount, recipientId); - }); - } - - /** - * Refund payment (rollback scenario) - */ - async refundPayment(paymentId: string): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Get payment details - const payment = await manager.query('SELECT * FROM payments WHERE id = $1 AND status = $2', [ - paymentId, - 'completed', - ]); - - if (!payment || payment.length === 0) { - throw new Error('Payment not found or already refunded'); - } - - const { user_id, recipient_id, amount } = payment[0]; - - // 2. Reverse the payment - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - amount, - user_id, - ]); - - await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ - amount, - recipient_id, - ]); - - // 3. Update payment status - await manager.query('UPDATE payments SET status = $1, refunded_at = NOW() WHERE id = $2', [ - 'refunded', - paymentId, - ]); - - // 4. Create refund log - await manager.query( - 'INSERT INTO transaction_logs (payment_id, type, amount) VALUES ($1, $2, $3)', - [paymentId, 'refund', amount], - ); - - this.logger.log(`Payment refunded: ${paymentId}`); - - return { success: true, paymentId }; - }); - } - - /** - * Helper method for payment logic - */ - private async processPaymentLogic( - manager: any, - userId: string, - amount: number, - recipientId: string, - ): Promise { - const sender = await manager.query( - 'UPDATE users SET balance = balance - $1 WHERE id = $2 AND balance >= $1 RETURNING *', - [amount, userId], - ); - - if (!sender || sender.length === 0) { - throw new Error('Insufficient balance'); + private readonly logger = new Logger(PaymentTransactionExample.name); + constructor(private readonly transactionService: TransactionService) { } + /** + * Process payment with transaction + * Ensures all steps succeed or all fail together + */ + async processPayment(userId: string, amount: number, recipientId: string): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Deduct from sender + const sender = await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2 AND balance >= $1 RETURNING *', [amount, userId]); + if (!sender || sender.length === 0) { + throw new Error('Insufficient balance'); + } + // 2. Add to recipient + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + amount, + recipientId, + ]); + // 3. Create payment record + const payment = await manager.query('INSERT INTO payments (user_id, recipient_id, amount, status) VALUES ($1, $2, $3, $4) RETURNING *', [userId, recipientId, amount, 'completed']); + // 4. Create transaction log + await manager.query('INSERT INTO transaction_logs (payment_id, type, amount) VALUES ($1, $2, $3)', [payment[0].id, 'payment', amount]); + this.logger.log(`Payment processed: ${amount} from ${userId} to ${recipientId}`); + return payment[0]; + }); + } + /** + * Process payment with retry on deadlock + */ + async processPaymentWithRetry(userId: string, amount: number, recipientId: string): Promise { + return this.transactionService.runWithRetry(async (manager) => { + return this.processPaymentLogic(manager, userId, amount, recipientId); + }, 3, // max retries + 1000); + } + /** + * Process payment with serializable isolation + * Prevents concurrent modifications + */ + async processPaymentSerializable(userId: string, amount: number, recipientId: string): Promise { + return this.transactionService.runWithIsolationLevel('SERIALIZABLE', async (manager) => { + return this.processPaymentLogic(manager, userId, amount, recipientId); + }); + } + /** + * Refund payment (rollback scenario) + */ + async refundPayment(paymentId: string): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Get payment details + const payment = await manager.query('SELECT * FROM payments WHERE id = $1 AND status = $2', [ + paymentId, + 'completed', + ]); + if (!payment || payment.length === 0) { + throw new Error('Payment not found or already refunded'); + } + const { user_id, recipient_id, amount } = payment[0]; + // 2. Reverse the payment + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + amount, + user_id, + ]); + await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [ + amount, + recipient_id, + ]); + // 3. Update payment status + await manager.query('UPDATE payments SET status = $1, refunded_at = NOW() WHERE id = $2', [ + 'refunded', + paymentId, + ]); + // 4. Create refund log + await manager.query('INSERT INTO transaction_logs (payment_id, type, amount) VALUES ($1, $2, $3)', [paymentId, 'refund', amount]); + this.logger.log(`Payment refunded: ${paymentId}`); + return { success: true, paymentId }; + }); + } + /** + * Helper method for payment logic + */ + private async processPaymentLogic(manager: unknown, userId: string, amount: number, recipientId: string): Promise { + const sender = await manager.query('UPDATE users SET balance = balance - $1 WHERE id = $2 AND balance >= $1 RETURNING *', [amount, userId]); + if (!sender || sender.length === 0) { + throw new Error('Insufficient balance'); + } + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + amount, + recipientId, + ]); + const payment = await manager.query('INSERT INTO payments (user_id, recipient_id, amount, status) VALUES ($1, $2, $3, $4) RETURNING *', [userId, recipientId, amount, 'completed']); + return payment[0]; } - - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - amount, - recipientId, - ]); - - const payment = await manager.query( - 'INSERT INTO payments (user_id, recipient_id, amount, status) VALUES ($1, $2, $3, $4) RETURNING *', - [userId, recipientId, amount, 'completed'], - ); - - return payment[0]; - } } diff --git a/src/common/database/examples/voting-transaction.example.ts b/src/common/database/examples/voting-transaction.example.ts index d4d90cc9..0c44e03e 100644 --- a/src/common/database/examples/voting-transaction.example.ts +++ b/src/common/database/examples/voting-transaction.example.ts @@ -1,293 +1,191 @@ import { Injectable, Logger } from '@nestjs/common'; import { TransactionService } from '../transaction.service'; - /** * Example: DAO Voting Transaction * Demonstrates atomic voting operations with proper locking */ @Injectable() export class VotingTransactionExample { - private readonly logger = new Logger(VotingTransactionExample.name); - - constructor(private readonly transactionService: TransactionService) {} - - /** - * Cast a vote in DAO proposal - * Ensures vote is recorded, counts are updated, and user can't double-vote - */ - async castVote( - userId: string, - proposalId: string, - voteType: 'for' | 'against' | 'abstain', - votingPower: number, - ): Promise { - return this.transactionService.runWithIsolationLevel( - 'SERIALIZABLE', // Prevent concurrent voting issues - async (manager) => { - // 1. Check if user already voted - const existingVote = await manager.query( - 'SELECT * FROM votes WHERE user_id = $1 AND proposal_id = $2', - [userId, proposalId], - ); - - if (existingVote && existingVote.length > 0) { - throw new Error('User has already voted on this proposal'); - } - - // 2. Check if proposal is still active - const proposal = await manager.query( - 'SELECT * FROM proposals WHERE id = $1 AND status = $2 FOR UPDATE', - [proposalId, 'active'], - ); - - if (!proposal || proposal.length === 0) { - throw new Error('Proposal not found or not active'); - } - - // 3. Verify user has voting power - const user = await manager.query('SELECT voting_power FROM users WHERE id = $1', [userId]); - - if (!user || user.length === 0 || user[0].voting_power < votingPower) { - throw new Error('Insufficient voting power'); - } - - // 4. Record the vote - const vote = await manager.query( - 'INSERT INTO votes (user_id, proposal_id, vote_type, voting_power) VALUES ($1, $2, $3, $4) RETURNING *', - [userId, proposalId, voteType, votingPower], - ); - - // 5. Update proposal vote counts - const updateField = - voteType === 'for' - ? 'votes_for' - : voteType === 'against' - ? 'votes_against' - : 'votes_abstain'; - await manager.query( - `UPDATE proposals SET ${updateField} = ${updateField} + $1, total_votes = total_votes + $1 WHERE id = $2`, - [votingPower, proposalId], - ); - - // 6. Check if proposal reached quorum - const updatedProposal = await manager.query('SELECT * FROM proposals WHERE id = $1', [ - proposalId, - ]); - - const { total_votes, quorum_required } = updatedProposal[0]; - - if (total_votes >= quorum_required) { - await manager.query('UPDATE proposals SET quorum_reached = true WHERE id = $1', [ - proposalId, - ]); - - this.logger.log(`Proposal ${proposalId} reached quorum`); + private readonly logger = new Logger(VotingTransactionExample.name); + constructor(private readonly transactionService: TransactionService) { } + /** + * Cast a vote in DAO proposal + * Ensures vote is recorded, counts are updated, and user can't double-vote + */ + async castVote(userId: string, proposalId: string, voteType: 'for' | 'against' | 'abstain', votingPower: number): Promise { + return this.transactionService.runWithIsolationLevel('SERIALIZABLE', // Prevent concurrent voting issues + async (manager) => { + // 1. Check if user already voted + const existingVote = await manager.query('SELECT * FROM votes WHERE user_id = $1 AND proposal_id = $2', [userId, proposalId]); + if (existingVote && existingVote.length > 0) { + throw new Error('User has already voted on this proposal'); + } + // 2. Check if proposal is still active + const proposal = await manager.query('SELECT * FROM proposals WHERE id = $1 AND status = $2 FOR UPDATE', [proposalId, 'active']); + if (!proposal || proposal.length === 0) { + throw new Error('Proposal not found or not active'); + } + // 3. Verify user has voting power + const user = await manager.query('SELECT voting_power FROM users WHERE id = $1', [userId]); + if (!user || user.length === 0 || user[0].voting_power < votingPower) { + throw new Error('Insufficient voting power'); + } + // 4. Record the vote + const vote = await manager.query('INSERT INTO votes (user_id, proposal_id, vote_type, voting_power) VALUES ($1, $2, $3, $4) RETURNING *', [userId, proposalId, voteType, votingPower]); + // 5. Update proposal vote counts + const updateField = voteType === 'for' + ? 'votes_for' + : voteType === 'against' + ? 'votes_against' + : 'votes_abstain'; + await manager.query(`UPDATE proposals SET ${updateField} = ${updateField} + $1, total_votes = total_votes + $1 WHERE id = $2`, [votingPower, proposalId]); + // 6. Check if proposal reached quorum + const updatedProposal = await manager.query('SELECT * FROM proposals WHERE id = $1', [ + proposalId, + ]); + const { total_votes, quorum_required } = updatedProposal[0]; + if (total_votes >= quorum_required) { + await manager.query('UPDATE proposals SET quorum_reached = true WHERE id = $1', [ + proposalId, + ]); + this.logger.log(`Proposal ${proposalId} reached quorum`); + } + // 7. Create activity log + await manager.query('INSERT INTO activity_logs (user_id, action, entity_type, entity_id) VALUES ($1, $2, $3, $4)', [userId, 'vote_cast', 'proposal', proposalId]); + this.logger.log(`Vote cast: ${userId} voted ${voteType} on ${proposalId}`); + return vote[0]; + }); + } + /** + * Change vote (if allowed) + */ + async changeVote(userId: string, proposalId: string, newVoteType: 'for' | 'against' | 'abstain'): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Get existing vote + const existingVote = await manager.query('SELECT * FROM votes WHERE user_id = $1 AND proposal_id = $2 FOR UPDATE', [userId, proposalId]); + if (!existingVote || existingVote.length === 0) { + throw new Error('No existing vote found'); + } + const { vote_type: oldVoteType, voting_power } = existingVote[0]; + if (oldVoteType === newVoteType) { + throw new Error('Vote type is the same'); + } + // 2. Check if proposal allows vote changes + const proposal = await manager.query('SELECT * FROM proposals WHERE id = $1 AND status = $2 AND allow_vote_change = true FOR UPDATE', [proposalId, 'active']); + if (!proposal || proposal.length === 0) { + throw new Error('Proposal does not allow vote changes or is not active'); + } + // 3. Update vote counts - remove old vote + const oldField = oldVoteType === 'for' + ? 'votes_for' + : oldVoteType === 'against' + ? 'votes_against' + : 'votes_abstain'; + await manager.query(`UPDATE proposals SET ${oldField} = ${oldField} - $1 WHERE id = $2`, [ + voting_power, + proposalId, + ]); + // 4. Update vote counts - add new vote + const newField = newVoteType === 'for' + ? 'votes_for' + : newVoteType === 'against' + ? 'votes_against' + : 'votes_abstain'; + await manager.query(`UPDATE proposals SET ${newField} = ${newField} + $1 WHERE id = $2`, [ + voting_power, + proposalId, + ]); + // 5. Update vote record + await manager.query('UPDATE votes SET vote_type = $1, changed_at = NOW() WHERE user_id = $2 AND proposal_id = $3', [newVoteType, userId, proposalId]); + // 6. Log the change + await manager.query('INSERT INTO activity_logs (user_id, action, entity_type, entity_id, metadata) VALUES ($1, $2, $3, $4, $5)', [ + userId, + 'vote_changed', + 'proposal', + proposalId, + JSON.stringify({ from: oldVoteType, to: newVoteType }), + ]); + this.logger.log(`Vote changed: ${userId} changed from ${oldVoteType} to ${newVoteType} on ${proposalId}`); + return { success: true, oldVoteType, newVoteType }; + }); + } + /** + * Execute proposal (after voting period ends) + */ + async executeProposal(proposalId: string): Promise { + return this.transactionService.runInTransaction(async (manager) => { + // 1. Get proposal with lock + const proposal = await manager.query('SELECT * FROM proposals WHERE id = $1 AND status = $2 FOR UPDATE', [proposalId, 'active']); + if (!proposal || proposal.length === 0) { + throw new Error('Proposal not found or not active'); + } + const { votes_for, votes_against, total_votes, quorum_required, approval_threshold } = proposal[0]; + // 2. Check if voting period ended + const now = new Date(); + const endTime = new Date(proposal[0].voting_end_time); + if (now < endTime) { + throw new Error('Voting period has not ended'); + } + // 3. Check quorum + if (total_votes < quorum_required) { + await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ + 'failed_quorum', + proposalId, + ]); + return { success: false, reason: 'Quorum not reached' }; + } + // 4. Check approval threshold + const approvalRate = (votes_for / total_votes) * 100; + if (approvalRate < approval_threshold) { + await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ + 'rejected', + proposalId, + ]); + return { success: false, reason: 'Approval threshold not met' }; + } + // 5. Execute proposal actions + const actions = await manager.query('SELECT * FROM proposal_actions WHERE proposal_id = $1 ORDER BY execution_order', [proposalId]); + for (const action of actions) { + // Execute each action (transfer funds, update settings, etc.) + await this.executeProposalAction(manager, action); + } + // 6. Mark proposal as executed + await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ + 'executed', + proposalId, + ]); + // 7. Log execution + await manager.query('INSERT INTO activity_logs (action, entity_type, entity_id, metadata) VALUES ($1, $2, $3, $4)', [ + 'proposal_executed', + 'proposal', + proposalId, + JSON.stringify({ votes_for, votes_against, total_votes }), + ]); + this.logger.log(`Proposal executed: ${proposalId}`); + return { success: true, proposalId, status: 'executed' }; + }); + } + /** + * Helper to execute individual proposal actions + */ + private async executeProposalAction(manager: unknown, action: unknown): Promise { + const { action_type, parameters } = action; + switch (action_type) { + case 'transfer_funds': + await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ + parameters.amount, + parameters.recipient, + ]); + break; + case 'update_setting': + await manager.query('UPDATE settings SET value = $1 WHERE key = $2', [ + parameters.value, + parameters.key, + ]); + break; + // Add more action types as needed + default: + this.logger.warn(`Unknown action type: ${action_type}`); } - - // 7. Create activity log - await manager.query( - 'INSERT INTO activity_logs (user_id, action, entity_type, entity_id) VALUES ($1, $2, $3, $4)', - [userId, 'vote_cast', 'proposal', proposalId], - ); - - this.logger.log(`Vote cast: ${userId} voted ${voteType} on ${proposalId}`); - - return vote[0]; - }, - ); - } - - /** - * Change vote (if allowed) - */ - async changeVote( - userId: string, - proposalId: string, - newVoteType: 'for' | 'against' | 'abstain', - ): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Get existing vote - const existingVote = await manager.query( - 'SELECT * FROM votes WHERE user_id = $1 AND proposal_id = $2 FOR UPDATE', - [userId, proposalId], - ); - - if (!existingVote || existingVote.length === 0) { - throw new Error('No existing vote found'); - } - - const { vote_type: oldVoteType, voting_power } = existingVote[0]; - - if (oldVoteType === newVoteType) { - throw new Error('Vote type is the same'); - } - - // 2. Check if proposal allows vote changes - const proposal = await manager.query( - 'SELECT * FROM proposals WHERE id = $1 AND status = $2 AND allow_vote_change = true FOR UPDATE', - [proposalId, 'active'], - ); - - if (!proposal || proposal.length === 0) { - throw new Error('Proposal does not allow vote changes or is not active'); - } - - // 3. Update vote counts - remove old vote - const oldField = - oldVoteType === 'for' - ? 'votes_for' - : oldVoteType === 'against' - ? 'votes_against' - : 'votes_abstain'; - await manager.query(`UPDATE proposals SET ${oldField} = ${oldField} - $1 WHERE id = $2`, [ - voting_power, - proposalId, - ]); - - // 4. Update vote counts - add new vote - const newField = - newVoteType === 'for' - ? 'votes_for' - : newVoteType === 'against' - ? 'votes_against' - : 'votes_abstain'; - await manager.query(`UPDATE proposals SET ${newField} = ${newField} + $1 WHERE id = $2`, [ - voting_power, - proposalId, - ]); - - // 5. Update vote record - await manager.query( - 'UPDATE votes SET vote_type = $1, changed_at = NOW() WHERE user_id = $2 AND proposal_id = $3', - [newVoteType, userId, proposalId], - ); - - // 6. Log the change - await manager.query( - 'INSERT INTO activity_logs (user_id, action, entity_type, entity_id, metadata) VALUES ($1, $2, $3, $4, $5)', - [ - userId, - 'vote_changed', - 'proposal', - proposalId, - JSON.stringify({ from: oldVoteType, to: newVoteType }), - ], - ); - - this.logger.log( - `Vote changed: ${userId} changed from ${oldVoteType} to ${newVoteType} on ${proposalId}`, - ); - - return { success: true, oldVoteType, newVoteType }; - }); - } - - /** - * Execute proposal (after voting period ends) - */ - async executeProposal(proposalId: string): Promise { - return this.transactionService.runInTransaction(async (manager) => { - // 1. Get proposal with lock - const proposal = await manager.query( - 'SELECT * FROM proposals WHERE id = $1 AND status = $2 FOR UPDATE', - [proposalId, 'active'], - ); - - if (!proposal || proposal.length === 0) { - throw new Error('Proposal not found or not active'); - } - - const { votes_for, votes_against, total_votes, quorum_required, approval_threshold } = - proposal[0]; - - // 2. Check if voting period ended - const now = new Date(); - const endTime = new Date(proposal[0].voting_end_time); - - if (now < endTime) { - throw new Error('Voting period has not ended'); - } - - // 3. Check quorum - if (total_votes < quorum_required) { - await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ - 'failed_quorum', - proposalId, - ]); - - return { success: false, reason: 'Quorum not reached' }; - } - - // 4. Check approval threshold - const approvalRate = (votes_for / total_votes) * 100; - - if (approvalRate < approval_threshold) { - await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ - 'rejected', - proposalId, - ]); - - return { success: false, reason: 'Approval threshold not met' }; - } - - // 5. Execute proposal actions - const actions = await manager.query( - 'SELECT * FROM proposal_actions WHERE proposal_id = $1 ORDER BY execution_order', - [proposalId], - ); - - for (const action of actions) { - // Execute each action (transfer funds, update settings, etc.) - await this.executeProposalAction(manager, action); - } - - // 6. Mark proposal as executed - await manager.query('UPDATE proposals SET status = $1, executed_at = NOW() WHERE id = $2', [ - 'executed', - proposalId, - ]); - - // 7. Log execution - await manager.query( - 'INSERT INTO activity_logs (action, entity_type, entity_id, metadata) VALUES ($1, $2, $3, $4)', - [ - 'proposal_executed', - 'proposal', - proposalId, - JSON.stringify({ votes_for, votes_against, total_votes }), - ], - ); - - this.logger.log(`Proposal executed: ${proposalId}`); - - return { success: true, proposalId, status: 'executed' }; - }); - } - - /** - * Helper to execute individual proposal actions - */ - private async executeProposalAction(manager: any, action: any): Promise { - const { action_type, parameters } = action; - - switch (action_type) { - case 'transfer_funds': - await manager.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [ - parameters.amount, - parameters.recipient, - ]); - break; - - case 'update_setting': - await manager.query('UPDATE settings SET value = $1 WHERE key = $2', [ - parameters.value, - parameters.key, - ]); - break; - - // Add more action types as needed - default: - this.logger.warn(`Unknown action type: ${action_type}`); } - } } diff --git a/src/common/database/transaction-helper.service.ts b/src/common/database/transaction-helper.service.ts index 3ca5cd8a..2048901e 100644 --- a/src/common/database/transaction-helper.service.ts +++ b/src/common/database/transaction-helper.service.ts @@ -1,147 +1,129 @@ import { Injectable, Logger } from '@nestjs/common'; import { DataSource, EntityManager } from 'typeorm'; - /** * Transaction helper utilities */ @Injectable() export class TransactionHelperService { - private readonly logger = new Logger(TransactionHelperService.name); - - constructor(private readonly dataSource: DataSource) {} - - /** - * Execute multiple operations in a single transaction - * Useful for complex business operations - */ - async executeInTransaction( - operations: Array<(manager: EntityManager) => Promise>, - ): Promise { - const queryRunner = this.dataSource.createQueryRunner(); - await queryRunner.connect(); - await queryRunner.startTransaction(); - - try { - this.logger.debug(`Starting transaction with ${operations.length} operations`); - const results = await Promise.all( - operations.map((operation) => operation(queryRunner.manager)), - ); - await queryRunner.commitTransaction(); - this.logger.debug('Transaction committed successfully'); - return results; - } catch (error) { - await queryRunner.rollbackTransaction(); - this.logger.error('Transaction rolled back:', error); - throw error; - } finally { - await queryRunner.release(); + private readonly logger = new Logger(TransactionHelperService.name); + constructor(private readonly dataSource: DataSource) { } + /** + * Execute multiple operations in a single transaction + * Useful for complex business operations + */ + async executeInTransaction(operations: Array<(manager: EntityManager) => Promise>): Promise { + const queryRunner = this.dataSource.createQueryRunner(); + await queryRunner.connect(); + await queryRunner.startTransaction(); + try { + this.logger.debug(`Starting transaction with ${operations.length} operations`); + const results = await Promise.all(operations.map((operation) => operation(queryRunner.manager))); + await queryRunner.commitTransaction(); + this.logger.debug('Transaction committed successfully'); + return results; + } + catch (error) { + await queryRunner.rollbackTransaction(); + this.logger.error('Transaction rolled back:', error); + throw error; + } + finally { + await queryRunner.release(); + } } - } - - /** - * Execute operations with conditional logic - */ - async executeWithRollback( - operations: Array<{ - operation: (manager: EntityManager) => Promise; - rollback?: (manager: EntityManager) => Promise; - condition?: () => boolean; - }>, - ): Promise { - const queryRunner = this.dataSource.createQueryRunner(); - await queryRunner.connect(); - await queryRunner.startTransaction(); - - try { - const results: T[] = []; - - for (const { operation, rollback, condition } of operations) { - if (condition && !condition()) { - this.logger.debug('Skipping operation due to condition'); - continue; + /** + * Execute operations with conditional logic + */ + async executeWithRollback(operations: Array<{ + operation: (manager: EntityManager) => Promise; + rollback?: (manager: EntityManager) => Promise; + condition?: () => boolean; + }>): Promise { + const queryRunner = this.dataSource.createQueryRunner(); + await queryRunner.connect(); + await queryRunner.startTransaction(); + try { + const results: T[] = []; + for (const { operation, rollback, condition } of operations) { + if (condition && !condition()) { + this.logger.debug('Skipping operation due to condition'); + continue; + } + const result = await operation(queryRunner.manager); + results.push(result); + // Store rollback function if provided + if (rollback) { + // In a real implementation, you'd store rollback functions + // This is a simplified version + } + } + await queryRunner.commitTransaction(); + this.logger.debug(`Transaction committed with ${results.length} operations`); + return results; + } + catch (error) { + await queryRunner.rollbackTransaction(); + this.logger.error('Transaction rolled back:', error); + // Execute rollback functions if available + for (const { rollback } of operations) { + if (rollback) { + try { + await rollback(queryRunner.manager); + } + catch (rollbackError) { + this.logger.error('Rollback operation failed:', rollbackError); + } + } + } + throw error; + } + finally { + await queryRunner.release(); } - - const result = await operation(queryRunner.manager); - results.push(result); - - // Store rollback function if provided - if (rollback) { - // In a real implementation, you'd store rollback functions - // This is a simplified version + } + /** + * Create savepoint for nested transactions + */ + async createSavepoint(manager: EntityManager, savepointName: string): Promise { + await manager.query(`SAVEPOINT ${savepointName}`); + this.logger.debug(`Created savepoint: ${savepointName}`); + } + /** + * Rollback to savepoint + */ + async rollbackToSavepoint(manager: EntityManager, savepointName: string): Promise { + await manager.query(`ROLLBACK TO SAVEPOINT ${savepointName}`); + this.logger.debug(`Rolled back to savepoint: ${savepointName}`); + } + /** + * Release savepoint + */ + async releaseSavepoint(manager: EntityManager, savepointName: string): Promise { + await manager.query(`RELEASE SAVEPOINT ${savepointName}`); + this.logger.debug(`Released savepoint: ${savepointName}`); + } + /** + * Check if transaction is active + */ + isInTransaction(manager: EntityManager): boolean { + return manager.queryRunner?.isTransactionActive || false; + } + /** + * Get transaction isolation level + */ + async getIsolationLevel(manager: EntityManager): Promise { + try { + const result = await manager.query('SHOW TRANSACTION ISOLATION LEVEL'); + return result[0]?.level || 'READ COMMITTED'; } - } - - await queryRunner.commitTransaction(); - this.logger.debug(`Transaction committed with ${results.length} operations`); - return results; - } catch (error) { - await queryRunner.rollbackTransaction(); - this.logger.error('Transaction rolled back:', error); - - // Execute rollback functions if available - for (const { rollback } of operations) { - if (rollback) { - try { - await rollback(queryRunner.manager); - } catch (rollbackError) { - this.logger.error('Rollback operation failed:', rollbackError); - } + catch { + return 'READ COMMITTED'; } - } - - throw error; - } finally { - await queryRunner.release(); } - } - - /** - * Create savepoint for nested transactions - */ - async createSavepoint(manager: EntityManager, savepointName: string): Promise { - await manager.query(`SAVEPOINT ${savepointName}`); - this.logger.debug(`Created savepoint: ${savepointName}`); - } - - /** - * Rollback to savepoint - */ - async rollbackToSavepoint(manager: EntityManager, savepointName: string): Promise { - await manager.query(`ROLLBACK TO SAVEPOINT ${savepointName}`); - this.logger.debug(`Rolled back to savepoint: ${savepointName}`); - } - - /** - * Release savepoint - */ - async releaseSavepoint(manager: EntityManager, savepointName: string): Promise { - await manager.query(`RELEASE SAVEPOINT ${savepointName}`); - this.logger.debug(`Released savepoint: ${savepointName}`); - } - - /** - * Check if transaction is active - */ - isInTransaction(manager: EntityManager): boolean { - return manager.queryRunner?.isTransactionActive || false; - } - - /** - * Get transaction isolation level - */ - async getIsolationLevel(manager: EntityManager): Promise { - try { - const result = await manager.query('SHOW TRANSACTION ISOLATION LEVEL'); - return result[0]?.level || 'READ COMMITTED'; - } catch { - return 'READ COMMITTED'; + /** + * Set transaction timeout + */ + async setTransactionTimeout(manager: EntityManager, timeoutMs: number): Promise { + await manager.query(`SET LOCK_TIMEOUT ${timeoutMs}`); } - } - - /** - * Set transaction timeout - */ - async setTransactionTimeout(manager: EntityManager, timeoutMs: number): Promise { - await manager.query(`SET LOCK_TIMEOUT ${timeoutMs}`); - } } diff --git a/src/common/database/transaction.service.ts b/src/common/database/transaction.service.ts index d00e258e..12424059 100644 --- a/src/common/database/transaction.service.ts +++ b/src/common/database/transaction.service.ts @@ -1,266 +1,224 @@ import { Injectable, Logger } from '@nestjs/common'; import { DataSource, EntityManager, QueryRunner } from 'typeorm'; - /** * Transaction monitoring interface */ export interface TransactionMetrics { - transactionId: string; - startTime: Date; - endTime?: Date; - duration?: number; - status: 'STARTED' | 'COMMITTED' | 'ROLLED_BACK'; - operations: string[]; - error?: string; + transactionId: string; + startTime: Date; + endTime?: Date; + duration?: number; + status: 'STARTED' | 'COMMITTED' | 'ROLLED_BACK'; + operations: string[]; + error?: string; } - /** * Transaction Service * Provides robust transaction management for critical operations */ @Injectable() export class TransactionService { - private readonly logger = new Logger(TransactionService.name); - private readonly activeTransactions = new Map(); - - constructor(private readonly dataSource: DataSource) {} - - /** - * Execute operations within a transaction - * Automatically handles commit and rollback - */ - async runInTransaction(operation: (manager: EntityManager) => Promise): Promise { - const queryRunner = this.dataSource.createQueryRunner(); - await queryRunner.connect(); - await queryRunner.startTransaction(); - - const transactionId = this.generateTransactionId(); - const startTime = new Date(); - const metrics: TransactionMetrics = { - transactionId, - startTime, - status: 'STARTED', - operations: [], - }; - this.activeTransactions.set(transactionId, metrics); - - try { - this.logger.debug(`Transaction ${transactionId} started`); - this.logger.log(`Transaction ${transactionId}: Starting operation execution`); - - const result = await operation(queryRunner.manager); - - await queryRunner.commitTransaction(); - const endTime = new Date(); - const duration = endTime.getTime() - startTime.getTime(); - - metrics.endTime = endTime; - metrics.duration = duration; - metrics.status = 'COMMITTED'; - - this.logger.log(`Transaction ${transactionId} committed successfully in ${duration}ms`); - return result; - } catch (error) { - await queryRunner.rollbackTransaction(); - const endTime = new Date(); - const duration = endTime.getTime() - startTime.getTime(); - - metrics.endTime = endTime; - metrics.duration = duration; - metrics.status = 'ROLLED_BACK'; - metrics.error = error instanceof Error ? error.message : String(error); - - this.logger.error(`Transaction ${transactionId} rolled back after ${duration}ms:`, error); - throw error; - } finally { - await queryRunner.release(); - this.logger.debug(`Transaction ${transactionId} resources released`); + private readonly logger = new Logger(TransactionService.name); + private readonly activeTransactions = new Map(); + constructor(private readonly dataSource: DataSource) { } + /** + * Execute operations within a transaction + * Automatically handles commit and rollback + */ + async runInTransaction(operation: (manager: EntityManager) => Promise): Promise { + const queryRunner = this.dataSource.createQueryRunner(); + await queryRunner.connect(); + await queryRunner.startTransaction(); + const transactionId = this.generateTransactionId(); + const startTime = new Date(); + const metrics: TransactionMetrics = { + transactionId, + startTime, + status: 'STARTED', + operations: [], + }; + this.activeTransactions.set(transactionId, metrics); + try { + this.logger.debug(`Transaction ${transactionId} started`); + this.logger.log(`Transaction ${transactionId}: Starting operation execution`); + const result = await operation(queryRunner.manager); + await queryRunner.commitTransaction(); + const endTime = new Date(); + const duration = endTime.getTime() - startTime.getTime(); + metrics.endTime = endTime; + metrics.duration = duration; + metrics.status = 'COMMITTED'; + this.logger.log(`Transaction ${transactionId} committed successfully in ${duration}ms`); + return result; + } + catch (error) { + await queryRunner.rollbackTransaction(); + const endTime = new Date(); + const duration = endTime.getTime() - startTime.getTime(); + metrics.endTime = endTime; + metrics.duration = duration; + metrics.status = 'ROLLED_BACK'; + metrics.error = error instanceof Error ? error.message : String(error); + this.logger.error(`Transaction ${transactionId} rolled back after ${duration}ms:`, error); + throw error; + } + finally { + await queryRunner.release(); + this.logger.debug(`Transaction ${transactionId} resources released`); + } } - } - - /** - * Execute operations with manual transaction control - * Useful for complex scenarios requiring custom logic - */ - async withTransaction(callback: (queryRunner: QueryRunner) => Promise): Promise { - const queryRunner = this.dataSource.createQueryRunner(); - await queryRunner.connect(); - - try { - return await callback(queryRunner); - } finally { - await queryRunner.release(); + /** + * Execute operations with manual transaction control + * Useful for complex scenarios requiring custom logic + */ + async withTransaction(callback: (queryRunner: QueryRunner) => Promise): Promise { + const queryRunner = this.dataSource.createQueryRunner(); + await queryRunner.connect(); + try { + return await callback(queryRunner); + } + finally { + await queryRunner.release(); + } } - } - - /** - * Execute operations with isolation level - */ - async runWithIsolationLevel( - isolationLevel: 'READ UNCOMMITTED' | 'READ COMMITTED' | 'REPEATABLE READ' | 'SERIALIZABLE', - operation: (manager: EntityManager) => Promise, - ): Promise { - const queryRunner = this.dataSource.createQueryRunner(); - await queryRunner.connect(); - - try { - await queryRunner.query(`SET TRANSACTION ISOLATION LEVEL ${isolationLevel}`); - await queryRunner.startTransaction(); - - this.logger.debug(`Transaction started with isolation level: ${isolationLevel}`); - - const result = await operation(queryRunner.manager); - - await queryRunner.commitTransaction(); - this.logger.debug('Transaction committed successfully'); - - return result; - } catch (error) { - await queryRunner.rollbackTransaction(); - this.logger.error('Transaction rolled back due to error:', error); - throw error; - } finally { - await queryRunner.release(); + /** + * Execute operations with isolation level + */ + async runWithIsolationLevel(isolationLevel: 'READ UNCOMMITTED' | 'READ COMMITTED' | 'REPEATABLE READ' | 'SERIALIZABLE', operation: (manager: EntityManager) => Promise): Promise { + const queryRunner = this.dataSource.createQueryRunner(); + await queryRunner.connect(); + try { + await queryRunner.query(`SET TRANSACTION ISOLATION LEVEL ${isolationLevel}`); + await queryRunner.startTransaction(); + this.logger.debug(`Transaction started with isolation level: ${isolationLevel}`); + const result = await operation(queryRunner.manager); + await queryRunner.commitTransaction(); + this.logger.debug('Transaction committed successfully'); + return result; + } + catch (error) { + await queryRunner.rollbackTransaction(); + this.logger.error('Transaction rolled back due to error:', error); + throw error; + } + finally { + await queryRunner.release(); + } } - } - - /** - * Execute operations with retry logic - */ - async runWithRetry( - operation: (manager: EntityManager) => Promise, - maxRetries: number = 3, - retryDelay: number = 1000, - ): Promise { - let lastError: Error | undefined; - - for (let attempt = 1; attempt <= maxRetries; attempt++) { - try { - return await this.runInTransaction(operation); - } catch (error) { - lastError = error as Error; - - // Check if error is retryable (deadlock, serialization failure, etc.) - if (this.isRetryableError(error) && attempt < maxRetries) { - this.logger.warn( - `Transaction failed (attempt ${attempt}/${maxRetries}), retrying in ${retryDelay}ms...`, - ); - await this.delay(retryDelay); - retryDelay *= 2; // Exponential backoff - } else { - throw error; + /** + * Execute operations with retry logic + */ + async runWithRetry(operation: (manager: EntityManager) => Promise, maxRetries: number = 3, retryDelay: number = 1000): Promise { + let lastError: Error | undefined; + for (let attempt = 1; attempt <= maxRetries; attempt++) { + try { + return await this.runInTransaction(operation); + } + catch (error) { + lastError = error as Error; + // Check if error is retryable (deadlock, serialization failure, etc.) + if (this.isRetryableError(error) && attempt < maxRetries) { + this.logger.warn(`Transaction failed (attempt ${attempt}/${maxRetries}), retrying in ${retryDelay}ms...`); + await this.delay(retryDelay); + retryDelay *= 2; // Exponential backoff + } + else { + throw error; + } + } } - } + throw lastError ?? new Error('Transaction failed'); } - - throw lastError ?? new Error('Transaction failed'); - } - - /** - * Execute multiple operations in parallel within a transaction - */ - async runParallelInTransaction( - operations: Array<(manager: EntityManager) => Promise>, - ): Promise { - return this.runInTransaction(async (manager) => { - return Promise.all(operations.map((op) => op(manager))); - }); - } - - /** - * Execute operations with savepoint support - */ - async runWithSavepoint( - savepointName: string, - operation: (manager: EntityManager) => Promise, - parentManager?: EntityManager, - ): Promise { - if (parentManager) { - // Use existing transaction with savepoint - await parentManager.query(`SAVEPOINT ${savepointName}`); - - try { - const result = await operation(parentManager); - await parentManager.query(`RELEASE SAVEPOINT ${savepointName}`); - return result; - } catch (error) { - await parentManager.query(`ROLLBACK TO SAVEPOINT ${savepointName}`); - throw error; - } - } else { - // Create new transaction - return this.runInTransaction(operation); + /** + * Execute multiple operations in parallel within a transaction + */ + async runParallelInTransaction(operations: Array<(manager: EntityManager) => Promise>): Promise { + return this.runInTransaction(async (manager) => { + return Promise.all(operations.map((op) => op(manager))); + }); + } + /** + * Execute operations with savepoint support + */ + async runWithSavepoint(savepointName: string, operation: (manager: EntityManager) => Promise, parentManager?: EntityManager): Promise { + if (parentManager) { + // Use existing transaction with savepoint + await parentManager.query(`SAVEPOINT ${savepointName}`); + try { + const result = await operation(parentManager); + await parentManager.query(`RELEASE SAVEPOINT ${savepointName}`); + return result; + } + catch (error) { + await parentManager.query(`ROLLBACK TO SAVEPOINT ${savepointName}`); + throw error; + } + } + else { + // Create new transaction + return this.runInTransaction(operation); + } + } + /** + * Check if error is retryable + */ + private isRetryableError(error: unknown): boolean { + const retryableErrors = [ + 'deadlock', + 'serialization failure', + 'could not serialize', + 'lock timeout', + 'connection', + ]; + const errorMessage = (error instanceof Error ? error.message : String(error)).toLowerCase(); + return retryableErrors.some((msg) => errorMessage.includes(msg)); + } + /** + * Delay helper for retry logic + */ + private delay(ms: number): Promise { + return new Promise((resolve) => setTimeout(resolve, ms)); + } + /** + * Get current transaction manager if in transaction + */ + getCurrentManager(): EntityManager { + return this.dataSource.manager; + } + /** + * Generate unique transaction ID + */ + private generateTransactionId(): string { + return `tx_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`; + } + /** + * Get transaction metrics + */ + getTransactionMetrics(): TransactionMetrics[] { + return Array.from(this.activeTransactions.values()); + } + /** + * Get active transactions count + */ + getActiveTransactionCount(): number { + return this.activeTransactions.size; + } + /** + * Clear completed transactions + */ + clearCompletedTransactions(): void { + const now = new Date(); + for (const [id, metrics] of this.activeTransactions.entries()) { + if (metrics.endTime && now.getTime() - metrics.endTime.getTime() > 300000) { + // 5 minutes + this.activeTransactions.delete(id); + } + } } - } - - /** - * Check if error is retryable - */ - private isRetryableError(error: unknown): boolean { - const retryableErrors = [ - 'deadlock', - 'serialization failure', - 'could not serialize', - 'lock timeout', - 'connection', - ]; - - const errorMessage = (error instanceof Error ? error.message : String(error)).toLowerCase(); - return retryableErrors.some((msg) => errorMessage.includes(msg)); - } - - /** - * Delay helper for retry logic - */ - private delay(ms: number): Promise { - return new Promise((resolve) => setTimeout(resolve, ms)); - } - - /** - * Get current transaction manager if in transaction - */ - getCurrentManager(): EntityManager { - return this.dataSource.manager; - } - - /** - * Generate unique transaction ID - */ - private generateTransactionId(): string { - return `tx_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`; - } - - /** - * Get transaction metrics - */ - getTransactionMetrics(): TransactionMetrics[] { - return Array.from(this.activeTransactions.values()); - } - - /** - * Get active transactions count - */ - getActiveTransactionCount(): number { - return this.activeTransactions.size; - } - - /** - * Clear completed transactions - */ - clearCompletedTransactions(): void { - const now = new Date(); - for (const [id, metrics] of this.activeTransactions.entries()) { - if (metrics.endTime && now.getTime() - metrics.endTime.getTime() > 300000) { - // 5 minutes - this.activeTransactions.delete(id); - } + /** + * Check if currently in a transaction + */ + isInTransaction(manager: EntityManager): boolean { + return manager.queryRunner?.isTransactionActive || false; } - } - - /** - * Check if currently in a transaction - */ - isInTransaction(manager: EntityManager): boolean { - return manager.queryRunner?.isTransactionActive || false; - } } diff --git a/src/common/database/transactional.decorator.ts b/src/common/database/transactional.decorator.ts index 0f956bfd..26338068 100644 --- a/src/common/database/transactional.decorator.ts +++ b/src/common/database/transactional.decorator.ts @@ -1,55 +1,40 @@ import { SetMetadata } from '@nestjs/common'; import { TransactionService } from './transaction.service'; - export const TRANSACTIONAL_KEY = 'transactional'; - export interface TransactionalOptions { - isolationLevel?: 'READ UNCOMMITTED' | 'READ COMMITTED' | 'REPEATABLE READ' | 'SERIALIZABLE'; - retry?: boolean; - maxRetries?: number; - retryDelay?: number; - timeout?: number; + isolationLevel?: 'READ UNCOMMITTED' | 'READ COMMITTED' | 'REPEATABLE READ' | 'SERIALIZABLE'; + retry?: boolean; + maxRetries?: number; + retryDelay?: number; + timeout?: number; } - /** * Transactional decorator * Wraps methods in database transactions with retry logic and error handling */ -export const Transactional = (options: TransactionalOptions = {}) => - function (target: any, propertyKey: string, descriptor: PropertyDescriptor): PropertyDescriptor { +export const Transactional = (options: TransactionalOptions = {}) => function (target: unknown, propertyKey: string, descriptor: PropertyDescriptor): PropertyDescriptor { const originalMethod = descriptor.value; - - descriptor.value = async function (...args: any[]) { - const transactionService: TransactionService = this.transactionService; - - if (!transactionService) { - throw new Error( - `TransactionService not injected in ${target.constructor.name}. ` + - 'Please inject it via constructor.', - ); - } - - const operationName = `${target.constructor.name}.${propertyKey}`; - - try { - return await transactionService.runWithRetry( - async (_manager) => { - return await originalMethod.apply(this, args); - }, - options.maxRetries ?? 3, - options.retryDelay ?? 1000, - ); - } catch (error) { - console.error(`Transaction failed in ${operationName}:`, error); - throw error; - } + descriptor.value = async function (...args: unknown[]) { + const transactionService: TransactionService = this.transactionService; + if (!transactionService) { + throw new Error(`TransactionService not injected in ${target.constructor.name}. ` + + 'Please inject it via constructor.'); + } + const operationName = `${target.constructor.name}.${propertyKey}`; + try { + return await transactionService.runWithRetry(async (_manager) => { + return await originalMethod.apply(this, args); + }, options.maxRetries ?? 3, options.retryDelay ?? 1000); + } + catch (error) { + console.error(`Transaction failed in ${operationName}:`, error); + throw error; + } }; - // Attach metadata (useful for interceptors or future enhancements) SetMetadata(TRANSACTIONAL_KEY, { - method: propertyKey, - options, + method: propertyKey, + options, }); - return descriptor; - }; +}; diff --git a/src/common/database/transactional.interceptor.ts b/src/common/database/transactional.interceptor.ts index deff4a01..e7324798 100644 --- a/src/common/database/transactional.interceptor.ts +++ b/src/common/database/transactional.interceptor.ts @@ -3,67 +3,48 @@ import { Reflector } from '@nestjs/core'; import { Observable } from 'rxjs'; import { TransactionService } from './transaction.service'; import { TRANSACTIONAL_KEY, TransactionalOptions } from './transactional.decorator'; - /** * Interceptor to automatically wrap methods in transactions */ @Injectable() export class TransactionalInterceptor implements NestInterceptor { - private readonly logger = new Logger(TransactionalInterceptor.name); - - constructor( - private readonly reflector: Reflector, - private readonly transactionService: TransactionService, - ) {} - - async intercept(context: ExecutionContext, next: CallHandler): Promise> { - const options = this.reflector.get( - TRANSACTIONAL_KEY, - context.getHandler(), - ); - - if (!options) { - return next.handle(); + private readonly logger = new Logger(TransactionalInterceptor.name); + constructor(private readonly reflector: Reflector, private readonly transactionService: TransactionService) { } + async intercept(context: ExecutionContext, next: CallHandler): Promise> { + const options = this.reflector.get(TRANSACTIONAL_KEY, context.getHandler()); + if (!options) { + return next.handle(); + } + const handler = context.getHandler(); + const className = context.getClass().name; + const methodName = handler.name; + this.logger.debug(`Executing transactional method: ${className}.${methodName}`); + try { + let result; + if (options.retry) { + result = await this.transactionService.runWithRetry(async (_manager) => { + // Execute the original method + return await next.handle().toPromise(); + }, options.maxRetries || 3, options.retryDelay || 1000); + } + else if (options.isolationLevel) { + result = await this.transactionService.runWithIsolationLevel(options.isolationLevel, async (_manager) => { + return await next.handle().toPromise(); + }); + } + else { + result = await this.transactionService.runInTransaction(async (_manager) => { + return await next.handle().toPromise(); + }); + } + return new Observable((subscriber) => { + subscriber.next(result); + subscriber.complete(); + }); + } + catch (error) { + this.logger.error(`Transaction failed in ${className}.${methodName}:`, error); + throw error; + } } - - const handler = context.getHandler(); - const className = context.getClass().name; - const methodName = handler.name; - - this.logger.debug(`Executing transactional method: ${className}.${methodName}`); - - try { - let result; - - if (options.retry) { - result = await this.transactionService.runWithRetry( - async (_manager) => { - // Execute the original method - return await next.handle().toPromise(); - }, - options.maxRetries || 3, - options.retryDelay || 1000, - ); - } else if (options.isolationLevel) { - result = await this.transactionService.runWithIsolationLevel( - options.isolationLevel, - async (_manager) => { - return await next.handle().toPromise(); - }, - ); - } else { - result = await this.transactionService.runInTransaction(async (_manager) => { - return await next.handle().toPromise(); - }); - } - - return new Observable((subscriber) => { - subscriber.next(result); - subscriber.complete(); - }); - } catch (error) { - this.logger.error(`Transaction failed in ${className}.${methodName}:`, error); - throw error; - } - } } diff --git a/src/common/decorators/roles.decorator.ts b/src/common/decorators/roles.decorator.ts index 53e4ae46..44abae91 100644 --- a/src/common/decorators/roles.decorator.ts +++ b/src/common/decorators/roles.decorator.ts @@ -1,18 +1,15 @@ import { SetMetadata } from '@nestjs/common'; - /** * #158 – System roles * * Extend this enum as the platform grows (e.g. MODERATOR, DAO_MEMBER, etc.). */ export enum Role { - USER = 'user', - MODERATOR = 'moderator', - ADMIN = 'admin', + USER = 'user', + MODERATOR = 'moderator', + ADMIN = 'admin' } - export const ROLES_KEY = 'roles'; - /** * Attach required roles to a route or controller. * @@ -25,4 +22,4 @@ export const ROLES_KEY = 'roles'; * \@Delete('posts/:id') * deletePost() {} */ -export const Roles = (...roles: string[]): any => SetMetadata(ROLES_KEY, roles); +export const Roles = (...roles: string[]): unknown => SetMetadata(ROLES_KEY, roles); diff --git a/src/common/decorators/translate.decorator.ts b/src/common/decorators/translate.decorator.ts index 7a727a6e..7317b743 100644 --- a/src/common/decorators/translate.decorator.ts +++ b/src/common/decorators/translate.decorator.ts @@ -1,14 +1,11 @@ import { createParamDecorator, ExecutionContext } from '@nestjs/common'; import { RequestWithLocale } from '../types/request-with-locale'; - /** * Resolved request locale when LanguageMiddleware runs (Accept-Language / ?lang=). * Falls back to I18N_DEFAULT_LOCALE or the decorator argument when unset. */ -export const CurrentLocale = createParamDecorator( - (fallbackLocale: string | undefined, ctx: ExecutionContext): string => { +export const CurrentLocale = createParamDecorator((fallbackLocale: string | undefined, ctx: ExecutionContext): string => { const req = ctx.switchToHttp().getRequest(); const envDefault = (process.env.I18N_DEFAULT_LOCALE || 'en').trim().toLowerCase(); return req.resolvedLocale ?? fallbackLocale ?? envDefault; - }, -); +}); diff --git a/src/common/dto/pagination.dto.spec.ts b/src/common/dto/pagination.dto.spec.ts index 88715195..2a3df5e0 100644 --- a/src/common/dto/pagination.dto.spec.ts +++ b/src/common/dto/pagination.dto.spec.ts @@ -2,41 +2,32 @@ import 'reflect-metadata'; import { validateSync } from 'class-validator'; import { PaginationQueryDto, CursorPaginationQueryDto } from './pagination.dto'; import { APP_CONSTANTS } from '../constants/app.constants'; - describe('Pagination DTO validation', () => { - const { DEFAULT_PAGE_SIZE, MAX_PAGE_SIZE } = APP_CONSTANTS; - - it('uses the default page size when no limit is provided', () => { - const dto = new PaginationQueryDto(); - const errors = validateSync(dto); - - expect(errors).toHaveLength(0); - expect(dto.limit).toBe(DEFAULT_PAGE_SIZE); - }); - - it('accepts a limit equal to the maximum page size', () => { - const dto = new PaginationQueryDto(); - dto.limit = MAX_PAGE_SIZE; - const errors = validateSync(dto); - - expect(errors).toHaveLength(0); - }); - - it('rejects a limit greater than the maximum page size', () => { - const dto = new PaginationQueryDto(); - dto.limit = MAX_PAGE_SIZE + 1; - const errors = validateSync(dto); - - expect(errors).toHaveLength(1); - expect(errors[0].constraints).toHaveProperty('max'); - }); - - it('validates cursor pagination limit against the same maximum', () => { - const dto = new CursorPaginationQueryDto(); - dto.limit = MAX_PAGE_SIZE + 1; - const errors = validateSync(dto); - - expect(errors).toHaveLength(1); - expect(errors[0].constraints).toHaveProperty('max'); - }); + const { DEFAULT_PAGE_SIZE, MAX_PAGE_SIZE } = APP_CONSTANTS; + it('uses the default page size when no limit is provided', () => { + const dto = new PaginationQueryDto(); + const errors = validateSync(dto); + expect(errors).toHaveLength(0); + expect(dto.limit).toBe(DEFAULT_PAGE_SIZE); + }); + it('accepts a limit equal to the maximum page size', () => { + const dto = new PaginationQueryDto(); + dto.limit = MAX_PAGE_SIZE; + const errors = validateSync(dto); + expect(errors).toHaveLength(0); + }); + it('rejects a limit greater than the maximum page size', () => { + const dto = new PaginationQueryDto(); + dto.limit = MAX_PAGE_SIZE + 1; + const errors = validateSync(dto); + expect(errors).toHaveLength(1); + expect(errors[0].constraints).toHaveProperty('max'); + }); + it('validates cursor pagination limit against the same maximum', () => { + const dto = new CursorPaginationQueryDto(); + dto.limit = MAX_PAGE_SIZE + 1; + const errors = validateSync(dto); + expect(errors).toHaveLength(1); + expect(errors[0].constraints).toHaveProperty('max'); + }); }); diff --git a/src/common/dto/pagination.dto.ts b/src/common/dto/pagination.dto.ts index dec54571..977eb10c 100644 --- a/src/common/dto/pagination.dto.ts +++ b/src/common/dto/pagination.dto.ts @@ -1,73 +1,59 @@ import { IsOptional, IsInt, Min, Max, IsString, IsIn, IsNumber } from 'class-validator'; import { Type } from 'class-transformer'; import { APP_CONSTANTS } from '../constants/app.constants'; - const { DEFAULT_PAGE_SIZE, MAX_PAGE_SIZE } = APP_CONSTANTS; - export enum SortOrder { - ASC = 'ASC', - DESC = 'DESC', + ASC = 'ASC', + DESC = 'DESC' } - export enum CursorDirection { - FORWARD = 'forward', - BACKWARD = 'backward', + FORWARD = 'forward', + BACKWARD = 'backward' } - export class PaginationQueryDto { - @IsOptional() - @Type(() => Number) - @IsInt() - @Min(1) - @IsNumber() - page?: number = 1; - - @IsOptional() - @Type(() => Number) - @IsInt() - @Min(1) - @Max(MAX_PAGE_SIZE, { message: `Page size cannot exceed ${MAX_PAGE_SIZE}` }) - @IsNumber() - limit?: number = DEFAULT_PAGE_SIZE; - - @IsOptional() - @IsString() - sortBy?: string = 'createdAt'; - - @IsOptional() - @IsIn([SortOrder.ASC, SortOrder.DESC]) - order?: SortOrder = SortOrder.DESC; - - @IsOptional() - @IsString() - search?: string; + @IsOptional() + @Type(() => Number) + @IsInt() + @Min(1) + @IsNumber() + page?: number = 1; + @IsOptional() + @Type(() => Number) + @IsInt() + @Min(1) + @Max(MAX_PAGE_SIZE, { message: `Page size cannot exceed ${MAX_PAGE_SIZE}` }) + @IsNumber() + limit?: number = DEFAULT_PAGE_SIZE; + @IsOptional() + @IsString() + sortBy?: string = 'createdAt'; + @IsOptional() + @IsIn([SortOrder.ASC, SortOrder.DESC]) + order?: SortOrder = SortOrder.DESC; + @IsOptional() + @IsString() + search?: string; } - export class CursorPaginationQueryDto { - @IsOptional() - @IsString() - cursor?: string; - - @IsOptional() - @Type(() => Number) - @IsInt() - @Min(1) - @Max(MAX_PAGE_SIZE, { message: `Page size cannot exceed ${MAX_PAGE_SIZE}` }) - limit?: number = DEFAULT_PAGE_SIZE; - - @IsOptional() - @IsString() - sortBy?: string = 'createdAt'; - - @IsOptional() - @IsIn([SortOrder.ASC, SortOrder.DESC]) - order?: SortOrder = SortOrder.DESC; - - @IsOptional() - @IsIn([CursorDirection.FORWARD, CursorDirection.BACKWARD]) - direction?: CursorDirection = CursorDirection.FORWARD; - - @IsOptional() - @IsString() - search?: string; + @IsOptional() + @IsString() + cursor?: string; + @IsOptional() + @Type(() => Number) + @IsInt() + @Min(1) + @Max(MAX_PAGE_SIZE, { message: `Page size cannot exceed ${MAX_PAGE_SIZE}` }) + limit?: number = DEFAULT_PAGE_SIZE; + @IsOptional() + @IsString() + sortBy?: string = 'createdAt'; + @IsOptional() + @IsIn([SortOrder.ASC, SortOrder.DESC]) + order?: SortOrder = SortOrder.DESC; + @IsOptional() + @IsIn([CursorDirection.FORWARD, CursorDirection.BACKWARD]) + direction?: CursorDirection = CursorDirection.FORWARD; + @IsOptional() + @IsString() + search?: string; } diff --git a/src/common/examples/timeout-example.controller.ts b/src/common/examples/timeout-example.controller.ts index 7e552b2e..55cfe653 100644 --- a/src/common/examples/timeout-example.controller.ts +++ b/src/common/examples/timeout-example.controller.ts @@ -1,43 +1,50 @@ import { Controller, Get, Post, Body } from '@nestjs/common'; import { ApiTags, ApiOperation } from '@nestjs/swagger'; import { Timeout } from '../interceptors/timeout.interceptor'; - @ApiTags('Timeout Examples') @Controller('examples') export class TimeoutExampleController { - @Get('quick') - @ApiOperation({ summary: 'Quick endpoint with custom timeout' }) - @Timeout(5000) // 5 seconds timeout - getQuickResponse(): { message: string } { - // This endpoint will timeout after 5 seconds - return { message: 'Quick response' }; - } - - @Get('slow') - @ApiOperation({ summary: 'Slow endpoint with longer timeout' }) - @Timeout(120000) // 2 minutes timeout - async getSlowResponse(): Promise<{ message: string }> { - // Simulate a slow operation - await new Promise((resolve) => setTimeout(resolve, 10000)); // 10 second delay - return { message: 'Slow response completed' }; - } - - @Post('process') - @ApiOperation({ summary: 'Processing endpoint with custom timeout' }) - @Timeout(60000) // 1 minute timeout - async processData(@Body() _data: any): Promise<{ result: string }> { - // Simulate data processing - await new Promise((resolve) => setTimeout(resolve, 30000)); // 30 second processing - return { result: 'Data processed successfully' }; - } - - @Get('default') - @ApiOperation({ summary: 'Endpoint using default timeout' }) - getDefaultTimeout(): { message: string; timeout: string } { - // This endpoint will use the default timeout from configuration - return { - message: 'Using default timeout', - timeout: 'Configured by REQUEST_TIMEOUT env var or default 30s', - }; - } + @Get('quick') + @ApiOperation({ summary: 'Quick endpoint with custom timeout' }) + @Timeout(5000) // 5 seconds timeout + getQuickResponse(): { + message: string; + } { + // This endpoint will timeout after 5 seconds + return { message: 'Quick response' }; + } + @Get('slow') + @ApiOperation({ summary: 'Slow endpoint with longer timeout' }) + @Timeout(120000) // 2 minutes timeout + async getSlowResponse(): Promise<{ + message: string; + }> { + // Simulate a slow operation + await new Promise((resolve) => setTimeout(resolve, 10000)); // 10 second delay + return { message: 'Slow response completed' }; + } + @Post('process') + @ApiOperation({ summary: 'Processing endpoint with custom timeout' }) + @Timeout(60000) // 1 minute timeout + async processData( + @Body() + _data: unknown): Promise<{ + result: string; + }> { + // Simulate data processing + await new Promise((resolve) => setTimeout(resolve, 30000)); // 30 second processing + return { result: 'Data processed successfully' }; + } + @Get('default') + @ApiOperation({ summary: 'Endpoint using default timeout' }) + getDefaultTimeout(): { + message: string; + timeout: string; + } { + // This endpoint will use the default timeout from configuration + return { + message: 'Using default timeout', + timeout: 'Configured by REQUEST_TIMEOUT env var or default 30s', + }; + } } diff --git a/src/common/examples/transaction-management.example.ts b/src/common/examples/transaction-management.example.ts index 44b26d1f..25cc3ac6 100644 --- a/src/common/examples/transaction-management.example.ts +++ b/src/common/examples/transaction-management.example.ts @@ -2,237 +2,211 @@ import { Injectable, Logger } from '@nestjs/common'; import { Repository } from 'typeorm'; import { TransactionService } from '../database/transaction.service'; import { TransactionHelperService } from '../database/transaction-helper.service'; - // Mock entities for example interface User { - id: string; - email: string; - password: string; - firstName: string; - lastName: string; - isEmailVerified: boolean; - emailVerificationToken?: string; - emailVerificationExpires?: Date; - lastLoginAt?: Date | null; - status?: string; - profileCompleted?: boolean; + id: string; + email: string; + password: string; + firstName: string; + lastName: string; + isEmailVerified: boolean; + emailVerificationToken?: string; + emailVerificationExpires?: Date; + lastLoginAt?: Date | null; + status?: string; + profileCompleted?: boolean; } - interface Payment { - id: string; - userId: string; - amount: number; - status: string; - description: string; - createdAt: Date; - completedAt?: Date; + id: string; + userId: string; + amount: number; + status: string; + description: string; + createdAt: Date; + completedAt?: Date; } - interface Invoice { - id: string; - userId: string; - paymentId: string; - amount: number; - status: string; - description: string; - createdAt: Date; - dueDate: Date; - paidAt?: Date; + id: string; + userId: string; + paymentId: string; + amount: number; + status: string; + description: string; + createdAt: Date; + dueDate: Date; + paidAt?: Date; } - /** * Example service demonstrating transaction management * Shows atomic operations for complex business logic */ @Injectable() export class TransactionExampleService { - private readonly logger = new Logger(TransactionExampleService.name); - - constructor( - private readonly userRepository: Repository, - private readonly transactionService: TransactionService, - private readonly transactionHelper: TransactionHelperService, - private readonly paymentRepository: Repository, - private readonly invoiceRepository: Repository, - ) {} - - /** - * Example: Atomic user registration with email verification - * All operations succeed or fail together - */ - async registerUserWithVerification(userData: { - email: string; - password: string; - firstName: string; - lastName: string; - }): Promise<{ userId: string; verificationToken: string }> { - return await this.transactionService.runInTransaction(async (_manager) => { - // Create user - const user = await this.userRepository.save({ - email: userData.email, - password: userData.password, - firstName: userData.firstName, - lastName: userData.lastName, - isEmailVerified: false, - lastLoginAt: null, - }); - - // Generate verification token - const verificationToken = this.generateVerificationToken(); - const expiresAt = new Date(Date.now() + 24 * 60 * 60 * 1000); - - // Save verification token - await this.userRepository.update(user.id, { - emailVerificationToken: verificationToken, - emailVerificationExpires: expiresAt, - } as any); - - this.logger.log( - `User registered with ID: ${user.id}, verification token: ${verificationToken}`, - ); - - return { - userId: user.id, - verificationToken, - }; - }); - } - - /** - * Example: Payment with invoice creation - * Demonstrates atomic payment processing - */ - async processPaymentWithInvoice(paymentData: { - userId: string; - amount: number; - description: string; - }): Promise<{ paymentId: string; invoiceId: string }> { - return await this.transactionService.runWithRetry(async (_manager) => { - // Create payment record - const payment = await this.paymentRepository.save({ - userId: paymentData.userId, - amount: paymentData.amount, - status: 'PENDING' as any, - description: paymentData.description, - createdAt: new Date(), - }); - - // Create invoice - const invoice = await this.invoiceRepository.save({ - userId: paymentData.userId, - paymentId: payment.id, - amount: paymentData.amount, - status: 'PENDING' as any, - description: paymentData.description, - createdAt: new Date(), - dueDate: new Date(Date.now() + 30 * 24 * 60 * 1000), // 30 days - }); - - // Simulate payment processing - await new Promise((resolve) => setTimeout(resolve, 1000)); - - // Update payment status - await this.paymentRepository.update(payment.id, { - status: 'COMPLETED' as any, - completedAt: new Date(), - } as any); - - // Update invoice status - await this.invoiceRepository.update(invoice.id, { - status: 'PAID' as any, - paidAt: new Date(), - } as any); - - this.logger.log(`Payment processed: ${payment.id}, Invoice: ${invoice.id}`); - - return { - paymentId: payment.id, - invoiceId: invoice.id, - }; - }); - } - - /** - * Example: Complex operation with conditional rollback - */ - async processWithConditionalRollback(userId: string, data: any): Promise { - return await this.transactionHelper.executeWithRollback([ - { - operation: async (_manager) => { - // Step 1: Update user - await this.userRepository.update(userId, { - lastLoginAt: new Date(), - status: 'ACTIVE', - } as any); - return { step: 'user_updated', userId }; - }, - rollback: async (_manager) => { - // Rollback user status - await this.userRepository.update(userId, { - lastLoginAt: null, - status: 'INACTIVE', - } as any); - this.logger.warn(`Rolled back user status for ${userId}`); - }, - }, - { - operation: async (_manager) => { - // Step 2: Create related record - const record = await this.someRecordRepository.save({ - userId, - data, - createdAt: new Date(), - }); - return { step: 'record_created', userId: record.id }; - }, - rollback: async (_manager) => { - // Rollback record creation - this.logger.warn('Rolled back record creation'); - }, - condition: () => Math.random() > 0.5, // 50% chance of success - }, - ]); - } - - /** - * Example: Savepoint usage for nested operations - */ - async complexNestedOperation(userId: string): Promise { - return await this.transactionService.runInTransaction(async (manager) => { - // Create savepoint for first operation - await this.transactionHelper.createSavepoint(manager, 'user_update'); - - try { - // Update user - await this.userRepository.update(userId, { - lastLoginAt: new Date(), - profileCompleted: true, - } as any); - - // Create savepoint for second operation - await this.transactionHelper.createSavepoint(manager, 'profile_setup'); - - // Second operation that might fail - if (Math.random() > 0.3) { - throw new Error('Profile setup failed'); - } - - this.logger.log('Profile setup completed successfully'); - } catch (error) { - // Rollback to first savepoint - await this.transactionHelper.rollbackToSavepoint(manager, 'user_update'); - this.logger.error('Profile setup failed, rolled back to user_update'); - throw error; - } - }); - } - - private generateVerificationToken(): string { - return Math.random().toString(36).substring(2, 15); - } - - // Mock repository for example - private get someRecordRepository(): Repository { - return this.userRepository as any; - } + private readonly logger = new Logger(TransactionExampleService.name); + constructor(private readonly userRepository: Repository, private readonly transactionService: TransactionService, private readonly transactionHelper: TransactionHelperService, private readonly paymentRepository: Repository, private readonly invoiceRepository: Repository) { } + /** + * Example: Atomic user registration with email verification + * All operations succeed or fail together + */ + async registerUserWithVerification(userData: { + email: string; + password: string; + firstName: string; + lastName: string; + }): Promise<{ + userId: string; + verificationToken: string; + }> { + return await this.transactionService.runInTransaction(async (_manager) => { + // Create user + const user = await this.userRepository.save({ + email: userData.email, + password: userData.password, + firstName: userData.firstName, + lastName: userData.lastName, + isEmailVerified: false, + lastLoginAt: null, + }); + // Generate verification token + const verificationToken = this.generateVerificationToken(); + const expiresAt = new Date(Date.now() + 24 * 60 * 60 * 1000); + // Save verification token + await this.userRepository.update(user.id, { + emailVerificationToken: verificationToken, + emailVerificationExpires: expiresAt, + } as unknown); + this.logger.log(`User registered with ID: ${user.id}, verification token: ${verificationToken}`); + return { + userId: user.id, + verificationToken, + }; + }); + } + /** + * Example: Payment with invoice creation + * Demonstrates atomic payment processing + */ + async processPaymentWithInvoice(paymentData: { + userId: string; + amount: number; + description: string; + }): Promise<{ + paymentId: string; + invoiceId: string; + }> { + return await this.transactionService.runWithRetry(async (_manager) => { + // Create payment record + const payment = await this.paymentRepository.save({ + userId: paymentData.userId, + amount: paymentData.amount, + status: 'PENDING' as unknown, + description: paymentData.description, + createdAt: new Date(), + }); + // Create invoice + const invoice = await this.invoiceRepository.save({ + userId: paymentData.userId, + paymentId: payment.id, + amount: paymentData.amount, + status: 'PENDING' as unknown, + description: paymentData.description, + createdAt: new Date(), + dueDate: new Date(Date.now() + 30 * 24 * 60 * 1000), // 30 days + }); + // Simulate payment processing + await new Promise((resolve) => setTimeout(resolve, 1000)); + // Update payment status + await this.paymentRepository.update(payment.id, { + status: 'COMPLETED' as unknown, + completedAt: new Date(), + } as unknown); + // Update invoice status + await this.invoiceRepository.update(invoice.id, { + status: 'PAID' as unknown, + paidAt: new Date(), + } as unknown); + this.logger.log(`Payment processed: ${payment.id}, Invoice: ${invoice.id}`); + return { + paymentId: payment.id, + invoiceId: invoice.id, + }; + }); + } + /** + * Example: Complex operation with conditional rollback + */ + async processWithConditionalRollback(userId: string, data: unknown): Promise { + return await this.transactionHelper.executeWithRollback([ + { + operation: async (_manager) => { + // Step 1: Update user + await this.userRepository.update(userId, { + lastLoginAt: new Date(), + status: 'ACTIVE', + } as unknown); + return { step: 'user_updated', userId }; + }, + rollback: async (_manager) => { + // Rollback user status + await this.userRepository.update(userId, { + lastLoginAt: null, + status: 'INACTIVE', + } as unknown); + this.logger.warn(`Rolled back user status for ${userId}`); + }, + }, + { + operation: async (_manager) => { + // Step 2: Create related record + const record = await this.someRecordRepository.save({ + userId, + data, + createdAt: new Date(), + }); + return { step: 'record_created', userId: record.id }; + }, + rollback: async (_manager) => { + // Rollback record creation + this.logger.warn('Rolled back record creation'); + }, + condition: () => Math.random() > 0.5, // 50% chance of success + }, + ]); + } + /** + * Example: Savepoint usage for nested operations + */ + async complexNestedOperation(userId: string): Promise { + return await this.transactionService.runInTransaction(async (manager) => { + // Create savepoint for first operation + await this.transactionHelper.createSavepoint(manager, 'user_update'); + try { + // Update user + await this.userRepository.update(userId, { + lastLoginAt: new Date(), + profileCompleted: true, + } as unknown); + // Create savepoint for second operation + await this.transactionHelper.createSavepoint(manager, 'profile_setup'); + // Second operation that might fail + if (Math.random() > 0.3) { + throw new Error('Profile setup failed'); + } + this.logger.log('Profile setup completed successfully'); + } + catch (error) { + // Rollback to first savepoint + await this.transactionHelper.rollbackToSavepoint(manager, 'user_update'); + this.logger.error('Profile setup failed, rolled back to user_update'); + throw error; + } + }); + } + private generateVerificationToken(): string { + return Math.random().toString(36).substring(2, 15); + } + // Mock repository for example + private get someRecordRepository(): Repository { + return this.userRepository as unknown; + } } diff --git a/src/common/export/export.service.ts b/src/common/export/export.service.ts index 5250dccf..404e6900 100644 --- a/src/common/export/export.service.ts +++ b/src/common/export/export.service.ts @@ -1,436 +1,356 @@ import { BadRequestException, Injectable, Logger, NotFoundException } from '@nestjs/common'; import { InjectQueue, Process, Processor } from '@nestjs/bull'; import { InjectRepository } from '@nestjs/typeorm'; -import { - Column, - CreateDateColumn, - Entity, - Index, - PrimaryGeneratedColumn, - Repository, - UpdateDateColumn, -} from 'typeorm'; +import { Column, CreateDateColumn, Entity, Index, PrimaryGeneratedColumn, Repository, UpdateDateColumn, } from 'typeorm'; import { Job, Queue } from 'bull'; import { User } from '../../users/entities/user.entity'; import { Enrollment } from '../../courses/entities/enrollment.entity'; - export type ExportFormat = 'json' | 'pdf'; - export enum UserExportStatus { - PENDING = 'pending', - IN_PROGRESS = 'in_progress', - COMPLETED = 'completed', - FAILED = 'failed', + PENDING = 'pending', + IN_PROGRESS = 'in_progress', + COMPLETED = 'completed', + FAILED = 'failed' } - @Entity('user_export_history') @Index(['userId', 'createdAt']) export class UserExportHistory { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ name: 'user_id' }) - @Index() - userId: string; - - @Column({ type: 'varchar' }) - format: ExportFormat; - - @Column({ - type: 'enum', - enum: UserExportStatus, - default: UserExportStatus.PENDING, - }) - @Index() - status: UserExportStatus; - - @Column({ name: 'file_name', nullable: true }) - fileName?: string; - - @Column({ name: 'mime_type', nullable: true }) - mimeType?: string; - - @Column({ name: 'file_content', type: 'text', nullable: true }) - fileContent?: string; - - @Column({ name: 'error_message', type: 'text', nullable: true }) - errorMessage?: string; - - @Column({ type: 'jsonb', nullable: true }) - metadata?: Record; - - @Column({ name: 'completed_at', type: 'timestamp', nullable: true }) - completedAt?: Date; - - @CreateDateColumn({ name: 'created_at' }) - createdAt: Date; - - @UpdateDateColumn({ name: 'updated_at' }) - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ name: 'user_id' }) + @Index() + userId: string; + @Column({ type: 'varchar' }) + format: ExportFormat; + @Column({ + type: 'enum', + enum: UserExportStatus, + default: UserExportStatus.PENDING, + }) + @Index() + status: UserExportStatus; + @Column({ name: 'file_name', nullable: true }) + fileName?: string; + @Column({ name: 'mime_type', nullable: true }) + mimeType?: string; + @Column({ name: 'file_content', type: 'text', nullable: true }) + fileContent?: string; + @Column({ name: 'error_message', type: 'text', nullable: true }) + errorMessage?: string; + @Column({ type: 'jsonb', nullable: true }) + metadata?: Record; + @Column({ name: 'completed_at', type: 'timestamp', nullable: true }) + completedAt?: Date; + @CreateDateColumn({ name: 'created_at' }) + createdAt: Date; + @UpdateDateColumn({ name: 'updated_at' }) + updatedAt: Date; } - interface ExportJobData { - exportId: string; - userId: string; - format: ExportFormat; + exportId: string; + userId: string; + format: ExportFormat; } - interface PreparedExportData { - user: { - id: string; - email: string; - firstName: string; - lastName: string; - role: string; - status: string; - tenantId?: string; - isEmailVerified: boolean; - createdAt: Date; - updatedAt: Date; - lastLoginAt?: Date; - }; - courseProgress: Array<{ - enrollmentId: string; - courseId: string; - courseTitle: string; - progress: number; - status: string; - enrolledAt: Date; - lastAccessedAt: Date; - }>; - exportMeta: { - generatedAt: string; - totalEnrollments: number; - completedCourses: number; - averageProgress: number; - }; + user: { + id: string; + email: string; + firstName: string; + lastName: string; + role: string; + status: string; + tenantId?: string; + isEmailVerified: boolean; + createdAt: Date; + updatedAt: Date; + lastLoginAt?: Date; + }; + courseProgress: Array<{ + enrollmentId: string; + courseId: string; + courseTitle: string; + progress: number; + status: string; + enrolledAt: Date; + lastAccessedAt: Date; + }>; + exportMeta: { + generatedAt: string; + totalEnrollments: number; + completedCourses: number; + averageProgress: number; + }; } - @Injectable() export class ExportService { - private readonly logger = new Logger(ExportService.name); - private static readonly QUEUE_NAME = 'user-data-export'; - private static readonly JOB_NAME = 'generate-user-data-export'; - - constructor( + private readonly logger = new Logger(ExportService.name); + private static readonly QUEUE_NAME = 'user-data-export'; + private static readonly JOB_NAME = 'generate-user-data-export'; + constructor( @InjectRepository(User) - private readonly userRepository: Repository, + private readonly userRepository: Repository, @InjectRepository(Enrollment) - private readonly enrollmentRepository: Repository, + private readonly enrollmentRepository: Repository, @InjectRepository(UserExportHistory) - private readonly exportHistoryRepository: Repository, + private readonly exportHistoryRepository: Repository, @InjectQueue(ExportService.QUEUE_NAME) - private readonly exportQueue: Queue, - ) {} - - async requestUserDataExport(userId: string, format: ExportFormat = 'json') { - this.ensureValidFormat(format); - - const user = await this.userRepository.findOne({ where: { id: userId } }); - if (!user) { - throw new NotFoundException('User not found'); + private readonly exportQueue: Queue) { } + async requestUserDataExport(userId: string, format: ExportFormat = 'json') { + this.ensureValidFormat(format); + const user = await this.userRepository.findOne({ where: { id: userId } }); + if (!user) { + throw new NotFoundException('User not found'); + } + const exportRecord = this.exportHistoryRepository.create({ + userId, + format, + status: UserExportStatus.PENDING, + metadata: { + requestedAt: new Date().toISOString(), + }, + }); + const savedRecord = await this.exportHistoryRepository.save(exportRecord); + await this.exportQueue.add(ExportService.JOB_NAME, { + exportId: savedRecord.id, + userId, + format, + } as ExportJobData, { + attempts: 3, + backoff: { + type: 'exponential', + delay: 2000, + }, + removeOnComplete: 50, + removeOnFail: 50, + }); + return { + exportId: savedRecord.id, + status: savedRecord.status, + message: 'Export request accepted and queued for background processing', + }; } - - const exportRecord = this.exportHistoryRepository.create({ - userId, - format, - status: UserExportStatus.PENDING, - metadata: { - requestedAt: new Date().toISOString(), - }, - }); - - const savedRecord = await this.exportHistoryRepository.save(exportRecord); - - await this.exportQueue.add( - ExportService.JOB_NAME, - { - exportId: savedRecord.id, - userId, - format, - } as ExportJobData, - { - attempts: 3, - backoff: { - type: 'exponential', - delay: 2000, - }, - removeOnComplete: 50, - removeOnFail: 50, - }, - ); - - return { - exportId: savedRecord.id, - status: savedRecord.status, - message: 'Export request accepted and queued for background processing', - }; - } - - async getUserExportHistory(userId: string) { - const history = await this.exportHistoryRepository.find({ - where: { userId }, - order: { createdAt: 'DESC' }, - }); - - return history.map((item) => ({ - id: item.id, - format: item.format, - status: item.status, - fileName: item.fileName, - mimeType: item.mimeType, - errorMessage: item.errorMessage, - metadata: item.metadata, - completedAt: item.completedAt, - createdAt: item.createdAt, - updatedAt: item.updatedAt, - })); - } - - async getCompletedExportFile( - userId: string, - exportId: string, - ): Promise<{ - fileName: string; - mimeType: string; - content: Buffer; - }> { - const record = await this.exportHistoryRepository.findOne({ - where: { id: exportId, userId }, - }); - - if (!record) { - throw new NotFoundException('Export record not found'); + async getUserExportHistory(userId: string) { + const history = await this.exportHistoryRepository.find({ + where: { userId }, + order: { createdAt: 'DESC' }, + }); + return history.map((item) => ({ + id: item.id, + format: item.format, + status: item.status, + fileName: item.fileName, + mimeType: item.mimeType, + errorMessage: item.errorMessage, + metadata: item.metadata, + completedAt: item.completedAt, + createdAt: item.createdAt, + updatedAt: item.updatedAt, + })); } - - if (record.status !== UserExportStatus.COMPLETED || !record.fileContent) { - throw new BadRequestException('Export is not ready yet'); + async getCompletedExportFile(userId: string, exportId: string): Promise<{ + fileName: string; + mimeType: string; + content: Buffer; + }> { + const record = await this.exportHistoryRepository.findOne({ + where: { id: exportId, userId }, + }); + if (!record) { + throw new NotFoundException('Export record not found'); + } + if (record.status !== UserExportStatus.COMPLETED || !record.fileContent) { + throw new BadRequestException('Export is not ready yet'); + } + return { + fileName: record.fileName || `user-export-${record.id}.${record.format}`, + mimeType: record.mimeType || 'application/octet-stream', + content: Buffer.from(record.fileContent, 'base64'), + }; } - - return { - fileName: record.fileName || `user-export-${record.id}.${record.format}`, - mimeType: record.mimeType || 'application/octet-stream', - content: Buffer.from(record.fileContent, 'base64'), - }; - } - - async processExportJob(jobData: ExportJobData): Promise { - const { exportId, userId, format } = jobData; - - const exportRecord = await this.exportHistoryRepository.findOne({ where: { id: exportId } }); - if (!exportRecord) { - this.logger.warn(`Export record not found: ${exportId}`); - return; + async processExportJob(jobData: ExportJobData): Promise { + const { exportId, userId, format } = jobData; + const exportRecord = await this.exportHistoryRepository.findOne({ where: { id: exportId } }); + if (!exportRecord) { + this.logger.warn(`Export record not found: ${exportId}`); + return; + } + try { + await this.exportHistoryRepository.update(exportId, { + status: UserExportStatus.IN_PROGRESS, + metadata: { + ...(exportRecord.metadata || {}), + startedAt: new Date().toISOString(), + }, + }); + const exportData = await this.prepareExportData(userId); + const preparedFile = format === 'pdf' ? this.generatePdfExport(exportData) : this.generateJsonExport(exportData); + await this.exportHistoryRepository.update(exportId, { + status: UserExportStatus.COMPLETED, + fileName: preparedFile.fileName, + mimeType: preparedFile.mimeType, + fileContent: preparedFile.content.toString('base64'), + completedAt: new Date(), + metadata: { + ...(exportRecord.metadata || {}), + completedAt: new Date().toISOString(), + payloadBytes: preparedFile.content.length, + }, + }); + } + catch (error) { + const message = error instanceof Error ? error.message : 'Unknown export error'; + this.logger.error(`Export processing failed for ${exportId}`, message); + await this.exportHistoryRepository.update(exportId, { + status: UserExportStatus.FAILED, + errorMessage: message, + metadata: { + ...(exportRecord.metadata || {}), + failedAt: new Date().toISOString(), + }, + }); + throw error; + } } - - try { - await this.exportHistoryRepository.update(exportId, { - status: UserExportStatus.IN_PROGRESS, - metadata: { - ...(exportRecord.metadata || {}), - startedAt: new Date().toISOString(), - }, - }); - - const exportData = await this.prepareExportData(userId); - const preparedFile = - format === 'pdf' ? this.generatePdfExport(exportData) : this.generateJsonExport(exportData); - - await this.exportHistoryRepository.update(exportId, { - status: UserExportStatus.COMPLETED, - fileName: preparedFile.fileName, - mimeType: preparedFile.mimeType, - fileContent: preparedFile.content.toString('base64'), - completedAt: new Date(), - metadata: { - ...(exportRecord.metadata || {}), - completedAt: new Date().toISOString(), - payloadBytes: preparedFile.content.length, - }, - }); - } catch (error) { - const message = error instanceof Error ? error.message : 'Unknown export error'; - this.logger.error(`Export processing failed for ${exportId}`, message); - - await this.exportHistoryRepository.update(exportId, { - status: UserExportStatus.FAILED, - errorMessage: message, - metadata: { - ...(exportRecord.metadata || {}), - failedAt: new Date().toISOString(), - }, - }); - - throw error; + private ensureValidFormat(format: string): asserts format is ExportFormat { + if (format !== 'json' && format !== 'pdf') { + throw new BadRequestException('Unsupported export format. Supported formats: json, pdf'); + } } - } - - private ensureValidFormat(format: string): asserts format is ExportFormat { - if (format !== 'json' && format !== 'pdf') { - throw new BadRequestException('Unsupported export format. Supported formats: json, pdf'); + private async prepareExportData(userId: string): Promise { + const user = await this.userRepository.findOne({ where: { id: userId } }); + if (!user) { + throw new NotFoundException('User not found'); + } + const enrollments = await this.enrollmentRepository.find({ + where: { userId }, + relations: ['course'], + order: { enrolledAt: 'DESC' }, + }); + const completedCourses = enrollments.filter((item) => item.progress >= 100).length; + const totalProgress = enrollments.reduce((sum, item) => sum + Number(item.progress || 0), 0); + const averageProgress = enrollments.length > 0 ? totalProgress / enrollments.length : 0; + return { + user: { + id: user.id, + email: user.email, + firstName: user.firstName, + lastName: user.lastName, + role: user.role, + status: user.status, + tenantId: user.tenantId, + isEmailVerified: user.isEmailVerified, + createdAt: user.createdAt, + updatedAt: user.updatedAt, + lastLoginAt: user.lastLoginAt, + }, + courseProgress: enrollments.map((enrollment) => ({ + enrollmentId: enrollment.id, + courseId: enrollment.courseId, + courseTitle: enrollment.course?.title || 'Unknown course', + progress: Number(enrollment.progress || 0), + status: enrollment.status, + enrolledAt: enrollment.enrolledAt, + lastAccessedAt: enrollment.lastAccessedAt, + })), + exportMeta: { + generatedAt: new Date().toISOString(), + totalEnrollments: enrollments.length, + completedCourses, + averageProgress: Number(averageProgress.toFixed(2)), + }, + }; } - } - - private async prepareExportData(userId: string): Promise { - const user = await this.userRepository.findOne({ where: { id: userId } }); - if (!user) { - throw new NotFoundException('User not found'); + private generateJsonExport(data: PreparedExportData): { + fileName: string; + mimeType: string; + content: Buffer; + } { + const timestamp = new Date().toISOString().replace(/[:.]/g, '-'); + return { + fileName: `user-data-export-${data.user.id}-${timestamp}.json`, + mimeType: 'application/json', + content: Buffer.from(JSON.stringify(data, null, 2), 'utf8'), + }; + } + private generatePdfExport(data: PreparedExportData): { + fileName: string; + mimeType: string; + content: Buffer; + } { + const timestamp = new Date().toISOString().replace(/[:.]/g, '-'); + const lines = [ + 'TeachLink User Data Export', + '', + `Generated: ${data.exportMeta.generatedAt}`, + '', + `User: ${data.user.firstName} ${data.user.lastName} (${data.user.email})`, + `Role: ${data.user.role}`, + `Status: ${data.user.status}`, + `Email Verified: ${data.user.isEmailVerified ? 'Yes' : 'No'}`, + '', + `Total Enrollments: ${data.exportMeta.totalEnrollments}`, + `Completed Courses: ${data.exportMeta.completedCourses}`, + `Average Progress: ${data.exportMeta.averageProgress}%`, + '', + 'Course Progress', + ...data.courseProgress.map((item, index) => `${index + 1}. ${item.courseTitle} | Progress: ${item.progress}% | Status: ${item.status}`), + ]; + return { + fileName: `user-data-export-${data.user.id}-${timestamp}.pdf`, + mimeType: 'application/pdf', + content: this.buildSimplePdf(lines), + }; + } + // Lightweight PDF writer to avoid additional dependencies for a simple report export. + private buildSimplePdf(lines: string[]): Buffer { + const escapePdfText = (value: string): string => value.replace(/\\/g, '\\\\').replace(/\(/g, '\\(').replace(/\)/g, '\\)'); + const bodyLines: string[] = ['BT', '/F1 11 Tf', '50 780 Td', '14 TL']; + lines.forEach((line, index) => { + const command = index === 0 ? 'Tj' : 'T*'; + bodyLines.push(`(${escapePdfText(line)}) ${command}`); + }); + bodyLines.push('ET'); + const stream = bodyLines.join('\n'); + const object1 = '1 0 obj\n<< /Type /Catalog /Pages 2 0 R >>\nendobj\n'; + const object2 = '2 0 obj\n<< /Type /Pages /Kids [3 0 R] /Count 1 >>\nendobj\n'; + const object3 = '3 0 obj\n<< /Type /Page /Parent 2 0 R /MediaBox [0 0 612 792] /Resources << /Font << /F1 4 0 R >> >> /Contents 5 0 R >>\nendobj\n'; + const object4 = '4 0 obj\n<< /Type /Font /Subtype /Type1 /BaseFont /Helvetica >>\nendobj\n'; + const object5 = `5 0 obj\n<< /Length ${Buffer.byteLength(stream, 'utf8')} >>\nstream\n${stream}\nendstream\nendobj\n`; + const header = '%PDF-1.4\n'; + const offset1 = header.length; + const offset2 = offset1 + object1.length; + const offset3 = offset2 + object2.length; + const offset4 = offset3 + object3.length; + const offset5 = offset4 + object4.length; + const body = object1 + object2 + object3 + object4 + object5; + const xrefOffset = header.length + body.length; + const xref = [ + 'xref', + '0 6', + '0000000000 65535 f ', + `${offset1.toString().padStart(10, '0')} 00000 n `, + `${offset2.toString().padStart(10, '0')} 00000 n `, + `${offset3.toString().padStart(10, '0')} 00000 n `, + `${offset4.toString().padStart(10, '0')} 00000 n `, + `${offset5.toString().padStart(10, '0')} 00000 n `, + 'trailer', + '<< /Size 6 /Root 1 0 R >>', + 'startxref', + `${xrefOffset}`, + '%%EOF', + ].join('\n'); + return Buffer.from(header + body + xref, 'utf8'); } - - const enrollments = await this.enrollmentRepository.find({ - where: { userId }, - relations: ['course'], - order: { enrolledAt: 'DESC' }, - }); - - const completedCourses = enrollments.filter((item) => item.progress >= 100).length; - const totalProgress = enrollments.reduce((sum, item) => sum + Number(item.progress || 0), 0); - const averageProgress = enrollments.length > 0 ? totalProgress / enrollments.length : 0; - - return { - user: { - id: user.id, - email: user.email, - firstName: user.firstName, - lastName: user.lastName, - role: user.role, - status: user.status, - tenantId: user.tenantId, - isEmailVerified: user.isEmailVerified, - createdAt: user.createdAt, - updatedAt: user.updatedAt, - lastLoginAt: user.lastLoginAt, - }, - courseProgress: enrollments.map((enrollment) => ({ - enrollmentId: enrollment.id, - courseId: enrollment.courseId, - courseTitle: enrollment.course?.title || 'Unknown course', - progress: Number(enrollment.progress || 0), - status: enrollment.status, - enrolledAt: enrollment.enrolledAt, - lastAccessedAt: enrollment.lastAccessedAt, - })), - exportMeta: { - generatedAt: new Date().toISOString(), - totalEnrollments: enrollments.length, - completedCourses, - averageProgress: Number(averageProgress.toFixed(2)), - }, - }; - } - - private generateJsonExport(data: PreparedExportData): { - fileName: string; - mimeType: string; - content: Buffer; - } { - const timestamp = new Date().toISOString().replace(/[:.]/g, '-'); - return { - fileName: `user-data-export-${data.user.id}-${timestamp}.json`, - mimeType: 'application/json', - content: Buffer.from(JSON.stringify(data, null, 2), 'utf8'), - }; - } - - private generatePdfExport(data: PreparedExportData): { - fileName: string; - mimeType: string; - content: Buffer; - } { - const timestamp = new Date().toISOString().replace(/[:.]/g, '-'); - const lines = [ - 'TeachLink User Data Export', - '', - `Generated: ${data.exportMeta.generatedAt}`, - '', - `User: ${data.user.firstName} ${data.user.lastName} (${data.user.email})`, - `Role: ${data.user.role}`, - `Status: ${data.user.status}`, - `Email Verified: ${data.user.isEmailVerified ? 'Yes' : 'No'}`, - '', - `Total Enrollments: ${data.exportMeta.totalEnrollments}`, - `Completed Courses: ${data.exportMeta.completedCourses}`, - `Average Progress: ${data.exportMeta.averageProgress}%`, - '', - 'Course Progress', - ...data.courseProgress.map( - (item, index) => - `${index + 1}. ${item.courseTitle} | Progress: ${item.progress}% | Status: ${item.status}`, - ), - ]; - - return { - fileName: `user-data-export-${data.user.id}-${timestamp}.pdf`, - mimeType: 'application/pdf', - content: this.buildSimplePdf(lines), - }; - } - - // Lightweight PDF writer to avoid additional dependencies for a simple report export. - private buildSimplePdf(lines: string[]): Buffer { - const escapePdfText = (value: string): string => - value.replace(/\\/g, '\\\\').replace(/\(/g, '\\(').replace(/\)/g, '\\)'); - - const bodyLines: string[] = ['BT', '/F1 11 Tf', '50 780 Td', '14 TL']; - lines.forEach((line, index) => { - const command = index === 0 ? 'Tj' : 'T*'; - bodyLines.push(`(${escapePdfText(line)}) ${command}`); - }); - bodyLines.push('ET'); - - const stream = bodyLines.join('\n'); - - const object1 = '1 0 obj\n<< /Type /Catalog /Pages 2 0 R >>\nendobj\n'; - const object2 = '2 0 obj\n<< /Type /Pages /Kids [3 0 R] /Count 1 >>\nendobj\n'; - const object3 = - '3 0 obj\n<< /Type /Page /Parent 2 0 R /MediaBox [0 0 612 792] /Resources << /Font << /F1 4 0 R >> >> /Contents 5 0 R >>\nendobj\n'; - const object4 = '4 0 obj\n<< /Type /Font /Subtype /Type1 /BaseFont /Helvetica >>\nendobj\n'; - const object5 = `5 0 obj\n<< /Length ${Buffer.byteLength(stream, 'utf8')} >>\nstream\n${stream}\nendstream\nendobj\n`; - - const header = '%PDF-1.4\n'; - - const offset1 = header.length; - const offset2 = offset1 + object1.length; - const offset3 = offset2 + object2.length; - const offset4 = offset3 + object3.length; - const offset5 = offset4 + object4.length; - - const body = object1 + object2 + object3 + object4 + object5; - const xrefOffset = header.length + body.length; - - const xref = [ - 'xref', - '0 6', - '0000000000 65535 f ', - `${offset1.toString().padStart(10, '0')} 00000 n `, - `${offset2.toString().padStart(10, '0')} 00000 n `, - `${offset3.toString().padStart(10, '0')} 00000 n `, - `${offset4.toString().padStart(10, '0')} 00000 n `, - `${offset5.toString().padStart(10, '0')} 00000 n `, - 'trailer', - '<< /Size 6 /Root 1 0 R >>', - 'startxref', - `${xrefOffset}`, - '%%EOF', - ].join('\n'); - - return Buffer.from(header + body + xref, 'utf8'); - } } - @Processor('user-data-export') export class UserDataExportProcessor { - private readonly logger = new Logger(UserDataExportProcessor.name); - - constructor(private readonly exportService: ExportService) {} - - @Process('generate-user-data-export') - async handleGenerateUserDataExport(job: Job): Promise { - this.logger.log(`Processing user export job: ${job.id}`); - await job.progress(20); - - await this.exportService.processExportJob(job.data); - - await job.progress(100); - } + private readonly logger = new Logger(UserDataExportProcessor.name); + constructor(private readonly exportService: ExportService) { } + @Process('generate-user-data-export') + async handleGenerateUserDataExport(job: Job): Promise { + this.logger.log(`Processing user export job: ${job.id}`); + await job.progress(20); + await this.exportService.processExportJob(job.data); + await job.progress(100); + } } diff --git a/src/common/guards/roles.guard.spec.ts b/src/common/guards/roles.guard.spec.ts index 5cd7852f..ea680589 100644 --- a/src/common/guards/roles.guard.spec.ts +++ b/src/common/guards/roles.guard.spec.ts @@ -1,64 +1,57 @@ import { RolesGuard } from '../guards/roles.guard'; import { Roles } from '../decorators/roles.decorator'; - describe('RolesGuard', () => { - let guard: RolesGuard; - let reflector: any; - - beforeEach(() => { - reflector = { - getAllAndOverride: jest.fn(), - }; - guard = new RolesGuard(reflector); - }); - - it('should allow access if no roles are required', () => { - reflector.getAllAndOverride.mockReturnValue(undefined); - const context = { - getHandler: () => {}, - getClass: () => {}, - switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['admin'] } }) }), - }; - expect(guard.canActivate(context as any)).toBe(true); - }); - - it('should deny access if user has no roles', () => { - reflector.getAllAndOverride.mockReturnValue(['admin']); - const context = { - getHandler: () => {}, - getClass: () => {}, - switchToHttp: () => ({ getRequest: () => ({ user: {} }) }), - }; - expect(guard.canActivate(context as any)).toBe(false); - }); - - it('should allow access if user has required role', () => { - reflector.getAllAndOverride.mockReturnValue(['admin']); - const context = { - getHandler: () => {}, - getClass: () => {}, - switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['admin', 'moderator'] } }) }), - }; - expect(guard.canActivate(context as any)).toBe(true); - }); - - it('should deny access if user does not have required role', () => { - reflector.getAllAndOverride.mockReturnValue(['admin']); - const context = { - getHandler: () => {}, - getClass: () => {}, - switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['user'] } }) }), - }; - expect(guard.canActivate(context as any)).toBe(false); - }); - - it('should support multiple roles per endpoint', () => { - reflector.getAllAndOverride.mockReturnValue(['admin', 'moderator']); - const context = { - getHandler: () => {}, - getClass: () => {}, - switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['moderator'] } }) }), - }; - expect(guard.canActivate(context as any)).toBe(true); - }); + let guard: RolesGuard; + let reflector: unknown; + beforeEach(() => { + reflector = { + getAllAndOverride: jest.fn(), + }; + guard = new RolesGuard(reflector); + }); + it('should allow access if no roles are required', () => { + reflector.getAllAndOverride.mockReturnValue(undefined); + const context = { + getHandler: () => { }, + getClass: () => { }, + switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['admin'] } }) }), + }; + expect(guard.canActivate(context as unknown)).toBe(true); + }); + it('should deny access if user has no roles', () => { + reflector.getAllAndOverride.mockReturnValue(['admin']); + const context = { + getHandler: () => { }, + getClass: () => { }, + switchToHttp: () => ({ getRequest: () => ({ user: {} }) }), + }; + expect(guard.canActivate(context as unknown)).toBe(false); + }); + it('should allow access if user has required role', () => { + reflector.getAllAndOverride.mockReturnValue(['admin']); + const context = { + getHandler: () => { }, + getClass: () => { }, + switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['admin', 'moderator'] } }) }), + }; + expect(guard.canActivate(context as unknown)).toBe(true); + }); + it('should deny access if user does not have required role', () => { + reflector.getAllAndOverride.mockReturnValue(['admin']); + const context = { + getHandler: () => { }, + getClass: () => { }, + switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['user'] } }) }), + }; + expect(guard.canActivate(context as unknown)).toBe(false); + }); + it('should support multiple roles per endpoint', () => { + reflector.getAllAndOverride.mockReturnValue(['admin', 'moderator']); + const context = { + getHandler: () => { }, + getClass: () => { }, + switchToHttp: () => ({ getRequest: () => ({ user: { roles: ['moderator'] } }) }), + }; + expect(guard.canActivate(context as unknown)).toBe(true); + }); }); diff --git a/src/common/guards/roles.guard.ts b/src/common/guards/roles.guard.ts index 89880b51..574c3854 100644 --- a/src/common/guards/roles.guard.ts +++ b/src/common/guards/roles.guard.ts @@ -1,25 +1,23 @@ import { Injectable, CanActivate, ExecutionContext } from '@nestjs/common'; import { Reflector } from '@nestjs/core'; import { ROLES_KEY } from '../decorators/roles.decorator'; - @Injectable() export class RolesGuard implements CanActivate { - constructor(private reflector: Reflector) {} - - canActivate(context: ExecutionContext): boolean { - const requiredRoles = this.reflector.getAllAndOverride(ROLES_KEY, [ - context.getHandler(), - context.getClass(), - ]); - if (!requiredRoles || requiredRoles.length === 0) { - return true; + constructor(private reflector: Reflector) { } + canActivate(context: ExecutionContext): boolean { + const requiredRoles = this.reflector.getAllAndOverride(ROLES_KEY, [ + context.getHandler(), + context.getClass(), + ]); + if (!requiredRoles || requiredRoles.length === 0) { + return true; + } + const request = context.switchToHttp().getRequest(); + const user = request.user; + if (!user || !user.roles) { + return false; + } + // Support multiple roles per endpoint + return requiredRoles.some((role) => user.roles.includes(role)); } - const request = context.switchToHttp().getRequest(); - const user = request.user; - if (!user || !user.roles) { - return false; - } - // Support multiple roles per endpoint - return requiredRoles.some((role) => user.roles.includes(role)); - } } diff --git a/src/common/guards/throttle.guard.ts b/src/common/guards/throttle.guard.ts index 4925dad8..6d477249 100644 --- a/src/common/guards/throttle.guard.ts +++ b/src/common/guards/throttle.guard.ts @@ -1,7 +1,6 @@ import { Injectable, ExecutionContext, Logger, HttpException, HttpStatus } from '@nestjs/common'; import { ThrottlerGuard, ThrottlerLimitDetail } from '@nestjs/throttler'; import { Request, Response } from 'express'; - /** * #155 – CustomThrottleGuard * @@ -12,44 +11,32 @@ import { Request, Response } from 'express'; */ @Injectable() export class CustomThrottleGuard extends ThrottlerGuard { - private readonly logger = new Logger(CustomThrottleGuard.name); - - /** Called by ThrottlerGuard when the limit is exceeded. */ - protected override async throwThrottlingException( - context: ExecutionContext, - throttlerLimitDetail: ThrottlerLimitDetail, - ): Promise { - const request = context.switchToHttp().getRequest(); - const response = context.switchToHttp().getResponse(); - - const ip = this.resolveClientIp(request); - const route = request.route?.path ?? request.url; - - this.logger.warn(`Rate limit exceeded: ip=${ip} method=${request.method} route=${route}`); - - // Inject standard rate-limit headers so clients can back off gracefully - // TTL in v6 is in seconds if defined that way in config, but throttlerLimitDetail.ttl is the value from config - const ttlSeconds = throttlerLimitDetail.ttl; - - response.setHeader('Retry-After', ttlSeconds); - response.setHeader('X-RateLimit-Limit', throttlerLimitDetail.limit); - response.setHeader('X-RateLimit-Remaining', 0); - response.setHeader('X-RateLimit-Reset', Math.floor(Date.now() / 1000) + ttlSeconds); - - throw new HttpException( - { - statusCode: HttpStatus.TOO_MANY_REQUESTS, - error: 'Too Many Requests', - message: 'You have exceeded the request rate limit. Please wait before retrying.', - retryAfterSeconds: ttlSeconds, - }, - HttpStatus.TOO_MANY_REQUESTS, - ); - } - - private resolveClientIp(request: Request): string { - const forwarded = request.headers['x-forwarded-for']; - if (typeof forwarded === 'string') return forwarded.split(',')[0].trim(); - return request.ip ?? request.socket?.remoteAddress ?? 'unknown'; - } + private readonly logger = new Logger(CustomThrottleGuard.name); + /** Called by ThrottlerGuard when the limit is exceeded. */ + protected override async throwThrottlingException(context: ExecutionContext, throttlerLimitDetail: ThrottlerLimitDetail): Promise { + const request = context.switchToHttp().getRequest(); + const response = context.switchToHttp().getResponse(); + const ip = this.resolveClientIp(request); + const route = request.route?.path ?? request.url; + this.logger.warn(`Rate limit exceeded: ip=${ip} method=${request.method} route=${route}`); + // Inject standard rate-limit headers so clients can back off gracefully + // TTL in v6 is in seconds if defined that way in config, but throttlerLimitDetail.ttl is the value from config + const ttlSeconds = throttlerLimitDetail.ttl; + response.setHeader('Retry-After', ttlSeconds); + response.setHeader('X-RateLimit-Limit', throttlerLimitDetail.limit); + response.setHeader('X-RateLimit-Remaining', 0); + response.setHeader('X-RateLimit-Reset', Math.floor(Date.now() / 1000) + ttlSeconds); + throw new HttpException({ + statusCode: HttpStatus.TOO_MANY_REQUESTS, + error: 'Too Many Requests', + message: 'You have exceeded the request rate limit. Please wait before retrying.', + retryAfterSeconds: ttlSeconds, + }, HttpStatus.TOO_MANY_REQUESTS); + } + private resolveClientIp(request: Request): string { + const forwarded = request.headers['x-forwarded-for']; + if (typeof forwarded === 'string') + return forwarded.split(',')[0].trim(); + return request.ip ?? request.socket?.remoteAddress ?? 'unknown'; + } } diff --git a/src/common/guards/ws-throttler.guard.ts b/src/common/guards/ws-throttler.guard.ts index b74d6c63..a28a51af 100644 --- a/src/common/guards/ws-throttler.guard.ts +++ b/src/common/guards/ws-throttler.guard.ts @@ -1,24 +1,18 @@ import { Injectable, ExecutionContext, Logger } from '@nestjs/common'; import { ThrottlerGuard, ThrottlerLimitDetail } from '@nestjs/throttler'; import { WsException } from '@nestjs/websockets'; - @Injectable() export class WsThrottlerGuard extends ThrottlerGuard { - private readonly wsLogger = new Logger(WsThrottlerGuard.name); - - protected async getTracker(req: Record): Promise { - const user = req.user; - const ip = req.conn?.remoteAddress || req.request?.connection?.remoteAddress || 'unknown'; - return user?.sub || user?.id || ip; - } - - protected async throwThrottlingException( - context: ExecutionContext, - _throttlerLimitDetail: ThrottlerLimitDetail, - ): Promise { - const client = context.switchToWs().getClient(); - const tracker = await this.getTracker(client); - this.wsLogger.warn(`WebSocket rate limit exceeded for ${tracker}`); - throw new WsException('Rate limit exceeded'); - } + private readonly wsLogger = new Logger(WsThrottlerGuard.name); + protected async getTracker(req: Record): Promise { + const user = req.user; + const ip = req.conn?.remoteAddress || req.request?.connection?.remoteAddress || 'unknown'; + return user?.sub || user?.id || ip; + } + protected async throwThrottlingException(context: ExecutionContext, _throttlerLimitDetail: ThrottlerLimitDetail): Promise { + const client = context.switchToWs().getClient(); + const tracker = await this.getTracker(client); + this.wsLogger.warn(`WebSocket rate limit exceeded for ${tracker}`); + throw new WsException('Rate limit exceeded'); + } } diff --git a/src/common/interceptors/api-error.interface.ts b/src/common/interceptors/api-error.interface.ts index ff1bbb5e..8ed250dc 100644 --- a/src/common/interceptors/api-error.interface.ts +++ b/src/common/interceptors/api-error.interface.ts @@ -1,16 +1,15 @@ export interface ApiError { - statusCode: number; - message: string | string[]; - error: string; - timestamp: string; - path: string; - /** Only populated in non-production environments */ - stack?: string; - /** Structured validation errors, if applicable */ - details?: ValidationErrorDetail[]; + statusCode: number; + message: string | string[]; + error: string; + timestamp: string; + path: string; + /** Only populated in non-production environments */ + stack?: string; + /** Structured validation errors, if applicable */ + details?: ValidationErrorDetail[]; } - export interface ValidationErrorDetail { - field: string; - constraints: Record; + field: string; + constraints: Record; } diff --git a/src/common/interceptors/api-version.interceptor.spec.ts b/src/common/interceptors/api-version.interceptor.spec.ts index 0bbc790b..79c7acf0 100644 --- a/src/common/interceptors/api-version.interceptor.spec.ts +++ b/src/common/interceptors/api-version.interceptor.spec.ts @@ -1,32 +1,24 @@ -import { - isVersionNeutralPath, - normalizeRequestedApiVersion, - parseSupportedApiVersions, -} from './api-version.interceptor'; - +import { isVersionNeutralPath, normalizeRequestedApiVersion, parseSupportedApiVersions, } from './api-version.interceptor'; describe('api version helpers', () => { - it('normalizes supported version formats', () => { - expect(normalizeRequestedApiVersion('1')).toBe('1'); - expect(normalizeRequestedApiVersion('v1')).toBe('1'); - expect(normalizeRequestedApiVersion('1.0')).toBe('1'); - expect(normalizeRequestedApiVersion('v1.0')).toBe('1'); - }); - - it('rejects invalid version formats', () => { - expect(normalizeRequestedApiVersion('latest')).toBeNull(); - expect(normalizeRequestedApiVersion('v1.2')).toBeNull(); - expect(normalizeRequestedApiVersion(undefined)).toBeNull(); - }); - - it('parses configured supported versions', () => { - expect(parseSupportedApiVersions('1, v1, 2')).toEqual(['1', '2']); - }); - - it('detects version-neutral routes', () => { - expect(isVersionNeutralPath('/')).toBe(true); - expect(isVersionNeutralPath('/health')).toBe(true); - expect(isVersionNeutralPath('/metrics/scheduled-tasks/dashboard')).toBe(true); - expect(isVersionNeutralPath('/webhooks/stripe')).toBe(true); - expect(isVersionNeutralPath('/users')).toBe(false); - }); + it('normalizes supported version formats', () => { + expect(normalizeRequestedApiVersion('1')).toBe('1'); + expect(normalizeRequestedApiVersion('v1')).toBe('1'); + expect(normalizeRequestedApiVersion('1.0')).toBe('1'); + expect(normalizeRequestedApiVersion('v1.0')).toBe('1'); + }); + it('rejects invalid version formats', () => { + expect(normalizeRequestedApiVersion('latest')).toBeNull(); + expect(normalizeRequestedApiVersion('v1.2')).toBeNull(); + expect(normalizeRequestedApiVersion(undefined)).toBeNull(); + }); + it('parses configured supported versions', () => { + expect(parseSupportedApiVersions('1, v1, 2')).toEqual(['1', '2']); + }); + it('detects version-neutral routes', () => { + expect(isVersionNeutralPath('/')).toBe(true); + expect(isVersionNeutralPath('/health')).toBe(true); + expect(isVersionNeutralPath('/metrics/scheduled-tasks/dashboard')).toBe(true); + expect(isVersionNeutralPath('/webhooks/stripe')).toBe(true); + expect(isVersionNeutralPath('/users')).toBe(false); + }); }); diff --git a/src/common/interceptors/api-version.interceptor.ts b/src/common/interceptors/api-version.interceptor.ts index 8acf2e70..cbc5d79f 100644 --- a/src/common/interceptors/api-version.interceptor.ts +++ b/src/common/interceptors/api-version.interceptor.ts @@ -1,98 +1,72 @@ import { CallHandler, ExecutionContext, Injectable, NestInterceptor } from '@nestjs/common'; import { Observable } from 'rxjs'; - export const API_VERSION_HEADER = process.env.API_VERSION_HEADER_NAME?.trim() || 'X-API-Version'; export const API_VERSION_HEADER_KEY = API_VERSION_HEADER.toLowerCase(); - const VERSION_NEUTRAL_PATH_PREFIXES = ['/api', '/health', '/metrics', '/webhooks']; const VERSION_NEUTRAL_EXACT_PATHS = ['/', '/api-json', '/favicon.ico']; - export interface VersionedRequest { - apiVersion?: string; - path?: string; - url?: string; - headers?: Record; + apiVersion?: string; + path?: string; + url?: string; + headers?: Record; } - export function normalizeRequestedApiVersion(version?: string | string[]): string | null { - if (!version) { - return null; - } - - const raw = Array.isArray(version) ? version[0] : version; - const trimmed = raw.trim(); - const match = trimmed.match(/^v?(\d+)(?:\.0+)?$/i); - - if (!match) { - return null; - } - - return match[1]; + if (!version) { + return null; + } + const raw = Array.isArray(version) ? version[0] : version; + const trimmed = raw.trim(); + const match = trimmed.match(/^v?(\d+)(?:\.0+)?$/i); + if (!match) { + return null; + } + return match[1]; } - export function normalizeConfiguredVersion(version: string): string { - const normalized = normalizeRequestedApiVersion(version); - return normalized || '1'; + const normalized = normalizeRequestedApiVersion(version); + return normalized || '1'; } - -export const DEFAULT_API_VERSION = normalizeConfiguredVersion( - process.env.API_DEFAULT_VERSION?.trim() || '1', -); - +export const DEFAULT_API_VERSION = normalizeConfiguredVersion(process.env.API_DEFAULT_VERSION?.trim() || '1'); export function parseSupportedApiVersions(raw = process.env.API_SUPPORTED_VERSIONS): string[] { - const configured = raw?.trim() ? raw : DEFAULT_API_VERSION; - const versions = configured - .split(',') - .map((version) => normalizeRequestedApiVersion(version)) - .filter((version): version is string => Boolean(version)); - - if (!versions.length) { - return [DEFAULT_API_VERSION]; - } - - return Array.from(new Set(versions)); + const configured = raw?.trim() ? raw : DEFAULT_API_VERSION; + const versions = configured + .split(',') + .map((version) => normalizeRequestedApiVersion(version)) + .filter((version): version is string => Boolean(version)); + if (!versions.length) { + return [DEFAULT_API_VERSION]; + } + return Array.from(new Set(versions)); } - export const SUPPORTED_API_VERSIONS = parseSupportedApiVersions(process.env.API_SUPPORTED_VERSIONS); - export function isVersionNeutralPath(pathOrUrl: string): boolean { - const path = (pathOrUrl || '/').split('?')[0]; - - if (VERSION_NEUTRAL_EXACT_PATHS.includes(path)) { - return true; - } - - return VERSION_NEUTRAL_PATH_PREFIXES.some( - (prefix) => path === prefix || path.startsWith(`${prefix}/`), - ); + const path = (pathOrUrl || '/').split('?')[0]; + if (VERSION_NEUTRAL_EXACT_PATHS.includes(path)) { + return true; + } + return VERSION_NEUTRAL_PATH_PREFIXES.some((prefix) => path === prefix || path.startsWith(`${prefix}/`)); } - @Injectable() export class ApiVersionInterceptor implements NestInterceptor { - intercept(context: ExecutionContext, next: CallHandler): Observable { - const http = context.switchToHttp(); - const request = http.getRequest(); - const response = http.getResponse<{ setHeader: (name: string, value: string) => void }>(); - - const path = request.path || request.url || '/'; - - const resolvedVersion = - request.apiVersion || - normalizeRequestedApiVersion(request.headers?.[API_VERSION_HEADER_KEY]) || - DEFAULT_API_VERSION; - - request.apiVersion = resolvedVersion; - - if (!isVersionNeutralPath(path)) { - response.setHeader(API_VERSION_HEADER, resolvedVersion); + intercept(context: ExecutionContext, next: CallHandler): Observable { + const http = context.switchToHttp(); + const request = http.getRequest(); + const response = http.getResponse<{ + setHeader: (name: string, value: string) => void; + }>(); + const path = request.path || request.url || '/'; + const resolvedVersion = request.apiVersion || + normalizeRequestedApiVersion(request.headers?.[API_VERSION_HEADER_KEY]) || + DEFAULT_API_VERSION; + request.apiVersion = resolvedVersion; + if (!isVersionNeutralPath(path)) { + response.setHeader(API_VERSION_HEADER, resolvedVersion); + } + return next.handle(); } - - return next.handle(); - } } - export function GetApiVersion(): ParameterDecorator { - return (_target: object, _propertyKey: string | symbol, _parameterIndex: number): void => { - // Intentionally a marker decorator for future injection. - }; + return (_target: object, _propertyKey: string | symbol, _parameterIndex: number): void => { + // Intentionally a marker decorator for future injection. + }; } diff --git a/src/common/interceptors/cache.interceptor.ts b/src/common/interceptors/cache.interceptor.ts index 00e483ee..797937bd 100644 --- a/src/common/interceptors/cache.interceptor.ts +++ b/src/common/interceptors/cache.interceptor.ts @@ -2,39 +2,32 @@ import { Injectable, NestInterceptor, ExecutionContext, CallHandler } from '@nes import { Cache } from 'cache-manager'; import { Observable } from 'rxjs'; import { tap } from 'rxjs/operators'; - @Injectable() export class CacheInterceptor implements NestInterceptor { - constructor(private cacheManager: Cache) {} - - async intercept(context: ExecutionContext, next: CallHandler): Promise> { - const req = context.switchToHttp().getRequest(); - const res = context.switchToHttp().getResponse(); - const key = req.originalUrl; - - // Only cache GET requests - if (req.method !== 'GET') { - // Invalidate cache for mutations - if (['POST', 'PUT', 'PATCH', 'DELETE'].includes(req.method)) { - await this.cacheManager.del(key); - } - return next.handle(); + constructor(private cacheManager: Cache) { } + async intercept(context: ExecutionContext, next: CallHandler): Promise> { + const req = context.switchToHttp().getRequest(); + const res = context.switchToHttp().getResponse(); + const key = req.originalUrl; + // Only cache GET requests + if (req.method !== 'GET') { + // Invalidate cache for mutations + if (['POST', 'PUT', 'PATCH', 'DELETE'].includes(req.method)) { + await this.cacheManager.del(key); + } + return next.handle(); + } + const cached = await this.cacheManager.get(key); + if (cached) { + res.setHeader('X-Cache', 'HIT'); + return new Observable((observer) => { + observer.next(cached); + observer.complete(); + }); + } + return next.handle().pipe(tap(async (data) => { + await this.cacheManager.set(key, data, parseInt(process.env.REDIS_TTL || '60', 10)); + res.setHeader('X-Cache', 'MISS'); + })); } - - const cached = await this.cacheManager.get(key); - if (cached) { - res.setHeader('X-Cache', 'HIT'); - return new Observable((observer) => { - observer.next(cached); - observer.complete(); - }); - } - - return next.handle().pipe( - tap(async (data) => { - await this.cacheManager.set(key, data, parseInt(process.env.REDIS_TTL || '60', 10)); - res.setHeader('X-Cache', 'MISS'); - }), - ); - } } diff --git a/src/common/interceptors/global-exception.filter.spec.ts b/src/common/interceptors/global-exception.filter.spec.ts index 40a68d7b..ba1dd5d7 100644 --- a/src/common/interceptors/global-exception.filter.spec.ts +++ b/src/common/interceptors/global-exception.filter.spec.ts @@ -2,62 +2,53 @@ import { GlobalExceptionFilter } from './global-exception.filter'; import { HttpStatus } from '@nestjs/common'; import { MulterError } from 'multer'; import { runWithCorrelationId } from '../utils/correlation.utils'; - describe('GlobalExceptionFilter', () => { - it('adds correlation ID to error response and header', () => { - const filter = new GlobalExceptionFilter(); - - const req: any = { method: 'GET', url: '/test' }; - const responseHeaders: Record = {}; - const res: any = { - status: (code: number) => { - res.statusCode = code; - return res; - }, - json: (body: any) => { - res.body = body; - return res; - }, - setHeader: (name: string, value: string) => { - responseHeaders[name.toLowerCase()] = value; - }, - getHeader: (name: string) => responseHeaders[name.toLowerCase()], - }; - - runWithCorrelationId(() => { - filter.catch(new Error('Test error'), { - switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), - } as any); - }, 'cid-123'); - - const body = res.body; - - expect(res.statusCode).toBe(HttpStatus.INTERNAL_SERVER_ERROR); - expect(body.correlationId).toBe('cid-123'); - expect(body.message).toBe('Test error'); - }); - - it('maps Multer file size errors to payload too large', () => { - const filter = new GlobalExceptionFilter(); - - const req: any = { method: 'POST', url: '/media/upload' }; - const res: any = { - status: (code: number) => { - res.statusCode = code; - return res; - }, - json: (body: any) => { - res.body = body; - return res; - }, - setHeader: jest.fn(), - }; - - filter.catch(new MulterError('LIMIT_FILE_SIZE', 'file'), { - switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), - } as any); - - expect(res.statusCode).toBe(HttpStatus.PAYLOAD_TOO_LARGE); - expect(res.body.message).toBe('Uploaded file exceeds the maximum allowed size.'); - }); + it('adds correlation ID to error response and header', () => { + const filter = new GlobalExceptionFilter(); + const req: unknown = { method: 'GET', url: '/test' }; + const responseHeaders: Record = {}; + const res: unknown = { + status: (code: number) => { + res.statusCode = code; + return res; + }, + json: (body: unknown) => { + res.body = body; + return res; + }, + setHeader: (name: string, value: string) => { + responseHeaders[name.toLowerCase()] = value; + }, + getHeader: (name: string) => responseHeaders[name.toLowerCase()], + }; + runWithCorrelationId(() => { + filter.catch(new Error('Test error'), { + switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), + } as unknown); + }, 'cid-123'); + const body = res.body; + expect(res.statusCode).toBe(HttpStatus.INTERNAL_SERVER_ERROR); + expect(body.correlationId).toBe('cid-123'); + expect(body.message).toBe('Test error'); + }); + it('maps Multer file size errors to payload too large', () => { + const filter = new GlobalExceptionFilter(); + const req: unknown = { method: 'POST', url: '/media/upload' }; + const res: unknown = { + status: (code: number) => { + res.statusCode = code; + return res; + }, + json: (body: unknown) => { + res.body = body; + return res; + }, + setHeader: jest.fn(), + }; + filter.catch(new MulterError('LIMIT_FILE_SIZE', 'file'), { + switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), + } as unknown); + expect(res.statusCode).toBe(HttpStatus.PAYLOAD_TOO_LARGE); + expect(res.body.message).toBe('Uploaded file exceeds the maximum allowed size.'); + }); }); diff --git a/src/common/interceptors/global-exception.filter.ts b/src/common/interceptors/global-exception.filter.ts index 66552fda..9b1aac74 100644 --- a/src/common/interceptors/global-exception.filter.ts +++ b/src/common/interceptors/global-exception.filter.ts @@ -1,246 +1,210 @@ -import { - ExceptionFilter, - Catch, - ArgumentsHost, - HttpException, - HttpStatus, - Logger, -} from '@nestjs/common'; +import { ExceptionFilter, Catch, ArgumentsHost, HttpException, HttpStatus, Logger, } from '@nestjs/common'; import { Request, Response } from 'express'; import { MulterError } from 'multer'; import { QueryFailedError, EntityNotFoundError } from 'typeorm'; import { ApiError, ValidationErrorDetail } from '../../interfaces/api-error.interface'; import { CORRELATION_ID_HEADER, getCorrelationId } from '../utils/correlation.utils'; - @Catch() export class GlobalExceptionFilter implements ExceptionFilter { - private readonly logger = new Logger(GlobalExceptionFilter.name); - private readonly isProduction = process.env.NODE_ENV === 'production'; - - catch(exception: unknown, host: ArgumentsHost): void { - const ctx = host.switchToHttp(); - const response = ctx.getResponse(); - const request = ctx.getRequest(); - - const { statusCode, message, error, details, stack } = this.resolveException(exception); - - const correlationId = getCorrelationId(); - - const errorResponse: ApiError = { - statusCode, - message, - error, - timestamp: new Date().toISOString(), - path: request.url, - correlationId, - ...(details?.length && { details }), - ...(!this.isProduction && stack && { stack }), - }; - - if (correlationId) { - response.setHeader(CORRELATION_ID_HEADER, correlationId); - } - - this.logger.error( - `[${request.method}] ${request.url} → ${statusCode} ${error}: ${ - Array.isArray(message) ? message.join(', ') : message - }`, - !this.isProduction ? stack : undefined, - GlobalExceptionFilter.name, - ); - - response.status(statusCode).json(errorResponse); - } - - // ─── Resolution helpers ──────────────────────────────────────────────────── - - private resolveException(exception: unknown): { - statusCode: number; - message: string | string[]; - error: string; - details?: ValidationErrorDetail[]; - stack?: string; - } { - // 1. NestJS HttpException (includes class-validator BadRequestException) - if (exception instanceof HttpException) { - return this.fromHttpException(exception); - } - - // 2. TypeORM – query/constraint failures - if (exception instanceof QueryFailedError) { - return this.fromQueryFailedError(exception); - } - - // 3. Multer upload failures - if (exception instanceof MulterError) { - return this.fromMulterError(exception); - } - - // 4. TypeORM – entity not found - if (exception instanceof EntityNotFoundError) { - return { - statusCode: HttpStatus.NOT_FOUND, - message: 'The requested resource was not found.', - error: 'Not Found', - stack: (exception as Error).stack, - }; - } - - // 5. Generic / unexpected Error - if (exception instanceof Error) { - return { - statusCode: HttpStatus.INTERNAL_SERVER_ERROR, - message: this.isProduction - ? 'An unexpected error occurred. Please try again later.' - : exception.message, - error: 'Internal Server Error', - stack: exception.stack, - }; - } - - // 6. Non-Error throw (strings, objects, etc.) - return { - statusCode: HttpStatus.INTERNAL_SERVER_ERROR, - message: 'An unexpected error occurred.', - error: 'Internal Server Error', - }; - } - - private fromHttpException(exception: HttpException): { - statusCode: number; - message: string | string[]; - error: string; - details?: ValidationErrorDetail[]; - stack?: string; - } { - const statusCode = exception.getStatus(); - const exceptionResponse = exception.getResponse(); - const stack = exception.stack; - - // class-validator wraps errors as { message: string[], error: string } - if (typeof exceptionResponse === 'object' && exceptionResponse !== null) { - const res = exceptionResponse as Record; - - const rawMessages = res['message']; - const messages: string[] = Array.isArray(rawMessages) - ? (rawMessages as string[]) - : typeof rawMessages === 'string' - ? [rawMessages] - : [exception.message]; - - // Parse class-validator constraint objects when present - const details = this.extractValidationDetails(rawMessages); - - return { - statusCode, - message: messages.length === 1 ? messages[0] : messages, - error: (res['error'] as string) ?? exception.message, - ...(details.length && { details }), - stack, - }; - } - - return { - statusCode, - message: typeof exceptionResponse === 'string' ? exceptionResponse : exception.message, - error: exception.message, - stack, - }; - } - - private fromQueryFailedError(exception: QueryFailedError): { - statusCode: number; - message: string; - error: string; - stack?: string; - } { - const driverError = (exception as QueryFailedError & { code?: string }).code; - - // PostgreSQL unique-violation - if (driverError === '23505') { - return { - statusCode: HttpStatus.CONFLICT, - message: 'A record with the provided value already exists.', - error: 'Conflict', - stack: exception.stack, - }; - } - - // PostgreSQL foreign-key violation - if (driverError === '23503') { - return { - statusCode: HttpStatus.UNPROCESSABLE_ENTITY, - message: 'Referenced resource does not exist.', - error: 'Unprocessable Entity', - stack: exception.stack, - }; + private readonly logger = new Logger(GlobalExceptionFilter.name); + private readonly isProduction = process.env.NODE_ENV === 'production'; + catch(exception: unknown, host: ArgumentsHost): void { + const ctx = host.switchToHttp(); + const response = ctx.getResponse(); + const request = ctx.getRequest(); + const { statusCode, message, error, details, stack } = this.resolveException(exception); + const correlationId = getCorrelationId(); + const errorResponse: ApiError = { + statusCode, + message, + error, + timestamp: new Date().toISOString(), + path: request.url, + correlationId, + ...(details?.length && { details }), + ...(!this.isProduction && stack && { stack }), + }; + if (correlationId) { + response.setHeader(CORRELATION_ID_HEADER, correlationId); + } + this.logger.error(`[${request.method}] ${request.url} → ${statusCode} ${error}: ${Array.isArray(message) ? message.join(', ') : message}`, !this.isProduction ? stack : undefined, GlobalExceptionFilter.name); + response.status(statusCode).json(errorResponse); } - - // Generic DB error – never expose query details in production - return { - statusCode: HttpStatus.INTERNAL_SERVER_ERROR, - message: this.isProduction ? 'A database error occurred.' : exception.message, - error: 'Database Error', - stack: exception.stack, - }; - } - - private fromMulterError(exception: MulterError): { - statusCode: number; - message: string; - error: string; - stack?: string; - } { - switch (exception.code) { - case 'LIMIT_FILE_SIZE': + // ─── Resolution helpers ──────────────────────────────────────────────────── + private resolveException(exception: unknown): { + statusCode: number; + message: string | string[]; + error: string; + details?: ValidationErrorDetail[]; + stack?: string; + } { + // 1. NestJS HttpException (includes class-validator BadRequestException) + if (exception instanceof HttpException) { + return this.fromHttpException(exception); + } + // 2. TypeORM – query/constraint failures + if (exception instanceof QueryFailedError) { + return this.fromQueryFailedError(exception); + } + // 3. Multer upload failures + if (exception instanceof MulterError) { + return this.fromMulterError(exception); + } + // 4. TypeORM – entity not found + if (exception instanceof EntityNotFoundError) { + return { + statusCode: HttpStatus.NOT_FOUND, + message: 'The requested resource was not found.', + error: 'Not Found', + stack: (exception as Error).stack, + }; + } + // 5. Generic / unexpected Error + if (exception instanceof Error) { + return { + statusCode: HttpStatus.INTERNAL_SERVER_ERROR, + message: this.isProduction + ? 'An unexpected error occurred. Please try again later.' + : exception.message, + error: 'Internal Server Error', + stack: exception.stack, + }; + } + // 6. Non-Error throw (strings, objects, etc.) return { - statusCode: HttpStatus.PAYLOAD_TOO_LARGE, - message: 'Uploaded file exceeds the maximum allowed size.', - error: 'Payload Too Large', - stack: exception.stack, + statusCode: HttpStatus.INTERNAL_SERVER_ERROR, + message: 'An unexpected error occurred.', + error: 'Internal Server Error', }; - case 'LIMIT_FILE_COUNT': - case 'LIMIT_PART_COUNT': - case 'LIMIT_FIELD_COUNT': - case 'LIMIT_FIELD_KEY': - case 'LIMIT_FIELD_VALUE': - case 'LIMIT_UNEXPECTED_FILE': + } + private fromHttpException(exception: HttpException): { + statusCode: number; + message: string | string[]; + error: string; + details?: ValidationErrorDetail[]; + stack?: string; + } { + const statusCode = exception.getStatus(); + const exceptionResponse = exception.getResponse(); + const stack = exception.stack; + // class-validator wraps errors as { message: string[], error: string } + if (typeof exceptionResponse === 'object' && exceptionResponse !== null) { + const res = exceptionResponse as Record; + const rawMessages = res['message']; + const messages: string[] = Array.isArray(rawMessages) + ? (rawMessages as string[]) + : typeof rawMessages === 'string' + ? [rawMessages] + : [exception.message]; + // Parse class-validator constraint objects when present + const details = this.extractValidationDetails(rawMessages); + return { + statusCode, + message: messages.length === 1 ? messages[0] : messages, + error: (res['error'] as string) ?? exception.message, + ...(details.length && { details }), + stack, + }; + } return { - statusCode: HttpStatus.BAD_REQUEST, - message: exception.message, - error: 'Bad Request', - stack: exception.stack, + statusCode, + message: typeof exceptionResponse === 'string' ? exceptionResponse : exception.message, + error: exception.message, + stack, }; - default: + } + private fromQueryFailedError(exception: QueryFailedError): { + statusCode: number; + message: string; + error: string; + stack?: string; + } { + const driverError = (exception as QueryFailedError & { + code?: string; + }).code; + // PostgreSQL unique-violation + if (driverError === '23505') { + return { + statusCode: HttpStatus.CONFLICT, + message: 'A record with the provided value already exists.', + error: 'Conflict', + stack: exception.stack, + }; + } + // PostgreSQL foreign-key violation + if (driverError === '23503') { + return { + statusCode: HttpStatus.UNPROCESSABLE_ENTITY, + message: 'Referenced resource does not exist.', + error: 'Unprocessable Entity', + stack: exception.stack, + }; + } + // Generic DB error – never expose query details in production return { - statusCode: HttpStatus.BAD_REQUEST, - message: 'Upload validation failed.', - error: 'Bad Request', - stack: exception.stack, + statusCode: HttpStatus.INTERNAL_SERVER_ERROR, + message: this.isProduction ? 'A database error occurred.' : exception.message, + error: 'Database Error', + stack: exception.stack, }; } - } - - /** - * Converts class-validator nested error objects into structured details when - * the raw message array contains constraint objects rather than plain strings. - */ - private extractValidationDetails(raw: unknown): ValidationErrorDetail[] { - if (!Array.isArray(raw)) return []; - - return raw.reduce((acc, item) => { - if ( - typeof item === 'object' && - item !== null && - 'property' in item && - 'constraints' in item - ) { - acc.push({ - property: (item as { property: string }).property, - constraints: (item as { constraints: Record }).constraints, - }); - } - return acc; - }, []); - } + private fromMulterError(exception: MulterError): { + statusCode: number; + message: string; + error: string; + stack?: string; + } { + switch (exception.code) { + case 'LIMIT_FILE_SIZE': + return { + statusCode: HttpStatus.PAYLOAD_TOO_LARGE, + message: 'Uploaded file exceeds the maximum allowed size.', + error: 'Payload Too Large', + stack: exception.stack, + }; + case 'LIMIT_FILE_COUNT': + case 'LIMIT_PART_COUNT': + case 'LIMIT_FIELD_COUNT': + case 'LIMIT_FIELD_KEY': + case 'LIMIT_FIELD_VALUE': + case 'LIMIT_UNEXPECTED_FILE': + return { + statusCode: HttpStatus.BAD_REQUEST, + message: exception.message, + error: 'Bad Request', + stack: exception.stack, + }; + default: + return { + statusCode: HttpStatus.BAD_REQUEST, + message: 'Upload validation failed.', + error: 'Bad Request', + stack: exception.stack, + }; + } + } + /** + * Converts class-validator nested error objects into structured details when + * the raw message array contains constraint objects rather than plain strings. + */ + private extractValidationDetails(raw: unknown): ValidationErrorDetail[] { + if (!Array.isArray(raw)) + return []; + return raw.reduce((acc, item) => { + if (typeof item === 'object' && + item !== null && + 'property' in item && + 'constraints' in item) { + acc.push({ + property: (item as { + property: string; + }).property, + constraints: (item as { + constraints: Record; + }).constraints, + }); + } + return acc; + }, []); + } } diff --git a/src/common/interceptors/logging.interceptor.spec.ts b/src/common/interceptors/logging.interceptor.spec.ts index 7c03d017..5997553b 100644 --- a/src/common/interceptors/logging.interceptor.spec.ts +++ b/src/common/interceptors/logging.interceptor.spec.ts @@ -1,33 +1,27 @@ import { LoggingInterceptor } from './logging.interceptor'; import { of, firstValueFrom } from 'rxjs'; - describe('LoggingInterceptor', () => { - it('attaches and propagates correlation ID header', async () => { - const interceptor = new LoggingInterceptor(); - - const req: any = { method: 'GET', url: '/spam', headers: {} }; - const headers: Record = {}; - const res: any = { - statusCode: 200, - setHeader: (name: string, value: string) => { - headers[name.toLowerCase()] = value; - }, - getHeader: (name: string) => headers[name.toLowerCase()], - }; - - const context: any = { - getType: () => 'http', - switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), - }; - - const next: any = { - handle: () => of({ success: true }), - }; - - await firstValueFrom(interceptor.intercept(context, next)); - - const correlationId = res.getHeader('x-request-id'); - expect(typeof correlationId).toBe('string'); - expect(correlationId).toMatch(/^cid-/); - }); + it('attaches and propagates correlation ID header', async () => { + const interceptor = new LoggingInterceptor(); + const req: unknown = { method: 'GET', url: '/spam', headers: {} }; + const headers: Record = {}; + const res: unknown = { + statusCode: 200, + setHeader: (name: string, value: string) => { + headers[name.toLowerCase()] = value; + }, + getHeader: (name: string) => headers[name.toLowerCase()], + }; + const context: unknown = { + getType: () => 'http', + switchToHttp: () => ({ getRequest: () => req, getResponse: () => res }), + }; + const next: unknown = { + handle: () => of({ success: true }), + }; + await firstValueFrom(interceptor.intercept(context, next)); + const correlationId = res.getHeader('x-request-id'); + expect(typeof correlationId).toBe('string'); + expect(correlationId).toMatch(/^cid-/); + }); }); diff --git a/src/common/interceptors/logging.interceptor.ts b/src/common/interceptors/logging.interceptor.ts index 33848feb..95ede6a9 100644 --- a/src/common/interceptors/logging.interceptor.ts +++ b/src/common/interceptors/logging.interceptor.ts @@ -2,30 +2,23 @@ import { Injectable, NestInterceptor, ExecutionContext, CallHandler, Logger } fr import { Observable, throwError } from 'rxjs'; import { tap, catchError } from 'rxjs/operators'; import { Request, Response } from 'express'; -import { - CORRELATION_ID_HEADER, - generateCorrelationId, - getCorrelationId, -} from '../utils/correlation.utils'; - +import { CORRELATION_ID_HEADER, generateCorrelationId, getCorrelationId, } from '../utils/correlation.utils'; export interface RequestLog { - requestId: string; - timestamp: string; - method: string; - url: string; - route: string; - ip: string; - userAgent: string; - userId?: string | number; - userRole?: string; + requestId: string; + timestamp: string; + method: string; + url: string; + route: string; + ip: string; + userAgent: string; + userId?: string | number; + userRole?: string; } - export interface ResponseLog extends RequestLog { - statusCode: number; - responseTimeMs: number; - contentLength?: number; + statusCode: number; + responseTimeMs: number; + contentLength?: number; } - /** * #154 – LoggingInterceptor * @@ -37,105 +30,87 @@ export interface ResponseLog extends RequestLog { */ @Injectable() export class LoggingInterceptor implements NestInterceptor { - private readonly logger = new Logger(LoggingInterceptor.name); - private readonly isProd = process.env.NODE_ENV === 'production'; - - intercept(context: ExecutionContext, next: CallHandler): Observable { - // Only intercept HTTP contexts (skip WebSockets, microservices, etc.) - if (context.getType() !== 'http') { - return next.handle(); + private readonly logger = new Logger(LoggingInterceptor.name); + private readonly isProd = process.env.NODE_ENV === 'production'; + intercept(context: ExecutionContext, next: CallHandler): Observable { + // Only intercept HTTP contexts (skip WebSockets, microservices, etc.) + if (context.getType() !== 'http') { + return next.handle(); + } + const httpCtx = context.switchToHttp(); + const request = httpCtx.getRequest; + }>(); + if (!request) { + return next.handle(); + } + const startTime = Date.now(); + const requestId = getCorrelationId() || generateCorrelationId(); + const response = httpCtx.getResponse(); + response?.setHeader(CORRELATION_ID_HEADER, requestId); + const baseLog: RequestLog = { + requestId, + timestamp: new Date().toISOString(), + method: request.method ?? 'UNKNOWN', + url: request.url, + route: request.route?.path ?? request.url, + ip: this.resolveClientIp(request), + userAgent: (request.headers['user-agent'] as string) ?? 'unknown', + ...(request.user?.id !== undefined && { userId: request.user.id as string | number }), + ...(request.user?.role !== undefined && { userRole: request.user.role as string }), + }; + this.logIncoming(baseLog); + return next.handle().pipe(tap(() => { + const res = httpCtx.getResponse(); + this.logOutgoing({ + ...baseLog, + statusCode: res.statusCode, + responseTimeMs: Date.now() - startTime, + contentLength: this.getContentLength(res), + }); + }), catchError((error: unknown) => { + const status = typeof error === 'object' && error !== null && 'status' in error + ? (error as { + status: number; + }).status + : 500; + this.logOutgoing({ + ...baseLog, + statusCode: status, + responseTimeMs: Date.now() - startTime, + }); + return throwError(() => error); + })); } - - const httpCtx = context.switchToHttp(); - const request = httpCtx.getRequest }>(); - - if (!request) { - return next.handle(); + // ─── Private helpers ─────────────────────────────────────────────────────── + private logIncoming(log: RequestLog): void { + const message = `→ ${log.method} ${log.url}`; + if (this.isProd) { + this.logger.log(JSON.stringify({ event: 'request.incoming', ...log })); + } + else { + this.logger.log(`${message} | id=${log.requestId} ip=${log.ip}${log.userId ? ` user=${log.userId}` : ''}`); + } } - - const startTime = Date.now(); - const requestId = getCorrelationId() || generateCorrelationId(); - - const response = httpCtx.getResponse(); - response?.setHeader(CORRELATION_ID_HEADER, requestId); - - const baseLog: RequestLog = { - requestId, - timestamp: new Date().toISOString(), - method: request.method ?? 'UNKNOWN', - url: request.url, - route: request.route?.path ?? request.url, - ip: this.resolveClientIp(request), - userAgent: (request.headers['user-agent'] as string) ?? 'unknown', - ...(request.user?.id !== undefined && { userId: request.user.id as string | number }), - ...(request.user?.role !== undefined && { userRole: request.user.role as string }), - }; - - this.logIncoming(baseLog); - - return next.handle().pipe( - tap(() => { - const res = httpCtx.getResponse(); - this.logOutgoing({ - ...baseLog, - statusCode: res.statusCode, - responseTimeMs: Date.now() - startTime, - contentLength: this.getContentLength(res), - }); - }), - catchError((error: unknown) => { - const status = - typeof error === 'object' && error !== null && 'status' in error - ? (error as { status: number }).status - : 500; - - this.logOutgoing({ - ...baseLog, - statusCode: status, - responseTimeMs: Date.now() - startTime, - }); - - return throwError(() => error); - }), - ); - } - - // ─── Private helpers ─────────────────────────────────────────────────────── - - private logIncoming(log: RequestLog): void { - const message = `→ ${log.method} ${log.url}`; - if (this.isProd) { - this.logger.log(JSON.stringify({ event: 'request.incoming', ...log })); - } else { - this.logger.log( - `${message} | id=${log.requestId} ip=${log.ip}${log.userId ? ` user=${log.userId}` : ''}`, - ); + private logOutgoing(log: ResponseLog): void { + const message = `← ${log.method} ${log.url} ${log.statusCode} ${log.responseTimeMs}ms`; + const level = log.statusCode >= 500 ? 'error' : log.statusCode >= 400 ? 'warn' : 'log'; + if (this.isProd) { + this.logger[level](JSON.stringify({ event: 'request.completed', ...log })); + } + else { + this.logger[level](`${message} | id=${log.requestId}${log.userId ? ` user=${log.userId}` : ''}`); + } } - } - - private logOutgoing(log: ResponseLog): void { - const message = `← ${log.method} ${log.url} ${log.statusCode} ${log.responseTimeMs}ms`; - const level = log.statusCode >= 500 ? 'error' : log.statusCode >= 400 ? 'warn' : 'log'; - - if (this.isProd) { - this.logger[level](JSON.stringify({ event: 'request.completed', ...log })); - } else { - this.logger[level]( - `${message} | id=${log.requestId}${log.userId ? ` user=${log.userId}` : ''}`, - ); + private resolveClientIp(request: Request): string { + const forwarded = request.headers['x-forwarded-for']; + if (typeof forwarded === 'string') { + return forwarded.split(',')[0].trim(); + } + return request.ip ?? request.socket?.remoteAddress ?? 'unknown'; } - } - - private resolveClientIp(request: Request): string { - const forwarded = request.headers['x-forwarded-for']; - if (typeof forwarded === 'string') { - return forwarded.split(',')[0].trim(); + private getContentLength(response: Response): number | undefined { + const header = response.getHeader('content-length'); + return header !== undefined ? Number(header) : undefined; } - return request.ip ?? request.socket?.remoteAddress ?? 'unknown'; - } - - private getContentLength(response: Response): number | undefined { - const header = response.getHeader('content-length'); - return header !== undefined ? Number(header) : undefined; - } } diff --git a/src/common/interceptors/monitoring.interceptor.ts b/src/common/interceptors/monitoring.interceptor.ts index e8f05f29..d780f91e 100644 --- a/src/common/interceptors/monitoring.interceptor.ts +++ b/src/common/interceptors/monitoring.interceptor.ts @@ -2,39 +2,32 @@ import { Injectable, NestInterceptor, ExecutionContext, CallHandler } from '@nes import { Observable } from 'rxjs'; import { tap } from 'rxjs/operators'; import { MetricsCollectionService } from '../../monitoring/metrics/metrics-collection.service'; - @Injectable() export class MonitoringInterceptor implements NestInterceptor { - constructor(private readonly metricsService: MetricsCollectionService) {} - - intercept(context: ExecutionContext, next: CallHandler): Observable { - const now = Date.now(); - const httpContext = context.switchToHttp(); - const request = httpContext.getRequest(); - - // Some requests might not be HTTP (e.g. WebSocket or Microservice), check if request exists - if (!request) { - return next.handle(); + constructor(private readonly metricsService: MetricsCollectionService) { } + intercept(context: ExecutionContext, next: CallHandler): Observable { + const now = Date.now(); + const httpContext = context.switchToHttp(); + const request = httpContext.getRequest(); + // Some requests might not be HTTP (e.g. WebSocket or Microservice), check if request exists + if (!request) { + return next.handle(); + } + const method = request.method || 'UNKNOWN'; + const route = request.route ? request.route.path : request.url; + return next.handle().pipe(tap({ + next: () => { + const response = httpContext.getResponse(); + const statusCode = response ? response.statusCode : 200; + const duration = (Date.now() - now) / 1000; + this.metricsService.recordHttpRequest(method, route, statusCode, duration); + }, + error: (error) => { + // Track errors too + const duration = (Date.now() - now) / 1000; + const status = error.status || 500; + this.metricsService.recordHttpRequest(method, route, status, duration); + }, + })); } - - const method = request.method || 'UNKNOWN'; - const route = request.route ? request.route.path : request.url; - - return next.handle().pipe( - tap({ - next: () => { - const response = httpContext.getResponse(); - const statusCode = response ? response.statusCode : 200; - const duration = (Date.now() - now) / 1000; - this.metricsService.recordHttpRequest(method, route, statusCode, duration); - }, - error: (error) => { - // Track errors too - const duration = (Date.now() - now) / 1000; - const status = error.status || 500; - this.metricsService.recordHttpRequest(method, route, status, duration); - }, - }), - ); - } } diff --git a/src/common/interceptors/response-transform.interceptor.ts b/src/common/interceptors/response-transform.interceptor.ts index ff650b01..e348cf00 100644 --- a/src/common/interceptors/response-transform.interceptor.ts +++ b/src/common/interceptors/response-transform.interceptor.ts @@ -2,55 +2,47 @@ import { Injectable, NestInterceptor, ExecutionContext, CallHandler } from '@nes import { Observable } from 'rxjs'; import { map } from 'rxjs/operators'; import { Response } from 'express'; - -export interface ApiResponse { - success: boolean; - message?: string; - data: T; - metadata?: Record; +export interface ApiResponse { + success: boolean; + message?: string; + data: T; + metadata?: Record; } - @Injectable() -export class ResponseTransformInterceptor implements NestInterceptor> { - intercept(context: ExecutionContext, next: CallHandler): Observable> { - const ctx = context.switchToHttp(); - const response = ctx.getResponse(); - - // Exclude file/stream responses (Content-Type or underlying stream) - const contentType = response.getHeader('Content-Type'); - if ( - response.headersSent || - (contentType && - (contentType.toString().includes('octet-stream') || - contentType.toString().includes('application/pdf') || - contentType.toString().startsWith('image/') || - contentType.toString().startsWith('audio/') || - contentType.toString().startsWith('video/'))) - ) { - // Return as Observable> by casting, since we skip transformation - return next.handle() as unknown as Observable>; - } - - return next.handle().pipe( - map((data: any) => { - // Allow controllers to return { data, message, metadata } for custom messages/metadata - let message: string | undefined; - let metadata: Record | undefined; - let responseData = data; - if (data && typeof data === 'object' && !Array.isArray(data)) { - if ('data' in data && typeof data.data !== 'undefined') { - responseData = data.data; - message = data.message; - metadata = data.metadata; - } +export class ResponseTransformInterceptor implements NestInterceptor> { + intercept(context: ExecutionContext, next: CallHandler): Observable> { + const ctx = context.switchToHttp(); + const response = ctx.getResponse(); + // Exclude file/stream responses (Content-Type or underlying stream) + const contentType = response.getHeader('Content-Type'); + if (response.headersSent || + (contentType && + (contentType.toString().includes('octet-stream') || + contentType.toString().includes('application/pdf') || + contentType.toString().startsWith('image/') || + contentType.toString().startsWith('audio/') || + contentType.toString().startsWith('video/')))) { + // Return as Observable> by casting, since we skip transformation + return next.handle() as unknown as Observable>; } - return { - success: true, - message, - data: responseData, - metadata, - }; - }), - ); - } + return next.handle().pipe(map((data: unknown) => { + // Allow controllers to return { data, message, metadata } for custom messages/metadata + let message: string | undefined; + let metadata: Record | undefined; + let responseData = data; + if (data && typeof data === 'object' && !Array.isArray(data)) { + if ('data' in data && typeof data.data !== 'undefined') { + responseData = data.data; + message = data.message; + metadata = data.metadata; + } + } + return { + success: true, + message, + data: responseData, + metadata, + }; + })); + } } diff --git a/src/common/interceptors/timeout.interceptor.ts b/src/common/interceptors/timeout.interceptor.ts index 6997dafb..156e0592 100644 --- a/src/common/interceptors/timeout.interceptor.ts +++ b/src/common/interceptors/timeout.interceptor.ts @@ -1,64 +1,48 @@ -import { - Injectable, - NestInterceptor, - ExecutionContext, - CallHandler, - BadGatewayException, - Logger, - Inject, -} from '@nestjs/common'; +import { Injectable, NestInterceptor, ExecutionContext, CallHandler, BadGatewayException, Logger, Inject, } from '@nestjs/common'; import { Observable, TimeoutError } from 'rxjs'; import { timeout, catchError } from 'rxjs/operators'; import { TimeoutConfigService } from '../timeout/timeout-config.service'; - export const DEFAULT_TIMEOUT = parseInt(process.env.REQUEST_TIMEOUT || '30000', 10); // 30 seconds default - export function Timeout(ms?: number): MethodDecorator { - return (target, propertyKey, descriptor) => { - Reflect.defineMetadata('timeout', ms, descriptor.value ?? target); - }; + return (target, propertyKey, descriptor) => { + Reflect.defineMetadata('timeout', ms, descriptor.value ?? target); + }; } - @Injectable() export class TimeoutInterceptor implements NestInterceptor { - private readonly logger = new Logger(TimeoutInterceptor.name); - - constructor(@Inject(TimeoutConfigService) private timeoutConfig: TimeoutConfigService) {} - - intercept(context: ExecutionContext, next: CallHandler): Observable { - const handler = context.getHandler(); - const customTimeout = Reflect.getMetadata('timeout', handler); - - const request = context.switchToHttp().getRequest(); - const method = request.method; - const url = request.url; - - // Determine timeout value with priority: decorator > config service > default - let timeoutValue: number; - if (customTimeout) { - timeoutValue = customTimeout; - this.logger.debug(`Using decorator timeout of ${timeoutValue}ms for ${method} ${url}`); - } else { - timeoutValue = this.timeoutConfig.getTimeoutForRequest(method, url); - this.logger.debug(`Using config timeout of ${timeoutValue}ms for ${method} ${url}`); - } - - return next.handle().pipe( - timeout(timeoutValue), - catchError((err) => { - if (err instanceof TimeoutError) { - this.logger.warn(`Request timeout: ${method} ${url} after ${timeoutValue}ms`); - throw new BadGatewayException({ - statusCode: 504, - message: `Request timed out after ${timeoutValue}ms`, - error: 'Timeout', - timestamp: new Date().toISOString(), - path: url, - method, - }); + private readonly logger = new Logger(TimeoutInterceptor.name); + constructor( + @Inject(TimeoutConfigService) + private timeoutConfig: TimeoutConfigService) { } + intercept(context: ExecutionContext, next: CallHandler): Observable { + const handler = context.getHandler(); + const customTimeout = Reflect.getMetadata('timeout', handler); + const request = context.switchToHttp().getRequest(); + const method = request.method; + const url = request.url; + // Determine timeout value with priority: decorator > config service > default + let timeoutValue: number; + if (customTimeout) { + timeoutValue = customTimeout; + this.logger.debug(`Using decorator timeout of ${timeoutValue}ms for ${method} ${url}`); + } + else { + timeoutValue = this.timeoutConfig.getTimeoutForRequest(method, url); + this.logger.debug(`Using config timeout of ${timeoutValue}ms for ${method} ${url}`); } - throw err; - }), - ); - } + return next.handle().pipe(timeout(timeoutValue), catchError((err) => { + if (err instanceof TimeoutError) { + this.logger.warn(`Request timeout: ${method} ${url} after ${timeoutValue}ms`); + throw new BadGatewayException({ + statusCode: 504, + message: `Request timed out after ${timeoutValue}ms`, + error: 'Timeout', + timestamp: new Date().toISOString(), + path: url, + method, + }); + } + throw err; + })); + } } diff --git a/src/common/lazy-loading/lazy-loading.module.ts b/src/common/lazy-loading/lazy-loading.module.ts index adc7aaa3..25c53f21 100644 --- a/src/common/lazy-loading/lazy-loading.module.ts +++ b/src/common/lazy-loading/lazy-loading.module.ts @@ -1,18 +1,17 @@ import { Module, Global, DynamicModule } from '@nestjs/common'; import { LazyModuleLoader } from './lazy-module-loader.service'; - @Global() @Module({ - providers: [LazyModuleLoader], - exports: [LazyModuleLoader], + providers: [LazyModuleLoader], + exports: [LazyModuleLoader], }) export class LazyLoadingModule { - static forRoot(): DynamicModule { - return { - module: LazyLoadingModule, - providers: [LazyModuleLoader], - exports: [LazyModuleLoader], - global: true, - }; - } + static forRoot(): DynamicModule { + return { + module: LazyLoadingModule, + providers: [LazyModuleLoader], + exports: [LazyModuleLoader], + global: true, + }; + } } diff --git a/src/common/lazy-loading/lazy-module-loader.service.ts b/src/common/lazy-loading/lazy-module-loader.service.ts index 9c0501c8..830ecf5f 100644 --- a/src/common/lazy-loading/lazy-module-loader.service.ts +++ b/src/common/lazy-loading/lazy-module-loader.service.ts @@ -1,222 +1,184 @@ import { Injectable, Logger, DynamicModule } from '@nestjs/common'; import { ModuleRef } from '@nestjs/core'; - export interface LazyModuleOptions { - moduleName: string; - featureFlag?: string; - dependencies?: string[]; + moduleName: string; + featureFlag?: string; + dependencies?: string[]; } - export interface ModuleLoadResult { - moduleName: string; - loadedAt: Date; - loadTimeMs: number; - success: boolean; - error?: Error; + moduleName: string; + loadedAt: Date; + loadTimeMs: number; + success: boolean; + error?: Error; } - @Injectable() export class LazyModuleLoader { - private readonly logger = new Logger(LazyModuleLoader.name); - private loadedModules = new Map>(); - private moduleLoadResults = new Map(); - private moduleRegistry = new Map Promise>(); - - constructor(private readonly moduleRef: ModuleRef) {} - - /** - * Register a lazy-loadable module - */ - register(moduleName: string, factory: () => Promise): void { - if (this.moduleRegistry.has(moduleName)) { - this.logger.warn(`Module ${moduleName} is already registered`); - return; + private readonly logger = new Logger(LazyModuleLoader.name); + private loadedModules = new Map>(); + private moduleLoadResults = new Map(); + private moduleRegistry = new Map Promise>(); + constructor(private readonly moduleRef: ModuleRef) { } + /** + * Register a lazy-loadable module + */ + register(moduleName: string, factory: () => Promise): void { + if (this.moduleRegistry.has(moduleName)) { + this.logger.warn(`Module ${moduleName} is already registered`); + return; + } + this.moduleRegistry.set(moduleName, factory); + this.logger.debug(`Registered lazy module: ${moduleName}`); } - this.moduleRegistry.set(moduleName, factory); - this.logger.debug(`Registered lazy module: ${moduleName}`); - } - - /** - * Load a module on-demand - */ - async load(moduleName: string): Promise { - // Check if already loaded - if (this.loadedModules.has(moduleName)) { - this.logger.debug(`Module ${moduleName} is already loaded`); - const loadedModule = this.loadedModules.get(moduleName); - if (loadedModule) { - return loadedModule; - } + /** + * Load a module on-demand + */ + async load(moduleName: string): Promise { + // Check if already loaded + if (this.loadedModules.has(moduleName)) { + this.logger.debug(`Module ${moduleName} is already loaded`); + const loadedModule = this.loadedModules.get(moduleName); + if (loadedModule) { + return loadedModule; + } + } + // Check if registered + const factory = this.moduleRegistry.get(moduleName); + if (!factory) { + this.logger.error(`Module ${moduleName} is not registered for lazy loading`); + return null; + } + // Load the module + const startTime = Date.now(); + this.logger.log(`Loading module: ${moduleName}...`); + const loadPromise = this.loadModuleInternal(moduleName, factory, startTime); + this.loadedModules.set(moduleName, loadPromise); + return loadPromise; } - - // Check if registered - const factory = this.moduleRegistry.get(moduleName); - if (!factory) { - this.logger.error(`Module ${moduleName} is not registered for lazy loading`); - return null; + /** + * Load multiple modules + */ + async loadMany(moduleNames: string[]): Promise { + const results = await Promise.all(moduleNames.map((name) => this.load(name))); + return results.filter((m): m is DynamicModule => m !== null); } - - // Load the module - const startTime = Date.now(); - this.logger.log(`Loading module: ${moduleName}...`); - - const loadPromise = this.loadModuleInternal(moduleName, factory, startTime); - this.loadedModules.set(moduleName, loadPromise); - - return loadPromise; - } - - /** - * Load multiple modules - */ - async loadMany(moduleNames: string[]): Promise { - const results = await Promise.all(moduleNames.map((name) => this.load(name))); - return results.filter((m): m is DynamicModule => m !== null); - } - - /** - * Check if a module is loaded - */ - isLoaded(moduleName: string): boolean { - return this.loadedModules.has(moduleName); - } - - /** - * Check if a module is registered - */ - isRegistered(moduleName: string): boolean { - return this.moduleRegistry.has(moduleName); - } - - /** - * Get list of loaded module names - */ - getLoadedModules(): string[] { - return Array.from(this.loadedModules.keys()); - } - - /** - * Get list of registered module names - */ - getRegisteredModules(): string[] { - return Array.from(this.moduleRegistry.keys()); - } - - /** - * Get load result for a module - */ - getLoadResult(moduleName: string): ModuleLoadResult | undefined { - return this.moduleLoadResults.get(moduleName); - } - - /** - * Get all load results - */ - getAllLoadResults(): ModuleLoadResult[] { - return Array.from(this.moduleLoadResults.values()); - } - - /** - * Get total load time for all loaded modules - */ - getTotalLoadTime(): number { - let total = 0; - for (const result of this.moduleLoadResults.values()) { - if (result.success) { - total += result.loadTimeMs; - } + /** + * Check if a module is loaded + */ + isLoaded(moduleName: string): boolean { + return this.loadedModules.has(moduleName); } - return total; - } - - /** - * Unload a module (for testing/memory management) - */ - async unload(moduleName: string): Promise { - if (!this.loadedModules.has(moduleName)) { - return false; + /** + * Check if a module is registered + */ + isRegistered(moduleName: string): boolean { + return this.moduleRegistry.has(moduleName); } - - this.loadedModules.delete(moduleName); - this.moduleLoadResults.delete(moduleName); - this.logger.log(`Unloaded module: ${moduleName}`); - return true; - } - - /** - * Preload modules that are likely to be needed - */ - async preload(moduleNames: string[]): Promise { - this.logger.log(`Preloading ${moduleNames.length} modules...`); - const startTime = Date.now(); - - await this.loadMany(moduleNames); - - const duration = Date.now() - startTime; - this.logger.log(`Preloaded ${moduleNames.length} modules in ${duration}ms`); - } - - /** - * Generate a report of all module loading activity - */ - generateReport(): { - registered: number; - loaded: number; - totalLoadTime: number; - averageLoadTime: number; - modules: ModuleLoadResult[]; - } { - const results = this.getAllLoadResults(); - const successfulLoads = results.filter((r) => r.success); - const totalLoadTime = this.getTotalLoadTime(); - - return { - registered: this.moduleRegistry.size, - loaded: this.loadedModules.size, - totalLoadTime, - averageLoadTime: successfulLoads.length > 0 ? totalLoadTime / successfulLoads.length : 0, - modules: results, - }; - } - - private async loadModuleInternal( - moduleName: string, - factory: () => Promise, - startTime: number, - ): Promise { - try { - const module = await factory(); - const loadTimeMs = Date.now() - startTime; - - const result: ModuleLoadResult = { - moduleName, - loadedAt: new Date(), - loadTimeMs, - success: true, - }; - - this.moduleLoadResults.set(moduleName, result); - this.logger.log(`Module ${moduleName} loaded in ${loadTimeMs}ms`); - - return module; - } catch (error) { - const loadTimeMs = Date.now() - startTime; - - const result: ModuleLoadResult = { - moduleName, - loadedAt: new Date(), - loadTimeMs, - success: false, - error: error as Error, - }; - - this.moduleLoadResults.set(moduleName, result); - this.logger.error( - `Failed to load module ${moduleName} after ${loadTimeMs}ms`, - (error as Error).message, - ); - - throw error; + /** + * Get list of loaded module names + */ + getLoadedModules(): string[] { + return Array.from(this.loadedModules.keys()); + } + /** + * Get list of registered module names + */ + getRegisteredModules(): string[] { + return Array.from(this.moduleRegistry.keys()); + } + /** + * Get load result for a module + */ + getLoadResult(moduleName: string): ModuleLoadResult | undefined { + return this.moduleLoadResults.get(moduleName); + } + /** + * Get all load results + */ + getAllLoadResults(): ModuleLoadResult[] { + return Array.from(this.moduleLoadResults.values()); + } + /** + * Get total load time for all loaded modules + */ + getTotalLoadTime(): number { + let total = 0; + for (const result of this.moduleLoadResults.values()) { + if (result.success) { + total += result.loadTimeMs; + } + } + return total; + } + /** + * Unload a module (for testing/memory management) + */ + async unload(moduleName: string): Promise { + if (!this.loadedModules.has(moduleName)) { + return false; + } + this.loadedModules.delete(moduleName); + this.moduleLoadResults.delete(moduleName); + this.logger.log(`Unloaded module: ${moduleName}`); + return true; + } + /** + * Preload modules that are likely to be needed + */ + async preload(moduleNames: string[]): Promise { + this.logger.log(`Preloading ${moduleNames.length} modules...`); + const startTime = Date.now(); + await this.loadMany(moduleNames); + const duration = Date.now() - startTime; + this.logger.log(`Preloaded ${moduleNames.length} modules in ${duration}ms`); + } + /** + * Generate a report of all module loading activity + */ + generateReport(): { + registered: number; + loaded: number; + totalLoadTime: number; + averageLoadTime: number; + modules: ModuleLoadResult[]; + } { + const results = this.getAllLoadResults(); + const successfulLoads = results.filter((r) => r.success); + const totalLoadTime = this.getTotalLoadTime(); + return { + registered: this.moduleRegistry.size, + loaded: this.loadedModules.size, + totalLoadTime, + averageLoadTime: successfulLoads.length > 0 ? totalLoadTime / successfulLoads.length : 0, + modules: results, + }; + } + private async loadModuleInternal(moduleName: string, factory: () => Promise, startTime: number): Promise { + try { + const module = await factory(); + const loadTimeMs = Date.now() - startTime; + const result: ModuleLoadResult = { + moduleName, + loadedAt: new Date(), + loadTimeMs, + success: true, + }; + this.moduleLoadResults.set(moduleName, result); + this.logger.log(`Module ${moduleName} loaded in ${loadTimeMs}ms`); + return module; + } + catch (error) { + const loadTimeMs = Date.now() - startTime; + const result: ModuleLoadResult = { + moduleName, + loadedAt: new Date(), + loadTimeMs, + success: false, + error: error as Error, + }; + this.moduleLoadResults.set(moduleName, result); + this.logger.error(`Failed to load module ${moduleName} after ${loadTimeMs}ms`, (error as Error).message); + throw error; + } } - } } diff --git a/src/common/lazy-loading/startup-logger.service.ts b/src/common/lazy-loading/startup-logger.service.ts index d59690f3..831553c3 100644 --- a/src/common/lazy-loading/startup-logger.service.ts +++ b/src/common/lazy-loading/startup-logger.service.ts @@ -1,192 +1,167 @@ import { Injectable, Logger, OnModuleInit, OnApplicationBootstrap } from '@nestjs/common'; - export interface StartupMetrics { - bootstrapStartTime: number; - bootstrapEndTime: number; - totalStartupTimeMs: number; - moduleInitTimeMs: number; - modulesLoaded: string[]; - modulesSkipped: string[]; - memoryUsage: NodeJS.MemoryUsage; + bootstrapStartTime: number; + bootstrapEndTime: number; + totalStartupTimeMs: number; + moduleInitTimeMs: number; + modulesLoaded: string[]; + modulesSkipped: string[]; + memoryUsage: NodeJS.MemoryUsage; } - export interface ModuleLoadMetric { - moduleName: string; - startTime: number; - endTime: number; - durationMs: number; - dependencies: string[]; + moduleName: string; + startTime: number; + endTime: number; + durationMs: number; + dependencies: string[]; } - @Injectable() export class StartupLogger implements OnModuleInit, OnApplicationBootstrap { - private readonly logger = new Logger(StartupLogger.name); - private bootstrapStartTime: number = 0; - private moduleInitStartTime: number = 0; - private moduleMetrics: Map = new Map(); - private modulesLoaded: string[] = []; - private modulesSkipped: string[] = []; - - constructor() { - this.bootstrapStartTime = Date.now(); - this.moduleInitStartTime = this.bootstrapStartTime; - } - - onModuleInit() { - this.moduleInitStartTime = Date.now(); - this.logger.log('Module initialization started'); - } - - onApplicationBootstrap() { - const metrics = this.generateMetrics(); - this.logStartupReport(metrics); - } - - /** - * Record a module being loaded - */ - recordModuleLoaded(moduleName: string, startTime: number, dependencies: string[] = []): void { - const endTime = Date.now(); - const durationMs = endTime - startTime; - - this.moduleMetrics.set(moduleName, { - moduleName, - startTime, - endTime, - durationMs, - dependencies, - }); - - this.modulesLoaded.push(moduleName); - this.logger.debug(`Module ${moduleName} loaded in ${durationMs}ms`); - } - - /** - * Record a module being skipped (feature flag disabled) - */ - recordModuleSkipped(moduleName: string, reason: string): void { - this.modulesSkipped.push(moduleName); - this.logger.debug(`Module ${moduleName} skipped: ${reason}`); - } - - /** - * Get all recorded metrics - */ - getMetrics(): StartupMetrics { - return this.generateMetrics(); - } - - /** - * Get module load metrics - */ - getModuleMetrics(): ModuleLoadMetric[] { - return Array.from(this.moduleMetrics.values()); - } - - /** - * Get slowest loading modules - */ - getSlowestModules(limit: number = 5): ModuleLoadMetric[] { - return this.getModuleMetrics() - .sort((a, b) => b.durationMs - a.durationMs) - .slice(0, limit); - } - - /** - * Get total startup time - */ - getTotalStartupTime(): number { - return Date.now() - this.bootstrapStartTime; - } - - /** - * Generate a complete startup report - */ - private generateMetrics(): StartupMetrics { - const now = Date.now(); - - return { - bootstrapStartTime: this.bootstrapStartTime, - bootstrapEndTime: now, - totalStartupTimeMs: now - this.bootstrapStartTime, - moduleInitTimeMs: now - this.moduleInitStartTime, - modulesLoaded: this.modulesLoaded, - modulesSkipped: this.modulesSkipped, - memoryUsage: process.memoryUsage(), - }; - } - - /** - * Log startup report - */ - private logStartupReport(metrics: StartupMetrics): void { - const totalModules = metrics.modulesLoaded.length + metrics.modulesSkipped.length; - const loadedCount = metrics.modulesLoaded.length; - const skippedCount = metrics.modulesSkipped.length; - - this.logger.log('========================================'); - this.logger.log(' STARTUP REPORT'); - this.logger.log('========================================'); - this.logger.log(`Total Startup Time: ${metrics.totalStartupTimeMs}ms`); - this.logger.log(`Module Init Time: ${metrics.moduleInitTimeMs}ms`); - this.logger.log(`Modules Loaded: ${loadedCount}/${totalModules}`); - this.logger.log(`Modules Skipped: ${skippedCount}/${totalModules}`); - - if (skippedCount > 0) { - this.logger.log(`Skipped Modules: ${metrics.modulesSkipped.join(', ')}`); + private readonly logger = new Logger(StartupLogger.name); + private bootstrapStartTime: number = 0; + private moduleInitStartTime: number = 0; + private moduleMetrics: Map = new Map(); + private modulesLoaded: string[] = []; + private modulesSkipped: string[] = []; + constructor() { + this.bootstrapStartTime = Date.now(); + this.moduleInitStartTime = this.bootstrapStartTime; } - - // Memory usage - const memoryMB = Math.round(metrics.memoryUsage.heapUsed / 1024 / 1024); - this.logger.log(`Memory Usage: ${memoryMB}MB`); - - // Slowest modules - const slowest = this.getSlowestModules(3); - if (slowest.length > 0) { - this.logger.log('Slowest Modules:'); - slowest.forEach((m) => { - this.logger.log(` - ${m.moduleName}: ${m.durationMs}ms`); - }); + onModuleInit() { + this.moduleInitStartTime = Date.now(); + this.logger.log('Module initialization started'); } - - this.logger.log('========================================'); - - // Performance warning if startup is slow - if (metrics.totalStartupTimeMs > 10000) { - this.logger.warn('Startup time exceeds 10 seconds - consider optimizing module loading'); + onApplicationBootstrap() { + const metrics = this.generateMetrics(); + this.logStartupReport(metrics); + } + /** + * Record a module being loaded + */ + recordModuleLoaded(moduleName: string, startTime: number, dependencies: string[] = []): void { + const endTime = Date.now(); + const durationMs = endTime - startTime; + this.moduleMetrics.set(moduleName, { + moduleName, + startTime, + endTime, + durationMs, + dependencies, + }); + this.modulesLoaded.push(moduleName); + this.logger.debug(`Module ${moduleName} loaded in ${durationMs}ms`); + } + /** + * Record a module being skipped (feature flag disabled) + */ + recordModuleSkipped(moduleName: string, reason: string): void { + this.modulesSkipped.push(moduleName); + this.logger.debug(`Module ${moduleName} skipped: ${reason}`); + } + /** + * Get all recorded metrics + */ + getMetrics(): StartupMetrics { + return this.generateMetrics(); + } + /** + * Get module load metrics + */ + getModuleMetrics(): ModuleLoadMetric[] { + return Array.from(this.moduleMetrics.values()); + } + /** + * Get slowest loading modules + */ + getSlowestModules(limit: number = 5): ModuleLoadMetric[] { + return this.getModuleMetrics() + .sort((a, b) => b.durationMs - a.durationMs) + .slice(0, limit); + } + /** + * Get total startup time + */ + getTotalStartupTime(): number { + return Date.now() - this.bootstrapStartTime; + } + /** + * Generate a complete startup report + */ + private generateMetrics(): StartupMetrics { + const now = Date.now(); + return { + bootstrapStartTime: this.bootstrapStartTime, + bootstrapEndTime: now, + totalStartupTimeMs: now - this.bootstrapStartTime, + moduleInitTimeMs: now - this.moduleInitStartTime, + modulesLoaded: this.modulesLoaded, + modulesSkipped: this.modulesSkipped, + memoryUsage: process.memoryUsage(), + }; + } + /** + * Log startup report + */ + private logStartupReport(metrics: StartupMetrics): void { + const totalModules = metrics.modulesLoaded.length + metrics.modulesSkipped.length; + const loadedCount = metrics.modulesLoaded.length; + const skippedCount = metrics.modulesSkipped.length; + this.logger.log('========================================'); + this.logger.log(' STARTUP REPORT'); + this.logger.log('========================================'); + this.logger.log(`Total Startup Time: ${metrics.totalStartupTimeMs}ms`); + this.logger.log(`Module Init Time: ${metrics.moduleInitTimeMs}ms`); + this.logger.log(`Modules Loaded: ${loadedCount}/${totalModules}`); + this.logger.log(`Modules Skipped: ${skippedCount}/${totalModules}`); + if (skippedCount > 0) { + this.logger.log(`Skipped Modules: ${metrics.modulesSkipped.join(', ')}`); + } + // Memory usage + const memoryMB = Math.round(metrics.memoryUsage.heapUsed / 1024 / 1024); + this.logger.log(`Memory Usage: ${memoryMB}MB`); + // Slowest modules + const slowest = this.getSlowestModules(3); + if (slowest.length > 0) { + this.logger.log('Slowest Modules:'); + slowest.forEach((m) => { + this.logger.log(` - ${m.moduleName}: ${m.durationMs}ms`); + }); + } + this.logger.log('========================================'); + // Performance warning if startup is slow + if (metrics.totalStartupTimeMs > 10000) { + this.logger.warn('Startup time exceeds 10 seconds - consider optimizing module loading'); + } + } + /** + * Generate JSON report for external monitoring + */ + generateJSONReport(): Record { + const metrics = this.generateMetrics(); + return { + timestamp: new Date().toISOString(), + startup: { + totalTimeMs: metrics.totalStartupTimeMs, + moduleInitTimeMs: metrics.moduleInitTimeMs, + }, + modules: { + total: metrics.modulesLoaded.length + metrics.modulesSkipped.length, + loaded: metrics.modulesLoaded.length, + skipped: metrics.modulesSkipped.length, + loadedList: metrics.modulesLoaded, + skippedList: metrics.modulesSkipped, + }, + memory: { + heapUsed: Math.round(metrics.memoryUsage.heapUsed / 1024 / 1024), + heapTotal: Math.round(metrics.memoryUsage.heapTotal / 1024 / 1024), + rss: Math.round(metrics.memoryUsage.rss / 1024 / 1024), + external: Math.round(metrics.memoryUsage.external / 1024 / 1024), + }, + moduleMetrics: this.getModuleMetrics().map((m) => ({ + name: m.moduleName, + loadTimeMs: m.durationMs, + dependencies: m.dependencies, + })), + }; } - } - - /** - * Generate JSON report for external monitoring - */ - generateJSONReport(): Record { - const metrics = this.generateMetrics(); - - return { - timestamp: new Date().toISOString(), - startup: { - totalTimeMs: metrics.totalStartupTimeMs, - moduleInitTimeMs: metrics.moduleInitTimeMs, - }, - modules: { - total: metrics.modulesLoaded.length + metrics.modulesSkipped.length, - loaded: metrics.modulesLoaded.length, - skipped: metrics.modulesSkipped.length, - loadedList: metrics.modulesLoaded, - skippedList: metrics.modulesSkipped, - }, - memory: { - heapUsed: Math.round(metrics.memoryUsage.heapUsed / 1024 / 1024), - heapTotal: Math.round(metrics.memoryUsage.heapTotal / 1024 / 1024), - rss: Math.round(metrics.memoryUsage.rss / 1024 / 1024), - external: Math.round(metrics.memoryUsage.external / 1024 / 1024), - }, - moduleMetrics: this.getModuleMetrics().map((m) => ({ - name: m.moduleName, - loadTimeMs: m.durationMs, - dependencies: m.dependencies, - })), - }; - } } diff --git a/src/common/middleware/csrf.middleware.ts b/src/common/middleware/csrf.middleware.ts index 5638fd20..d6d3634d 100644 --- a/src/common/middleware/csrf.middleware.ts +++ b/src/common/middleware/csrf.middleware.ts @@ -2,64 +2,49 @@ import { Injectable, NestMiddleware, UnauthorizedException } from '@nestjs/commo import { Request, Response, NextFunction } from 'express'; import { ConfigService } from '@nestjs/config'; import { CsrfService } from '../csrf/csrf.service'; - @Injectable() export class CsrfMiddleware implements NestMiddleware { - constructor( - private csrfService: CsrfService, - private configService: ConfigService, - ) {} - - use(req: Request, res: Response, next: NextFunction): void { - // Skip CSRF for GET, HEAD, OPTIONS requests - if (['GET', 'HEAD', 'OPTIONS'].includes(req.method)) { - this.generateCsrfToken(req, res); - return next(); + constructor(private csrfService: CsrfService, private configService: ConfigService) { } + use(req: Request, res: Response, next: NextFunction): void { + // Skip CSRF for GET, HEAD, OPTIONS requests + if (['GET', 'HEAD', 'OPTIONS'].includes(req.method)) { + this.generateCsrfToken(req, res); + return next(); + } + // Validate CSRF token for state-changing requests + if (['POST', 'PUT', 'DELETE', 'PATCH'].includes(req.method)) { + this.validateCsrfToken(req); + } + next(); } - - // Validate CSRF token for state-changing requests - if (['POST', 'PUT', 'DELETE', 'PATCH'].includes(req.method)) { - this.validateCsrfToken(req); + private generateCsrfToken(req: Request, res: Response): void { + const sessionId = this.getSessionId(req); + const existingToken = this.csrfService.getToken(sessionId); + if (existingToken) { + res.setHeader('X-CSRF-Token', existingToken); + (req as unknown).csrfToken = existingToken; + return; + } + // Generate new token + const token = this.csrfService.generateToken(sessionId); + res.setHeader('X-CSRF-Token', token); + (req as unknown).csrfToken = token; } - - next(); - } - - private generateCsrfToken(req: Request, res: Response): void { - const sessionId = this.getSessionId(req); - const existingToken = this.csrfService.getToken(sessionId); - - if (existingToken) { - res.setHeader('X-CSRF-Token', existingToken); - (req as any).csrfToken = existingToken; - return; + private validateCsrfToken(req: Request): void { + const sessionId = this.getSessionId(req); + const tokenFromHeader = req.headers['x-csrf-token'] as string; + const tokenFromBody = req.body?._csrf; + const submittedToken = tokenFromHeader || tokenFromBody; + if (!submittedToken || !this.csrfService.validateToken(sessionId, submittedToken)) { + throw new UnauthorizedException('Invalid CSRF token'); + } } - - // Generate new token - const token = this.csrfService.generateToken(sessionId); - res.setHeader('X-CSRF-Token', token); - (req as any).csrfToken = token; - } - - private validateCsrfToken(req: Request): void { - const sessionId = this.getSessionId(req); - - const tokenFromHeader = req.headers['x-csrf-token'] as string; - const tokenFromBody = req.body?._csrf; - const submittedToken = tokenFromHeader || tokenFromBody; - - if (!submittedToken || !this.csrfService.validateToken(sessionId, submittedToken)) { - throw new UnauthorizedException('Invalid CSRF token'); + private getSessionId(req: Request): string { + // Try to get session ID from session + if ((req as unknown).session?.id) { + return (req as unknown).session.id; + } + // Fallback to IP address (less secure, but better than nothing) + return req.ip || req.connection.remoteAddress || 'unknown'; } - } - - private getSessionId(req: Request): string { - // Try to get session ID from session - if ((req as any).session?.id) { - return (req as any).session.id; - } - - // Fallback to IP address (less secure, but better than nothing) - return req.ip || req.connection.remoteAddress || 'unknown'; - } } diff --git a/src/common/modules/api-versioning.module.ts b/src/common/modules/api-versioning.module.ts index ccac22c6..18f791fc 100644 --- a/src/common/modules/api-versioning.module.ts +++ b/src/common/modules/api-versioning.module.ts @@ -1,91 +1,57 @@ -import { - BadRequestException, - Injectable, - MiddlewareConsumer, - Module, - NestMiddleware, - NestModule, - NotAcceptableException, - RequestMethod, -} from '@nestjs/common'; +import { BadRequestException, Injectable, MiddlewareConsumer, Module, NestMiddleware, NestModule, NotAcceptableException, RequestMethod, } from '@nestjs/common'; import { APP_INTERCEPTOR } from '@nestjs/core'; import type { NextFunction, Request, Response } from 'express'; -import { - ApiVersionInterceptor, - API_VERSION_HEADER, - API_VERSION_HEADER_KEY, - DEFAULT_API_VERSION, - isVersionNeutralPath, - normalizeRequestedApiVersion, - SUPPORTED_API_VERSIONS, - VersionedRequest, -} from '../interceptors/api-version.interceptor'; - +import { ApiVersionInterceptor, API_VERSION_HEADER, API_VERSION_HEADER_KEY, DEFAULT_API_VERSION, isVersionNeutralPath, normalizeRequestedApiVersion, SUPPORTED_API_VERSIONS, VersionedRequest, } from '../interceptors/api-version.interceptor'; export const API_VERSIONING_DOCUMENTATION = [ - 'TeachLink uses header-based API versioning.', - `Send ${API_VERSION_HEADER}: ${DEFAULT_API_VERSION} on versioned endpoints.`, - `Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}.`, - 'Health, metrics, root, and webhook endpoints are version-neutral.', + 'TeachLink uses header-based API versioning.', + `Send ${API_VERSION_HEADER}: ${DEFAULT_API_VERSION} on versioned endpoints.`, + `Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}.`, + 'Health, metrics, root, and webhook endpoints are version-neutral.', ].join(' '); - @Injectable() export class ApiVersionValidationMiddleware implements NestMiddleware { - use( - req: Request & VersionedRequest & { headers: Record }, - res: Response, - next: NextFunction, - ): void { - const path = req.path || req.url || '/'; - - if (isVersionNeutralPath(path)) { - req.apiVersion = DEFAULT_API_VERSION; - next(); - return; + use(req: Request & VersionedRequest & { + headers: Record; + }, res: Response, next: NextFunction): void { + const path = req.path || req.url || '/'; + if (isVersionNeutralPath(path)) { + req.apiVersion = DEFAULT_API_VERSION; + next(); + return; + } + const rawVersion = req.headers[API_VERSION_HEADER_KEY]; + if (!rawVersion) { + throw new BadRequestException(`Missing required ${API_VERSION_HEADER} header. Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}`); + } + const normalizedVersion = normalizeRequestedApiVersion(rawVersion); + if (!normalizedVersion) { + throw new BadRequestException(`Invalid ${API_VERSION_HEADER} header value "${Array.isArray(rawVersion) ? rawVersion[0] : rawVersion}". Expected values like "1" or "v1".`); + } + if (!SUPPORTED_API_VERSIONS.includes(normalizedVersion)) { + throw new NotAcceptableException(`Unsupported API version "${normalizedVersion}". Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}`); + } + req.headers[API_VERSION_HEADER_KEY] = normalizedVersion; + req.apiVersion = normalizedVersion; + res.setHeader(API_VERSION_HEADER, normalizedVersion); + next(); } - - const rawVersion = req.headers[API_VERSION_HEADER_KEY]; - if (!rawVersion) { - throw new BadRequestException( - `Missing required ${API_VERSION_HEADER} header. Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}`, - ); - } - - const normalizedVersion = normalizeRequestedApiVersion(rawVersion); - if (!normalizedVersion) { - throw new BadRequestException( - `Invalid ${API_VERSION_HEADER} header value "${Array.isArray(rawVersion) ? rawVersion[0] : rawVersion}". Expected values like "1" or "v1".`, - ); - } - - if (!SUPPORTED_API_VERSIONS.includes(normalizedVersion)) { - throw new NotAcceptableException( - `Unsupported API version "${normalizedVersion}". Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}`, - ); - } - - req.headers[API_VERSION_HEADER_KEY] = normalizedVersion; - req.apiVersion = normalizedVersion; - res.setHeader(API_VERSION_HEADER, normalizedVersion); - next(); - } } - @Module({ - providers: [ - ApiVersionValidationMiddleware, - ApiVersionInterceptor, - { - provide: APP_INTERCEPTOR, - useClass: ApiVersionInterceptor, - }, - ], - exports: [ApiVersionValidationMiddleware, ApiVersionInterceptor], + providers: [ + ApiVersionValidationMiddleware, + ApiVersionInterceptor, + { + provide: APP_INTERCEPTOR, + useClass: ApiVersionInterceptor, + }, + ], + exports: [ApiVersionValidationMiddleware, ApiVersionInterceptor], }) export class ApiVersioningModule implements NestModule { - configure(consumer: MiddlewareConsumer): void { - consumer.apply(ApiVersionValidationMiddleware).forRoutes({ - path: '*', - method: RequestMethod.ALL, - }); - } + configure(consumer: MiddlewareConsumer): void { + consumer.apply(ApiVersionValidationMiddleware).forRoutes({ + path: '*', + method: RequestMethod.ALL, + }); + } } diff --git a/src/common/naming/naming.module.ts b/src/common/naming/naming.module.ts index 612e8971..4bf40235 100644 --- a/src/common/naming/naming.module.ts +++ b/src/common/naming/naming.module.ts @@ -1,10 +1,9 @@ // src/common/naming/naming.module.ts - import { Module } from '@nestjs/common'; import { NamingService } from './naming.service'; - @Module({ - providers: [NamingService], - exports: [NamingService], + providers: [NamingService], + exports: [NamingService], }) -export class NamingModule {} +export class NamingModule { +} diff --git a/src/common/naming/naming.service.ts b/src/common/naming/naming.service.ts index 159dc1f4..98d73954 100644 --- a/src/common/naming/naming.service.ts +++ b/src/common/naming/naming.service.ts @@ -1,27 +1,21 @@ // src/common/naming/naming.service.ts - import { Injectable } from '@nestjs/common'; - @Injectable() export class NamingService { - toCamelCase(input: string): string { - return input.toLowerCase().replace(/[-_\s]+(.)?/g, (_, c) => (c ? c.toUpperCase() : '')); - } - - toSnakeCase(input: string): string { - return input.replace(/[A-Z]/g, (letter) => `_${letter.toLowerCase()}`).replace(/^_/, ''); - } - - toPascalCase(input: string): string { - return input.replace(/(^\w|[-_\s]+\w)/g, (word) => word.replace(/[-_\s]+/, '').toUpperCase()); - } - - toKebabCase(input: string): string { - return input.replace(/[A-Z]/g, (letter) => `-${letter.toLowerCase()}`).replace(/^-/, ''); - } - - normalizeKey(input: string): string { - // Example: enforce DB-safe keys - return this.toSnakeCase(input).replace(/[^a-z0-9_]/g, ''); - } + toCamelCase(input: string): string { + return input.toLowerCase().replace(/[-_\s]+(.)?/g, (_, c) => (c ? c.toUpperCase() : '')); + } + toSnakeCase(input: string): string { + return input.replace(/[A-Z]/g, (letter) => `_${letter.toLowerCase()}`).replace(/^_/, ''); + } + toPascalCase(input: string): string { + return input.replace(/(^\w|[-_\s]+\w)/g, (word) => word.replace(/[-_\s]+/, '').toUpperCase()); + } + toKebabCase(input: string): string { + return input.replace(/[A-Z]/g, (letter) => `-${letter.toLowerCase()}`).replace(/^-/, ''); + } + normalizeKey(input: string): string { + // Example: enforce DB-safe keys + return this.toSnakeCase(input).replace(/[^a-z0-9_]/g, ''); + } } diff --git a/src/common/services/shutdown-state.service.ts b/src/common/services/shutdown-state.service.ts index e6f07206..0b4b708b 100644 --- a/src/common/services/shutdown-state.service.ts +++ b/src/common/services/shutdown-state.service.ts @@ -1,14 +1,11 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class ShutdownStateService { - private shuttingDown = false; - - markShuttingDown(): void { - this.shuttingDown = true; - } - - isShuttingDown(): boolean { - return this.shuttingDown; - } + private shuttingDown = false; + markShuttingDown(): void { + this.shuttingDown = true; + } + isShuttingDown(): boolean { + return this.shuttingDown; + } } diff --git a/src/common/timeout/timeout-config.service.ts b/src/common/timeout/timeout-config.service.ts index 65244092..dc7bbff9 100644 --- a/src/common/timeout/timeout-config.service.ts +++ b/src/common/timeout/timeout-config.service.ts @@ -1,105 +1,88 @@ import { Injectable } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; - export interface TimeoutConfig { - default: number; - endpoints: Record; - methods: Record; + default: number; + endpoints: Record; + methods: Record; } - @Injectable() export class TimeoutConfigService { - private readonly config: TimeoutConfig; - - constructor(private configService: ConfigService) { - this.config = this.loadConfig(); - } - - private loadConfig(): TimeoutConfig { - return { - default: parseInt(process.env.REQUEST_TIMEOUT || '30000', 10), // 30 seconds - endpoints: { - // API endpoints with custom timeouts - '/auth/login': 10000, // 10 seconds for login - '/auth/register': 15000, // 15 seconds for registration - '/payments/create-payment-intent': 20000, // 20 seconds for payment processing - '/media/upload': 120000, // 2 minutes for file upload - '/backup/create': 300000, // 5 minutes for backup creation - '/search': 15000, // 15 seconds for search - '/email-marketing/campaigns/send': 60000, // 1 minute for campaign sending - }, - methods: { - // HTTP method-specific timeouts - GET: 30000, // 30 seconds for GET requests - POST: 60000, // 1 minute for POST requests - PUT: 45000, // 45 seconds for PUT requests - DELETE: 30000, // 30 seconds for DELETE requests - PATCH: 45000, // 45 seconds for PATCH requests - }, - }; - } - - getDefaultTimeout(): number { - return this.config.default; - } - - getEndpointTimeout(path: string): number | null { - // Check for exact path match - if (this.config.endpoints[path]) { - return this.config.endpoints[path]; + private readonly config: TimeoutConfig; + constructor(private configService: ConfigService) { + this.config = this.loadConfig(); } - - // Check for pattern matches - for (const [pattern, timeout] of Object.entries(this.config.endpoints)) { - if (this.matchesPattern(path, pattern)) { - return timeout; - } + private loadConfig(): TimeoutConfig { + return { + default: parseInt(process.env.REQUEST_TIMEOUT || '30000', 10), // 30 seconds + endpoints: { + // API endpoints with custom timeouts + '/auth/login': 10000, // 10 seconds for login + '/auth/register': 15000, // 15 seconds for registration + '/payments/create-payment-intent': 20000, // 20 seconds for payment processing + '/media/upload': 120000, // 2 minutes for file upload + '/backup/create': 300000, // 5 minutes for backup creation + '/search': 15000, // 15 seconds for search + '/email-marketing/campaigns/send': 60000, // 1 minute for campaign sending + }, + methods: { + // HTTP method-specific timeouts + GET: 30000, // 30 seconds for GET requests + POST: 60000, // 1 minute for POST requests + PUT: 45000, // 45 seconds for PUT requests + DELETE: 30000, // 30 seconds for DELETE requests + PATCH: 45000, // 45 seconds for PATCH requests + }, + }; } - - return null; - } - - getMethodTimeout(method: string): number | null { - return this.config.methods[method.toUpperCase()] || null; - } - - getTimeoutForRequest(method: string, path: string): number { - // Priority: endpoint > method > default - const endpointTimeout = this.getEndpointTimeout(path); - if (endpointTimeout) { - return endpointTimeout; + getDefaultTimeout(): number { + return this.config.default; } - - const methodTimeout = this.getMethodTimeout(method); - if (methodTimeout) { - return methodTimeout; + getEndpointTimeout(path: string): number | null { + // Check for exact path match + if (this.config.endpoints[path]) { + return this.config.endpoints[path]; + } + // Check for pattern matches + for (const [pattern, timeout] of Object.entries(this.config.endpoints)) { + if (this.matchesPattern(path, pattern)) { + return timeout; + } + } + return null; } - - return this.getDefaultTimeout(); - } - - private matchesPattern(path: string, pattern: string): boolean { - // Simple pattern matching - can be enhanced with regex - if (pattern.includes('*')) { - const regexPattern = pattern.replace(/\*/g, '.*'); - return new RegExp(`^${regexPattern}$`).test(path); + getMethodTimeout(method: string): number | null { + return this.config.methods[method.toUpperCase()] || null; + } + getTimeoutForRequest(method: string, path: string): number { + // Priority: endpoint > method > default + const endpointTimeout = this.getEndpointTimeout(path); + if (endpointTimeout) { + return endpointTimeout; + } + const methodTimeout = this.getMethodTimeout(method); + if (methodTimeout) { + return methodTimeout; + } + return this.getDefaultTimeout(); + } + private matchesPattern(path: string, pattern: string): boolean { + // Simple pattern matching - can be enhanced with regex + if (pattern.includes('*')) { + const regexPattern = pattern.replace(/\*/g, '.*'); + return new RegExp(`^${regexPattern}$`).test(path); + } + return false; + } + updateEndpointTimeout(path: string, timeout: number): void { + this.config.endpoints[path] = timeout; + } + updateMethodTimeout(method: string, timeout: number): void { + this.config.methods[method.toUpperCase()] = timeout; + } + updateDefaultTimeout(timeout: number): void { + this.config.default = timeout; + } + getConfig(): TimeoutConfig { + return { ...this.config }; } - return false; - } - - updateEndpointTimeout(path: string, timeout: number): void { - this.config.endpoints[path] = timeout; - } - - updateMethodTimeout(method: string, timeout: number): void { - this.config.methods[method.toUpperCase()] = timeout; - } - - updateDefaultTimeout(timeout: number): void { - this.config.default = timeout; - } - - getConfig(): TimeoutConfig { - return { ...this.config }; - } } diff --git a/src/common/timeout/timeout.controller.ts b/src/common/timeout/timeout.controller.ts index 44270e96..81b3bf23 100644 --- a/src/common/timeout/timeout.controller.ts +++ b/src/common/timeout/timeout.controller.ts @@ -2,76 +2,95 @@ import { Controller, Get, Put, UseGuards, Body, HttpCode, HttpStatus } from '@ne import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth } from '@nestjs/swagger'; import { JwtAuthGuard } from '../../auth/guards/jwt-auth.guard'; import { TimeoutConfigService, TimeoutConfig } from './timeout-config.service'; - @ApiTags('Timeout Configuration') @ApiBearerAuth() @Controller('timeout') @UseGuards(JwtAuthGuard) export class TimeoutController { - constructor(private readonly timeoutConfig: TimeoutConfigService) {} - - @Get('config') - @ApiOperation({ summary: 'Get current timeout configuration' }) - @ApiResponse({ status: 200, description: 'Timeout configuration retrieved successfully' }) - getConfig(): TimeoutConfig { - return this.timeoutConfig.getConfig(); - } - - @Put('default') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Update default timeout' }) - @ApiResponse({ status: 200, description: 'Default timeout updated successfully' }) - updateDefaultTimeout(@Body() body: { timeout: number }): { message: string; timeout: number } { - this.timeoutConfig.updateDefaultTimeout(body.timeout); - return { - message: 'Default timeout updated successfully', - timeout: body.timeout, - }; - } - - @Put('endpoint') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Update endpoint timeout' }) - @ApiResponse({ status: 200, description: 'Endpoint timeout updated successfully' }) - updateEndpointTimeout(@Body() body: { path: string; timeout: number }): { message: string } { - this.timeoutConfig.updateEndpointTimeout(body.path, body.timeout); - return { - message: `Endpoint timeout updated successfully for ${body.path}`, - }; - } - - @Put('method') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Update HTTP method timeout' }) - @ApiResponse({ status: 200, description: 'Method timeout updated successfully' }) - updateMethodTimeout(@Body() body: { method: string; timeout: number }): { message: string } { - this.timeoutConfig.updateMethodTimeout(body.method, body.timeout); - return { - message: `Method timeout updated successfully for ${body.method}`, - }; - } - - @Get('check') - @ApiOperation({ summary: 'Get timeout for a specific request' }) - @ApiResponse({ status: 200, description: 'Timeout calculated successfully' }) - checkTimeout(@Body() body: { method: string; path: string }): { - timeout: number; - source: string; - } { - const timeout = this.timeoutConfig.getTimeoutForRequest(body.method, body.path); - const endpointTimeout = this.timeoutConfig.getEndpointTimeout(body.path); - const methodTimeout = this.timeoutConfig.getMethodTimeout(body.method); - - let source = 'default'; - if (endpointTimeout) { - source = 'endpoint'; - } else if (methodTimeout) { - source = 'method'; + constructor(private readonly timeoutConfig: TimeoutConfigService) { } + @Get('config') + @ApiOperation({ summary: 'Get current timeout configuration' }) + @ApiResponse({ status: 200, description: 'Timeout configuration retrieved successfully' }) + getConfig(): TimeoutConfig { + return this.timeoutConfig.getConfig(); + } + @Put('default') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Update default timeout' }) + @ApiResponse({ status: 200, description: 'Default timeout updated successfully' }) + updateDefaultTimeout( + @Body() + body: { + timeout: number; + }): { + message: string; + timeout: number; + } { + this.timeoutConfig.updateDefaultTimeout(body.timeout); + return { + message: 'Default timeout updated successfully', + timeout: body.timeout, + }; + } + @Put('endpoint') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Update endpoint timeout' }) + @ApiResponse({ status: 200, description: 'Endpoint timeout updated successfully' }) + updateEndpointTimeout( + @Body() + body: { + path: string; + timeout: number; + }): { + message: string; + } { + this.timeoutConfig.updateEndpointTimeout(body.path, body.timeout); + return { + message: `Endpoint timeout updated successfully for ${body.path}`, + }; + } + @Put('method') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Update HTTP method timeout' }) + @ApiResponse({ status: 200, description: 'Method timeout updated successfully' }) + updateMethodTimeout( + @Body() + body: { + method: string; + timeout: number; + }): { + message: string; + } { + this.timeoutConfig.updateMethodTimeout(body.method, body.timeout); + return { + message: `Method timeout updated successfully for ${body.method}`, + }; + } + @Get('check') + @ApiOperation({ summary: 'Get timeout for a specific request' }) + @ApiResponse({ status: 200, description: 'Timeout calculated successfully' }) + checkTimeout( + @Body() + body: { + method: string; + path: string; + }): { + timeout: number; + source: string; + } { + const timeout = this.timeoutConfig.getTimeoutForRequest(body.method, body.path); + const endpointTimeout = this.timeoutConfig.getEndpointTimeout(body.path); + const methodTimeout = this.timeoutConfig.getMethodTimeout(body.method); + let source = 'default'; + if (endpointTimeout) { + source = 'endpoint'; + } + else if (methodTimeout) { + source = 'method'; + } + return { + timeout, + source, + }; } - - return { - timeout, - source, - }; - } } diff --git a/src/common/timeout/timeout.module.ts b/src/common/timeout/timeout.module.ts index e2f60ba8..bc835423 100644 --- a/src/common/timeout/timeout.module.ts +++ b/src/common/timeout/timeout.module.ts @@ -2,10 +2,10 @@ import { Module } from '@nestjs/common'; import { TimeoutConfigService } from './timeout-config.service'; import { TimeoutController } from './timeout.controller'; import { TimeoutExampleController } from '../examples/timeout-example.controller'; - @Module({ - providers: [TimeoutConfigService], - controllers: [TimeoutController, TimeoutExampleController], - exports: [TimeoutConfigService], + providers: [TimeoutConfigService], + controllers: [TimeoutController, TimeoutExampleController], + exports: [TimeoutConfigService], }) -export class TimeoutModule {} +export class TimeoutModule { +} diff --git a/src/common/types/file.types.ts b/src/common/types/file.types.ts index 32b89dbb..420748f1 100644 --- a/src/common/types/file.types.ts +++ b/src/common/types/file.types.ts @@ -3,14 +3,14 @@ * Used across CDN, Media, and file upload services */ export interface UploadedFile { - fieldname: string; - originalname: string; - encoding: string; - mimetype: string; - size: number; - destination?: string; - filename?: string; - path?: string; - buffer: Buffer; - stream?: NodeJS.ReadableStream; + fieldname: string; + originalname: string; + encoding: string; + mimetype: string; + size: number; + destination?: string; + filename?: string; + path?: string; + buffer: Buffer; + stream?: NodeJS.ReadableStream; } diff --git a/src/common/types/request-with-locale.ts b/src/common/types/request-with-locale.ts index 967d236d..77fdb084 100644 --- a/src/common/types/request-with-locale.ts +++ b/src/common/types/request-with-locale.ts @@ -1,3 +1,4 @@ import { Request } from 'express'; - -export type RequestWithLocale = Request & { resolvedLocale?: string }; +export type RequestWithLocale = Request & { + resolvedLocale?: string; +}; diff --git a/src/common/utils/bull-redis.util.ts b/src/common/utils/bull-redis.util.ts index f09c2553..ce80bfe1 100644 --- a/src/common/utils/bull-redis.util.ts +++ b/src/common/utils/bull-redis.util.ts @@ -1,19 +1,15 @@ import Redis, { RedisOptions } from 'ioredis'; - export const createBullRedisClient = (type: string, redisOpts?: RedisOptions) => { - const options: RedisOptions = { - ...(redisOpts ?? {}), - }; - - if (type !== 'client') { - options.enableReadyCheck = false; - options.maxRetriesPerRequest = null; - } - - const client = new Redis(options); - client.on('error', () => { - // Avoid unhandled error events while Redis is unavailable. - }); - - return client; + const options: RedisOptions = { + ...(redisOpts ?? {}), + }; + if (type !== 'client') { + options.enableReadyCheck = false; + options.maxRetriesPerRequest = null; + } + const client = new Redis(options); + client.on('error', () => { + // Avoid unhandled error events while Redis is unavailable. + }); + return client; }; diff --git a/src/common/utils/correlation.utils.spec.ts b/src/common/utils/correlation.utils.spec.ts index cc313122..1fa4b219 100644 --- a/src/common/utils/correlation.utils.spec.ts +++ b/src/common/utils/correlation.utils.spec.ts @@ -1,50 +1,40 @@ -import { - correlationMiddleware, - getCorrelationId, - injectCorrelationIdToHeaders, - CORRELATION_ID_HEADER, -} from './correlation.utils'; - +import { correlationMiddleware, getCorrelationId, injectCorrelationIdToHeaders, CORRELATION_ID_HEADER, } from './correlation.utils'; describe('correlation.utils', () => { - it('generates and propagates correlation ID through middleware', (done) => { - const req: any = { method: 'GET', url: '/test', headers: {} }; - const headers: Record = {}; - const res: any = { - setHeader: (name: string, value: string) => { - headers[name.toLowerCase()] = value; - }, - getHeader: (name: string) => headers[name.toLowerCase()], - }; - - correlationMiddleware(req, res, () => { - const id = getCorrelationId(); - expect(typeof id).toBe('string'); - expect(res.getHeader(CORRELATION_ID_HEADER)).toBe(id); - done(); + it('generates and propagates correlation ID through middleware', (done) => { + const req: unknown = { method: 'GET', url: '/test', headers: {} }; + const headers: Record = {}; + const res: unknown = { + setHeader: (name: string, value: string) => { + headers[name.toLowerCase()] = value; + }, + getHeader: (name: string) => headers[name.toLowerCase()], + }; + correlationMiddleware(req, res, () => { + const id = getCorrelationId(); + expect(typeof id).toBe('string'); + expect(res.getHeader(CORRELATION_ID_HEADER)).toBe(id); + done(); + }); }); - }); - - it('respects incoming x-request-id header', (done) => { - const incomingId = 'test-correlation-id'; - const req: any = { method: 'GET', url: '/test', headers: { 'x-request-id': incomingId } }; - const headers: Record = {}; - const res: any = { - setHeader: (name: string, value: string) => { - headers[name.toLowerCase()] = value; - }, - getHeader: (name: string) => headers[name.toLowerCase()], - }; - - correlationMiddleware(req, res, () => { - expect(getCorrelationId()).toBe(incomingId); - expect(res.getHeader(CORRELATION_ID_HEADER)).toBe(incomingId); - done(); + it('respects incoming x-request-id header', (done) => { + const incomingId = 'test-correlation-id'; + const req: unknown = { method: 'GET', url: '/test', headers: { 'x-request-id': incomingId } }; + const headers: Record = {}; + const res: unknown = { + setHeader: (name: string, value: string) => { + headers[name.toLowerCase()] = value; + }, + getHeader: (name: string) => headers[name.toLowerCase()], + }; + correlationMiddleware(req, res, () => { + expect(getCorrelationId()).toBe(incomingId); + expect(res.getHeader(CORRELATION_ID_HEADER)).toBe(incomingId); + done(); + }); + }); + it('injects correlation header into outgoing request headers', () => { + const custom = injectCorrelationIdToHeaders({ Authorization: 'Bearer token' }, 'cid-1'); + expect(custom[CORRELATION_ID_HEADER]).toBe('cid-1'); + expect(custom.Authorization).toBe('Bearer token'); }); - }); - - it('injects correlation header into outgoing request headers', () => { - const custom = injectCorrelationIdToHeaders({ Authorization: 'Bearer token' }, 'cid-1'); - expect(custom[CORRELATION_ID_HEADER]).toBe('cid-1'); - expect(custom.Authorization).toBe('Bearer token'); - }); }); diff --git a/src/common/utils/correlation.utils.ts b/src/common/utils/correlation.utils.ts index fb8a171f..cd582a1e 100644 --- a/src/common/utils/correlation.utils.ts +++ b/src/common/utils/correlation.utils.ts @@ -1,51 +1,39 @@ import { AsyncLocalStorage } from 'async_hooks'; import { Request, Response, NextFunction } from 'express'; - export const CORRELATION_ID_HEADER = 'x-request-id'; - export interface CorrelationContext { - correlationId: string; + correlationId: string; } - const correlationStorage = new AsyncLocalStorage(); - export function generateCorrelationId(): string { - return `cid-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 10)}`; + return `cid-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 10)}`; } - export function getCorrelationId(): string | undefined { - const store = correlationStorage.getStore(); - return store?.correlationId; + const store = correlationStorage.getStore(); + return store?.correlationId; } - export function setCorrelationId(req: Request, res: Response, correlationId: string): void { - (req as Request & { correlationId?: string }).correlationId = correlationId; - res.setHeader(CORRELATION_ID_HEADER, correlationId); + (req as Request & { + correlationId?: string; + }).correlationId = correlationId; + res.setHeader(CORRELATION_ID_HEADER, correlationId); } - export function correlationMiddleware(req: Request, res: Response, next: NextFunction): void { - const incoming = - (req.headers[CORRELATION_ID_HEADER] as string) || (req.headers['x-correlation-id'] as string); - const correlationId = incoming || generateCorrelationId(); - - correlationStorage.run({ correlationId }, () => { - setCorrelationId(req, res, correlationId); - next(); - }); + const incoming = (req.headers[CORRELATION_ID_HEADER] as string) || (req.headers['x-correlation-id'] as string); + const correlationId = incoming || generateCorrelationId(); + correlationStorage.run({ correlationId }, () => { + setCorrelationId(req, res, correlationId); + next(); + }); } - export function runWithCorrelationId(callback: () => T, correlationId?: string): T { - const id = correlationId || generateCorrelationId(); - return correlationStorage.run({ correlationId: id }, callback); + const id = correlationId || generateCorrelationId(); + return correlationStorage.run({ correlationId: id }, callback); } - -export function injectCorrelationIdToHeaders( - headers: Record = {}, - correlationId?: string, -): Record { - const id = correlationId || getCorrelationId() || generateCorrelationId(); - return { - ...headers, - [CORRELATION_ID_HEADER]: id, - }; +export function injectCorrelationIdToHeaders(headers: Record = {}, correlationId?: string): Record { + const id = correlationId || getCorrelationId() || generateCorrelationId(); + return { + ...headers, + [CORRELATION_ID_HEADER]: id, + }; } diff --git a/src/common/utils/pagination.util.spec.ts b/src/common/utils/pagination.util.spec.ts index 52a4e1f0..26a46f4a 100644 --- a/src/common/utils/pagination.util.spec.ts +++ b/src/common/utils/pagination.util.spec.ts @@ -1,354 +1,276 @@ import { BadRequestException } from '@nestjs/common'; -import { - generateCursor, - decodeCursor, - validateCursor, - paginateWithCursor, - paginate, -} from './pagination.util'; +import { generateCursor, decodeCursor, validateCursor, paginateWithCursor, paginate, } from './pagination.util'; import { SortOrder, CursorDirection } from '../dto/pagination.dto'; - // --------------------------------------------------------------------------- // Helpers // --------------------------------------------------------------------------- - -function buildMockQueryBuilder(items: any[]) { - const qb: any = { - alias: 'entity', - andWhere: jest.fn().mockReturnThis(), - orderBy: jest.fn().mockReturnThis(), - addOrderBy: jest.fn().mockReturnThis(), - skip: jest.fn().mockReturnThis(), - take: jest.fn().mockReturnThis(), - getCount: jest.fn().mockResolvedValue(items.length), - getMany: jest.fn().mockResolvedValue(items), - }; - return qb; +function buildMockQueryBuilder(items: unknown[]) { + const qb: unknown = { + alias: 'entity', + andWhere: jest.fn().mockReturnThis(), + orderBy: jest.fn().mockReturnThis(), + addOrderBy: jest.fn().mockReturnThis(), + skip: jest.fn().mockReturnThis(), + take: jest.fn().mockReturnThis(), + getCount: jest.fn().mockResolvedValue(items.length), + getMany: jest.fn().mockResolvedValue(items), + }; + return qb; } - function makeItems(count: number) { - return Array.from({ length: count }, (_, i) => ({ - id: `id-${String(i + 1).padStart(3, '0')}`, - createdAt: new Date(2024, 0, count - i).toISOString(), - title: `Course ${i + 1}`, - })); + return Array.from({ length: count }, (_, i) => ({ + id: `id-${String(i + 1).padStart(3, '0')}`, + createdAt: new Date(2024, 0, count - i).toISOString(), + title: `Course ${i + 1}`, + })); } - // --------------------------------------------------------------------------- // generateCursor // --------------------------------------------------------------------------- - describe('generateCursor', () => { - it('produces a non-empty base64 string', () => { - const entity = { id: 'abc-123', createdAt: '2024-06-01T00:00:00.000Z' }; - const cursor = generateCursor(entity, 'createdAt'); - expect(typeof cursor).toBe('string'); - expect(cursor.length).toBeGreaterThan(0); - }); - - it('encodes id and sortValue', () => { - const entity = { id: 'uuid-x', createdAt: '2024-06-01T00:00:00.000Z' }; - const cursor = generateCursor(entity, 'createdAt'); - const decoded = JSON.parse(Buffer.from(cursor, 'base64').toString('utf8')); - expect(decoded.id).toBe('uuid-x'); - expect(decoded.sortValue).toBe('2024-06-01T00:00:00.000Z'); - }); - - it('generates different cursors for different entities', () => { - const e1 = { id: 'id-1', createdAt: '2024-01-01' }; - const e2 = { id: 'id-2', createdAt: '2024-01-02' }; - expect(generateCursor(e1, 'createdAt')).not.toBe(generateCursor(e2, 'createdAt')); - }); - - it('generates different cursors for same createdAt but different ids', () => { - const ts = '2024-01-01T00:00:00.000Z'; - const e1 = { id: 'id-1', createdAt: ts }; - const e2 = { id: 'id-2', createdAt: ts }; - expect(generateCursor(e1, 'createdAt')).not.toBe(generateCursor(e2, 'createdAt')); - }); + it('produces a non-empty base64 string', () => { + const entity = { id: 'abc-123', createdAt: '2024-06-01T00:00:00.000Z' }; + const cursor = generateCursor(entity, 'createdAt'); + expect(typeof cursor).toBe('string'); + expect(cursor.length).toBeGreaterThan(0); + }); + it('encodes id and sortValue', () => { + const entity = { id: 'uuid-x', createdAt: '2024-06-01T00:00:00.000Z' }; + const cursor = generateCursor(entity, 'createdAt'); + const decoded = JSON.parse(Buffer.from(cursor, 'base64').toString('utf8')); + expect(decoded.id).toBe('uuid-x'); + expect(decoded.sortValue).toBe('2024-06-01T00:00:00.000Z'); + }); + it('generates different cursors for different entities', () => { + const e1 = { id: 'id-1', createdAt: '2024-01-01' }; + const e2 = { id: 'id-2', createdAt: '2024-01-02' }; + expect(generateCursor(e1, 'createdAt')).not.toBe(generateCursor(e2, 'createdAt')); + }); + it('generates different cursors for same createdAt but different ids', () => { + const ts = '2024-01-01T00:00:00.000Z'; + const e1 = { id: 'id-1', createdAt: ts }; + const e2 = { id: 'id-2', createdAt: ts }; + expect(generateCursor(e1, 'createdAt')).not.toBe(generateCursor(e2, 'createdAt')); + }); }); - // --------------------------------------------------------------------------- // decodeCursor // --------------------------------------------------------------------------- - describe('decodeCursor', () => { - it('round-trips correctly with generateCursor', () => { - const entity = { id: 'test-id', createdAt: '2024-03-15T12:00:00.000Z' }; - const cursor = generateCursor(entity, 'createdAt'); - const decoded = decodeCursor(cursor); - expect(decoded.id).toBe('test-id'); - expect(decoded.sortValue).toBe('2024-03-15T12:00:00.000Z'); - }); - - it('throws BadRequestException for non-base64 input', () => { - expect(() => decodeCursor('!!!not-base64!!!')).toThrow(BadRequestException); - }); - - it('throws BadRequestException when id field is missing', () => { - const bad = Buffer.from(JSON.stringify({ sortValue: '2024-01-01' })).toString('base64'); - expect(() => decodeCursor(bad)).toThrow(BadRequestException); - }); - - it('throws BadRequestException when sortValue field is missing', () => { - const bad = Buffer.from(JSON.stringify({ id: 'some-id' })).toString('base64'); - expect(() => decodeCursor(bad)).toThrow(BadRequestException); - }); - - it('throws BadRequestException for valid base64 but invalid JSON', () => { - const bad = Buffer.from('not json at all').toString('base64'); - expect(() => decodeCursor(bad)).toThrow(BadRequestException); - }); + it('round-trips correctly with generateCursor', () => { + const entity = { id: 'test-id', createdAt: '2024-03-15T12:00:00.000Z' }; + const cursor = generateCursor(entity, 'createdAt'); + const decoded = decodeCursor(cursor); + expect(decoded.id).toBe('test-id'); + expect(decoded.sortValue).toBe('2024-03-15T12:00:00.000Z'); + }); + it('throws BadRequestException for non-base64 input', () => { + expect(() => decodeCursor('!!!not-base64!!!')).toThrow(BadRequestException); + }); + it('throws BadRequestException when id field is missing', () => { + const bad = Buffer.from(JSON.stringify({ sortValue: '2024-01-01' })).toString('base64'); + expect(() => decodeCursor(bad)).toThrow(BadRequestException); + }); + it('throws BadRequestException when sortValue field is missing', () => { + const bad = Buffer.from(JSON.stringify({ id: 'some-id' })).toString('base64'); + expect(() => decodeCursor(bad)).toThrow(BadRequestException); + }); + it('throws BadRequestException for valid base64 but invalid JSON', () => { + const bad = Buffer.from('not json at all').toString('base64'); + expect(() => decodeCursor(bad)).toThrow(BadRequestException); + }); }); - // --------------------------------------------------------------------------- // validateCursor // --------------------------------------------------------------------------- - describe('validateCursor', () => { - it('returns true for a cursor produced by generateCursor', () => { - const cursor = generateCursor({ id: 'id-1', createdAt: '2024-01-01' }, 'createdAt'); - expect(validateCursor(cursor)).toBe(true); - }); - - it('returns false for a random string', () => { - expect(validateCursor('garbage')).toBe(false); - }); - - it('returns false for empty string', () => { - expect(validateCursor('')).toBe(false); - }); + it('returns true for a cursor produced by generateCursor', () => { + const cursor = generateCursor({ id: 'id-1', createdAt: '2024-01-01' }, 'createdAt'); + expect(validateCursor(cursor)).toBe(true); + }); + it('returns false for a random string', () => { + expect(validateCursor('garbage')).toBe(false); + }); + it('returns false for empty string', () => { + expect(validateCursor('')).toBe(false); + }); }); - // --------------------------------------------------------------------------- // paginateWithCursor — forward pagination // --------------------------------------------------------------------------- - describe('paginateWithCursor — forward (default)', () => { - it('returns all items and no cursors when fewer items than limit', async () => { - const items = makeItems(3); - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { limit: 10 }); - - expect(result.data).toHaveLength(3); - expect(result.meta.hasNextPage).toBe(false); - expect(result.meta.hasPrevPage).toBe(false); - expect(result.meta.nextCursor).toBeNull(); - expect(result.meta.prevCursor).toBeNull(); - expect(result.meta.limit).toBe(10); - }); - - it('returns nextCursor and hasNextPage=true when limit+1 items returned', async () => { - const items = makeItems(11); // limit=10 → 11th item signals more - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { limit: 10 }); - - expect(result.data).toHaveLength(10); - expect(result.meta.hasNextPage).toBe(true); - expect(result.meta.nextCursor).not.toBeNull(); - expect(result.meta.prevCursor).toBeNull(); // no cursor provided → first page - }); - - it('sets hasPrevPage=true and prevCursor when cursor is provided', async () => { - const items = makeItems(5); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { cursor, limit: 10 }); - - expect(result.meta.hasPrevPage).toBe(true); - expect(result.meta.prevCursor).not.toBeNull(); - }); - - it('applies WHERE condition when cursor is provided', async () => { - const items = makeItems(3); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - await paginateWithCursor(qb, { cursor, limit: 10 }); - - expect(qb.andWhere).toHaveBeenCalledTimes(1); - const [whereStr] = qb.andWhere.mock.calls[0]; - expect(whereStr).toContain('entity.createdAt'); - expect(whereStr).toContain('entity.id'); - }); - - it('does NOT call andWhere when no cursor is provided', async () => { - const qb = buildMockQueryBuilder(makeItems(3)); - - await paginateWithCursor(qb, { limit: 10 }); - - expect(qb.andWhere).not.toHaveBeenCalled(); - }); - - it('uses DESC operator (<) for forward DESC pagination', async () => { - const items = makeItems(3); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - await paginateWithCursor(qb, { cursor, limit: 5, order: SortOrder.DESC }); - - const [whereStr] = qb.andWhere.mock.calls[0]; - expect(whereStr).toContain('<'); - }); - - it('uses ASC operator (>) for forward ASC pagination', async () => { - const items = makeItems(3); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - await paginateWithCursor(qb, { cursor, limit: 5, order: SortOrder.ASC }); - - const [whereStr] = qb.andWhere.mock.calls[0]; - expect(whereStr).toContain('>'); - }); - - it('applies orderBy with the specified sortBy field', async () => { - const qb = buildMockQueryBuilder(makeItems(2)); - - await paginateWithCursor(qb, { sortBy: 'title', order: SortOrder.ASC }); - - expect(qb.orderBy).toHaveBeenCalledWith('entity.title', SortOrder.ASC); - }); - - it('uses default limit of 10 when not specified', async () => { - const qb = buildMockQueryBuilder(makeItems(0)); - - const result = await paginateWithCursor(qb, {}); - - expect(result.meta.limit).toBe(10); - // take() called with limit+1 - expect(qb.take).toHaveBeenCalledWith(11); - }); + it('returns all items and no cursors when fewer items than limit', async () => { + const items = makeItems(3); + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { limit: 10 }); + expect(result.data).toHaveLength(3); + expect(result.meta.hasNextPage).toBe(false); + expect(result.meta.hasPrevPage).toBe(false); + expect(result.meta.nextCursor).toBeNull(); + expect(result.meta.prevCursor).toBeNull(); + expect(result.meta.limit).toBe(10); + }); + it('returns nextCursor and hasNextPage=true when limit+1 items returned', async () => { + const items = makeItems(11); // limit=10 → 11th item signals more + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { limit: 10 }); + expect(result.data).toHaveLength(10); + expect(result.meta.hasNextPage).toBe(true); + expect(result.meta.nextCursor).not.toBeNull(); + expect(result.meta.prevCursor).toBeNull(); // no cursor provided → first page + }); + it('sets hasPrevPage=true and prevCursor when cursor is provided', async () => { + const items = makeItems(5); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { cursor, limit: 10 }); + expect(result.meta.hasPrevPage).toBe(true); + expect(result.meta.prevCursor).not.toBeNull(); + }); + it('applies WHERE condition when cursor is provided', async () => { + const items = makeItems(3); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + await paginateWithCursor(qb, { cursor, limit: 10 }); + expect(qb.andWhere).toHaveBeenCalledTimes(1); + const [whereStr] = qb.andWhere.mock.calls[0]; + expect(whereStr).toContain('entity.createdAt'); + expect(whereStr).toContain('entity.id'); + }); + it('does NOT call andWhere when no cursor is provided', async () => { + const qb = buildMockQueryBuilder(makeItems(3)); + await paginateWithCursor(qb, { limit: 10 }); + expect(qb.andWhere).not.toHaveBeenCalled(); + }); + it('uses DESC operator (<) for forward DESC pagination', async () => { + const items = makeItems(3); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + await paginateWithCursor(qb, { cursor, limit: 5, order: SortOrder.DESC }); + const [whereStr] = qb.andWhere.mock.calls[0]; + expect(whereStr).toContain('<'); + }); + it('uses ASC operator (>) for forward ASC pagination', async () => { + const items = makeItems(3); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + await paginateWithCursor(qb, { cursor, limit: 5, order: SortOrder.ASC }); + const [whereStr] = qb.andWhere.mock.calls[0]; + expect(whereStr).toContain('>'); + }); + it('applies orderBy with the specified sortBy field', async () => { + const qb = buildMockQueryBuilder(makeItems(2)); + await paginateWithCursor(qb, { sortBy: 'title', order: SortOrder.ASC }); + expect(qb.orderBy).toHaveBeenCalledWith('entity.title', SortOrder.ASC); + }); + it('uses default limit of 10 when not specified', async () => { + const qb = buildMockQueryBuilder(makeItems(0)); + const result = await paginateWithCursor(qb, {}); + expect(result.meta.limit).toBe(10); + // take() called with limit+1 + expect(qb.take).toHaveBeenCalledWith(11); + }); }); - // --------------------------------------------------------------------------- // paginateWithCursor — backward pagination // --------------------------------------------------------------------------- - describe('paginateWithCursor — backward', () => { - it('reverses results to natural order', async () => { - // Mock returns items in reversed sort order (ASC after inversion) - const items = [ - { id: 'id-001', createdAt: '2024-01-01' }, - { id: 'id-002', createdAt: '2024-01-02' }, - { id: 'id-003', createdAt: '2024-01-03' }, - ]; - const cursor = generateCursor({ id: 'id-004', createdAt: '2024-01-04' }, 'createdAt'); - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { - cursor, - limit: 10, - direction: CursorDirection.BACKWARD, + it('reverses results to natural order', async () => { + // Mock returns items in reversed sort order (ASC after inversion) + const items = [ + { id: 'id-001', createdAt: '2024-01-01' }, + { id: 'id-002', createdAt: '2024-01-02' }, + { id: 'id-003', createdAt: '2024-01-03' }, + ]; + const cursor = generateCursor({ id: 'id-004', createdAt: '2024-01-04' }, 'createdAt'); + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { + cursor, + limit: 10, + direction: CursorDirection.BACKWARD, + }); + // Reversed: [id-003, id-002, id-001] + expect(result.data[0].id).toBe('id-003'); + expect(result.data[2].id).toBe('id-001'); }); - - // Reversed: [id-003, id-002, id-001] - expect(result.data[0].id).toBe('id-003'); - expect(result.data[2].id).toBe('id-001'); - }); - - it('sets hasNextPage=true when cursor is provided on backward navigation', async () => { - const items = makeItems(3); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { - cursor, - limit: 10, - direction: CursorDirection.BACKWARD, + it('sets hasNextPage=true when cursor is provided on backward navigation', async () => { + const items = makeItems(3); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { + cursor, + limit: 10, + direction: CursorDirection.BACKWARD, + }); + expect(result.meta.hasNextPage).toBe(true); + expect(result.meta.nextCursor).not.toBeNull(); }); - - expect(result.meta.hasNextPage).toBe(true); - expect(result.meta.nextCursor).not.toBeNull(); - }); - - it('sets hasPrevPage=true and prevCursor when limit+1 items returned', async () => { - const items = makeItems(11); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - const result = await paginateWithCursor(qb, { - cursor, - limit: 10, - direction: CursorDirection.BACKWARD, + it('sets hasPrevPage=true and prevCursor when limit+1 items returned', async () => { + const items = makeItems(11); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + const result = await paginateWithCursor(qb, { + cursor, + limit: 10, + direction: CursorDirection.BACKWARD, + }); + expect(result.data).toHaveLength(10); + expect(result.meta.hasPrevPage).toBe(true); + expect(result.meta.prevCursor).not.toBeNull(); }); - - expect(result.data).toHaveLength(10); - expect(result.meta.hasPrevPage).toBe(true); - expect(result.meta.prevCursor).not.toBeNull(); - }); - - it('inverts sort order for backward direction (DESC → ASC)', async () => { - const cursor = generateCursor({ id: 'id-x', createdAt: '2024-01-10' }, 'createdAt'); - const qb = buildMockQueryBuilder([]); - - await paginateWithCursor(qb, { - cursor, - limit: 5, - order: SortOrder.DESC, - direction: CursorDirection.BACKWARD, + it('inverts sort order for backward direction (DESC → ASC)', async () => { + const cursor = generateCursor({ id: 'id-x', createdAt: '2024-01-10' }, 'createdAt'); + const qb = buildMockQueryBuilder([]); + await paginateWithCursor(qb, { + cursor, + limit: 5, + order: SortOrder.DESC, + direction: CursorDirection.BACKWARD, + }); + expect(qb.orderBy).toHaveBeenCalledWith('entity.createdAt', SortOrder.ASC); }); - - expect(qb.orderBy).toHaveBeenCalledWith('entity.createdAt', SortOrder.ASC); - }); - - it('uses > operator for backward DESC pagination', async () => { - const items = makeItems(2); - const cursor = generateCursor(items[0], 'createdAt'); - const qb = buildMockQueryBuilder(items); - - await paginateWithCursor(qb, { - cursor, - limit: 5, - order: SortOrder.DESC, - direction: CursorDirection.BACKWARD, + it('uses > operator for backward DESC pagination', async () => { + const items = makeItems(2); + const cursor = generateCursor(items[0], 'createdAt'); + const qb = buildMockQueryBuilder(items); + await paginateWithCursor(qb, { + cursor, + limit: 5, + order: SortOrder.DESC, + direction: CursorDirection.BACKWARD, + }); + const [whereStr] = qb.andWhere.mock.calls[0]; + expect(whereStr).toContain('>'); }); - - const [whereStr] = qb.andWhere.mock.calls[0]; - expect(whereStr).toContain('>'); - }); }); - // --------------------------------------------------------------------------- // paginate (offset-based — regression) // --------------------------------------------------------------------------- - describe('paginate (offset-based)', () => { - it('returns correct metadata for a given page', async () => { - const allItems = makeItems(25); - const pageItems = allItems.slice(10, 20); - const qb = buildMockQueryBuilder(pageItems); - qb.getCount.mockResolvedValue(25); - - const result = await paginate(qb, { page: 2, limit: 10 }); - - expect(result.meta.totalItems).toBe(25); - expect(result.meta.currentPage).toBe(2); - expect(result.meta.totalPages).toBe(3); - expect(result.meta.itemsPerPage).toBe(10); - expect(result.meta.itemCount).toBe(pageItems.length); - }); - - it('applies skip and take correctly', async () => { - const qb = buildMockQueryBuilder([]); - qb.getCount.mockResolvedValue(0); - - await paginate(qb, { page: 3, limit: 5 }); - - expect(qb.skip).toHaveBeenCalledWith(10); - expect(qb.take).toHaveBeenCalledWith(5); - }); - - it('defaults to page 1 and limit 10', async () => { - const qb = buildMockQueryBuilder([]); - qb.getCount.mockResolvedValue(0); - - const result = await paginate(qb, {}); - - expect(result.meta.currentPage).toBe(1); - expect(result.meta.itemsPerPage).toBe(10); - }); + it('returns correct metadata for a given page', async () => { + const allItems = makeItems(25); + const pageItems = allItems.slice(10, 20); + const qb = buildMockQueryBuilder(pageItems); + qb.getCount.mockResolvedValue(25); + const result = await paginate(qb, { page: 2, limit: 10 }); + expect(result.meta.totalItems).toBe(25); + expect(result.meta.currentPage).toBe(2); + expect(result.meta.totalPages).toBe(3); + expect(result.meta.itemsPerPage).toBe(10); + expect(result.meta.itemCount).toBe(pageItems.length); + }); + it('applies skip and take correctly', async () => { + const qb = buildMockQueryBuilder([]); + qb.getCount.mockResolvedValue(0); + await paginate(qb, { page: 3, limit: 5 }); + expect(qb.skip).toHaveBeenCalledWith(10); + expect(qb.take).toHaveBeenCalledWith(5); + }); + it('defaults to page 1 and limit 10', async () => { + const qb = buildMockQueryBuilder([]); + qb.getCount.mockResolvedValue(0); + const result = await paginate(qb, {}); + expect(result.meta.currentPage).toBe(1); + expect(result.meta.itemsPerPage).toBe(10); + }); }); diff --git a/src/common/utils/pagination.util.ts b/src/common/utils/pagination.util.ts index 4599c312..5194cc28 100644 --- a/src/common/utils/pagination.util.ts +++ b/src/common/utils/pagination.util.ts @@ -1,110 +1,95 @@ import { BadRequestException } from '@nestjs/common'; import { SelectQueryBuilder } from 'typeorm'; -import { - PaginationQueryDto, - SortOrder, - CursorPaginationQueryDto, - CursorDirection, -} from '../dto/pagination.dto'; +import { PaginationQueryDto, SortOrder, CursorPaginationQueryDto, CursorDirection, } from '../dto/pagination.dto'; import { APP_CONSTANTS } from '../constants/app.constants'; - const { DEFAULT_PAGE_SIZE } = APP_CONSTANTS; - export interface PaginatedResponse { - data: T[]; - meta: { - totalItems: number; - itemCount: number; - itemsPerPage: number; - totalPages: number; - currentPage: number; - }; + data: T[]; + meta: { + totalItems: number; + itemCount: number; + itemsPerPage: number; + totalPages: number; + currentPage: number; + }; } - export interface CursorPaginatedResponse { - data: T[]; - meta: { - nextCursor: string | null; - prevCursor: string | null; - hasNextPage: boolean; - hasPrevPage: boolean; - limit: number; - }; -} - -export async function paginate( - queryBuilder: SelectQueryBuilder, - options: PaginationQueryDto, -): Promise> { - const page = options.page || 1; - const limit = options.limit || DEFAULT_PAGE_SIZE; - const skip = (page - 1) * limit; - - // Apply sorting - if (options.sortBy) { - const alias = queryBuilder.alias; - queryBuilder.orderBy(`${alias}.${options.sortBy}`, options.order); - } - - // Clone query to get count without pagination limits - const totalItems = await queryBuilder.getCount(); - - // Apply pagination - const data = await queryBuilder.skip(skip).take(limit).getMany(); - - const totalPages = Math.ceil(totalItems / limit); - - return { - data, + data: T[]; meta: { - totalItems, - itemCount: data.length, - itemsPerPage: limit, - totalPages, - currentPage: page, - }, - }; + nextCursor: string | null; + prevCursor: string | null; + hasNextPage: boolean; + hasPrevPage: boolean; + limit: number; + }; +} +export async function paginate(queryBuilder: SelectQueryBuilder, options: PaginationQueryDto): Promise> { + const page = options.page || 1; + const limit = options.limit || DEFAULT_PAGE_SIZE; + const skip = (page - 1) * limit; + // Apply sorting + if (options.sortBy) { + const alias = queryBuilder.alias; + queryBuilder.orderBy(`${alias}.${options.sortBy}`, options.order); + } + // Clone query to get count without pagination limits + const totalItems = await queryBuilder.getCount(); + // Apply pagination + const data = await queryBuilder.skip(skip).take(limit).getMany(); + const totalPages = Math.ceil(totalItems / limit); + return { + data, + meta: { + totalItems, + itemCount: data.length, + itemsPerPage: limit, + totalPages, + currentPage: page, + }, + }; } - /** * Encodes entity fields into a base64 opaque cursor string. * The cursor captures the sort field value and the entity id for stable pagination. */ -export function generateCursor(entity: Record, sortBy: string): string { - const cursorData = { id: entity.id, sortValue: entity[sortBy] }; - return Buffer.from(JSON.stringify(cursorData)).toString('base64'); +export function generateCursor(entity: Record, sortBy: string): string { + const cursorData = { id: entity.id, sortValue: entity[sortBy] }; + return Buffer.from(JSON.stringify(cursorData)).toString('base64'); } - /** * Decodes a cursor string back to its constituent fields. * Throws BadRequestException if the cursor is malformed or missing required fields. */ -export function decodeCursor(cursor: string): { id: string; sortValue: any } { - try { - const json = Buffer.from(cursor, 'base64').toString('utf8'); - const data = JSON.parse(json); - if (typeof data.id !== 'string' || data.sortValue === undefined) { - throw new BadRequestException('Invalid cursor structure'); +export function decodeCursor(cursor: string): { + id: string; + sortValue: unknown; +} { + try { + const json = Buffer.from(cursor, 'base64').toString('utf8'); + const data = JSON.parse(json); + if (typeof data.id !== 'string' || data.sortValue === undefined) { + throw new BadRequestException('Invalid cursor structure'); + } + return data; + } + catch (error) { + if (error instanceof BadRequestException) + throw error; + throw new BadRequestException('Invalid cursor value'); } - return data; - } catch (error) { - if (error instanceof BadRequestException) throw error; - throw new BadRequestException('Invalid cursor value'); - } } - /** * Returns true if the cursor can be decoded without errors, false otherwise. */ export function validateCursor(cursor: string): boolean { - try { - decodeCursor(cursor); - return true; - } catch { - return false; - } + try { + decodeCursor(cursor); + return true; + } + catch { + return false; + } } - /** * Cursor-based pagination for TypeORM query builders. * @@ -115,81 +100,62 @@ export function validateCursor(cursor: string): boolean { * The cursor encodes {id, sortValue} of a boundary item so pages stay stable even * as new records are inserted. */ -export async function paginateWithCursor>( - queryBuilder: SelectQueryBuilder, - options: CursorPaginationQueryDto, -): Promise> { - const limit = options.limit || DEFAULT_PAGE_SIZE; - const sortBy = options.sortBy || 'createdAt'; - const order = options.order || SortOrder.DESC; - const direction = options.direction || CursorDirection.FORWARD; - const alias = queryBuilder.alias; - - const isForward = direction === CursorDirection.FORWARD; - const isDesc = order === SortOrder.DESC; - - if (options.cursor) { - const { id: cursorId, sortValue } = decodeCursor(options.cursor); - - // Determine the comparison operator for WHERE clause: - // Forward + DESC → get items older than cursor → primary field uses '<' - // Forward + ASC → get items newer than cursor → primary field uses '>' - // Backward + DESC → get items newer than cursor → primary field uses '>' - // Backward + ASC → get items older than cursor → primary field uses '<' - const useGreaterThan = (isForward && !isDesc) || (!isForward && isDesc); - const op = useGreaterThan ? '>' : '<'; - - queryBuilder.andWhere( - `(${alias}.${sortBy} ${op} :sortValue` + - ` OR (${alias}.${sortBy} = :sortValue AND ${alias}.id ${op} :cursorId))`, - { sortValue, cursorId }, - ); - } - - // For backward pagination the sort is inverted so we retrieve the nearest items; - // results are reversed after fetch to restore natural reading order. - const effectiveOrder: SortOrder = isForward ? order : isDesc ? SortOrder.ASC : SortOrder.DESC; - const effectiveIdOrder: SortOrder = isForward - ? isDesc - ? SortOrder.DESC - : SortOrder.ASC - : isDesc - ? SortOrder.ASC - : SortOrder.DESC; - - queryBuilder - .orderBy(`${alias}.${sortBy}`, effectiveOrder) - .addOrderBy(`${alias}.id`, effectiveIdOrder); - - // Fetch one extra item to determine whether another page exists in this direction - const rawItems = await queryBuilder.take(limit + 1).getMany(); - const hasMore = rawItems.length > limit; - const pageData = rawItems.slice(0, limit); - - if (!isForward) { - pageData.reverse(); - } - - // nextCursor points to the last item on this page (used to paginate forward) - const nextCursor = - pageData.length > 0 && (isForward ? hasMore : !!options.cursor) - ? generateCursor(pageData[pageData.length - 1], sortBy) - : null; - - // prevCursor points to the first item on this page (used to paginate backward) - const prevCursor = - pageData.length > 0 && (!isForward ? hasMore : !!options.cursor) - ? generateCursor(pageData[0], sortBy) - : null; - - return { - data: pageData, - meta: { - nextCursor, - prevCursor, - hasNextPage: isForward ? hasMore : !!options.cursor, - hasPrevPage: !isForward ? hasMore : !!options.cursor, - limit, - }, - }; +export async function paginateWithCursor>(queryBuilder: SelectQueryBuilder, options: CursorPaginationQueryDto): Promise> { + const limit = options.limit || DEFAULT_PAGE_SIZE; + const sortBy = options.sortBy || 'createdAt'; + const order = options.order || SortOrder.DESC; + const direction = options.direction || CursorDirection.FORWARD; + const alias = queryBuilder.alias; + const isForward = direction === CursorDirection.FORWARD; + const isDesc = order === SortOrder.DESC; + if (options.cursor) { + const { id: cursorId, sortValue } = decodeCursor(options.cursor); + // Determine the comparison operator for WHERE clause: + // Forward + DESC → get items older than cursor → primary field uses '<' + // Forward + ASC → get items newer than cursor → primary field uses '>' + // Backward + DESC → get items newer than cursor → primary field uses '>' + // Backward + ASC → get items older than cursor → primary field uses '<' + const useGreaterThan = (isForward && !isDesc) || (!isForward && isDesc); + const op = useGreaterThan ? '>' : '<'; + queryBuilder.andWhere(`(${alias}.${sortBy} ${op} :sortValue` + + ` OR (${alias}.${sortBy} = :sortValue AND ${alias}.id ${op} :cursorId))`, { sortValue, cursorId }); + } + // For backward pagination the sort is inverted so we retrieve the nearest items; + // results are reversed after fetch to restore natural reading order. + const effectiveOrder: SortOrder = isForward ? order : isDesc ? SortOrder.ASC : SortOrder.DESC; + const effectiveIdOrder: SortOrder = isForward + ? isDesc + ? SortOrder.DESC + : SortOrder.ASC + : isDesc + ? SortOrder.ASC + : SortOrder.DESC; + queryBuilder + .orderBy(`${alias}.${sortBy}`, effectiveOrder) + .addOrderBy(`${alias}.id`, effectiveIdOrder); + // Fetch one extra item to determine whether another page exists in this direction + const rawItems = await queryBuilder.take(limit + 1).getMany(); + const hasMore = rawItems.length > limit; + const pageData = rawItems.slice(0, limit); + if (!isForward) { + pageData.reverse(); + } + // nextCursor points to the last item on this page (used to paginate forward) + const nextCursor = pageData.length > 0 && (isForward ? hasMore : !!options.cursor) + ? generateCursor(pageData[pageData.length - 1], sortBy) + : null; + // prevCursor points to the first item on this page (used to paginate backward) + const prevCursor = pageData.length > 0 && (!isForward ? hasMore : !!options.cursor) + ? generateCursor(pageData[0], sortBy) + : null; + return { + data: pageData, + meta: { + nextCursor, + prevCursor, + hasNextPage: isForward ? hasMore : !!options.cursor, + hasPrevPage: !isForward ? hasMore : !!options.cursor, + limit, + }, + }; } diff --git a/src/common/utils/pii-sanitizer.utils.ts b/src/common/utils/pii-sanitizer.utils.ts index 774f39a2..63f86de8 100644 --- a/src/common/utils/pii-sanitizer.utils.ts +++ b/src/common/utils/pii-sanitizer.utils.ts @@ -1,95 +1,88 @@ /** * Utility for sanitizing Personally Identifiable Information (PII) from logs. */ - /** * Masks an email address to protect PII. * Example: "john.doe@example.com" -> "j***e@example.com" */ export function sanitizeEmail(email: string): string { - if (!email || typeof email !== 'string') return email; - const parts = email.split('@'); - if (parts.length !== 2) return email; - - const [user, domain] = parts; - if (user.length <= 2) { - return `***@${domain}`; - } - - return `${user[0]}***${user[user.length - 1]}@${domain}`; + if (!email || typeof email !== 'string') + return email; + const parts = email.split('@'); + if (parts.length !== 2) + return email; + const [user, domain] = parts; + if (user.length <= 2) { + return `***@${domain}`; + } + return `${user[0]}***${user[user.length - 1]}@${domain}`; } - /** * Masks a name or generic string. * Example: "John" -> "J***" */ export function sanitizeName(name: string): string { - if (!name || typeof name !== 'string') return name; - if (name.length <= 1) return '***'; - return `${name[0]}***`; + if (!name || typeof name !== 'string') + return name; + if (name.length <= 1) + return '***'; + return `${name[0]}***`; } - /** * Recursively sanitizes PII from an object or string. * Replaces values of sensitive keys with masks. */ -export function sanitizePii(data: any): any { - if (data === null || data === undefined) { - return data; - } - - if (typeof data === 'string') { - // Basic email masking within strings - return data.replace(/[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g, (email) => - sanitizeEmail(email), - ); - } - - if (Array.isArray(data)) { - return data.map((item) => sanitizePii(item)); - } - - if (typeof data === 'object') { - const sanitized: any = {}; - const sensitiveKeys = [ - 'email', - 'useremail', - 'contactemail', - 'owneremail', - 'recipientemail', - 'firstname', - 'lastname', - 'fullname', - 'phone', - 'phonenumber', - 'password', - 'token', - 'secret', - 'auth', - 'authorization', - 'bearer', - 'apikey', - 'stripekey', - 'awskey', - 'accesskey', - 'privatekey', - ]; - - for (const [key, value] of Object.entries(data)) { - const lowerKey = key.toLowerCase(); - - if (sensitiveKeys.some((sk) => lowerKey.includes(sk))) { - if (typeof value === 'string' && value.includes('@')) { - sanitized[key] = sanitizeEmail(value); - } else { - sanitized[key] = '***'; +export function sanitizePii(data: unknown): unknown { + if (data === null || data === undefined) { + return data; + } + if (typeof data === 'string') { + // Basic email masking within strings + return data.replace(/[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g, (email) => sanitizeEmail(email)); + } + if (Array.isArray(data)) { + return data.map((item) => sanitizePii(item)); + } + if (typeof data === 'object') { + const sanitized: unknown = {}; + const sensitiveKeys = [ + 'email', + 'useremail', + 'contactemail', + 'owneremail', + 'recipientemail', + 'firstname', + 'lastname', + 'fullname', + 'phone', + 'phonenumber', + 'password', + 'token', + 'secret', + 'auth', + 'authorization', + 'bearer', + 'apikey', + 'stripekey', + 'awskey', + 'accesskey', + 'privatekey', + ]; + for (const [key, value] of Object.entries(data)) { + const lowerKey = key.toLowerCase(); + if (sensitiveKeys.some((sk) => lowerKey.includes(sk))) { + if (typeof value === 'string' && value.includes('@')) { + sanitized[key] = sanitizeEmail(value); + } + else { + sanitized[key] = '***'; + } + } + else { + sanitized[key] = sanitizePii(value); + } } - } else { - sanitized[key] = sanitizePii(value); - } + return sanitized; } - return sanitized; - } - - return data; + return data; } diff --git a/src/common/utils/sanitization.utils.spec.ts b/src/common/utils/sanitization.utils.spec.ts index b8512934..473baab4 100644 --- a/src/common/utils/sanitization.utils.spec.ts +++ b/src/common/utils/sanitization.utils.spec.ts @@ -1,32 +1,24 @@ import { sanitizeSqlLike, enforceWhitelistedValue } from './sanitization.utils'; - describe('sanitization.utils', () => { - describe('sanitizeSqlLike', () => { - it('trims whitespace and escapes %, _, and \\', () => { - const raw = " test%_\\' OR 1=1 -- "; - const escaped = sanitizeSqlLike(raw); - - expect(escaped).toBe("test\\%\\_\\\\' OR 1=1 --"); + describe('sanitizeSqlLike', () => { + it('trims whitespace and escapes %, _, and \\', () => { + const raw = " test%_\\' OR 1=1 -- "; + const escaped = sanitizeSqlLike(raw); + expect(escaped).toBe("test\\%\\_\\\\' OR 1=1 --"); + }); + it('normalizes control characters to space', () => { + const raw = 'foo\nbar\tbaz\rqux'; + const escaped = sanitizeSqlLike(raw); + expect(escaped).toBe('foo bar baz qux'); + }); }); - - it('normalizes control characters to space', () => { - const raw = 'foo\nbar\tbaz\rqux'; - const escaped = sanitizeSqlLike(raw); - - expect(escaped).toBe('foo bar baz qux'); + describe('enforceWhitelistedValue', () => { + it('returns value from allowlist', () => { + const value = enforceWhitelistedValue('active', ['active', 'inactive'] as const, 'status'); + expect(value).toBe('active'); + }); + it('throws if value is not allowlisted', () => { + expect(() => enforceWhitelistedValue('hacked' as unknown, ['active', 'inactive'] as const, 'status')).toThrow(/Invalid value for status/); + }); }); - }); - - describe('enforceWhitelistedValue', () => { - it('returns value from allowlist', () => { - const value = enforceWhitelistedValue('active', ['active', 'inactive'] as const, 'status'); - expect(value).toBe('active'); - }); - - it('throws if value is not allowlisted', () => { - expect(() => - enforceWhitelistedValue('hacked' as any, ['active', 'inactive'] as const, 'status'), - ).toThrow(/Invalid value for status/); - }); - }); }); diff --git a/src/common/utils/sanitization.utils.ts b/src/common/utils/sanitization.utils.ts index fa9381c6..c7ff9336 100644 --- a/src/common/utils/sanitization.utils.ts +++ b/src/common/utils/sanitization.utils.ts @@ -1,32 +1,20 @@ export function sanitizeSqlLike(input: string): string { - if (typeof input !== 'string') { - throw new TypeError('Expected a string for SQL LIKE sanitization'); - } - - const trimmed = input.trim(); - - // Prevent CR/LF/Tab injection and normalize whitespace - const normalized = trimmed.replace(/[\r\n\t]+/g, ' '); - - // Escape SQL wildcard and escape characters for LIKE operators. - // This makes sure user-supplied `%`, `_`, and `\\` are treated literally. - return normalized.replace(/[\\%_]/g, (char) => `\\${char}`); + if (typeof input !== 'string') { + throw new TypeError('Expected a string for SQL LIKE sanitization'); + } + const trimmed = input.trim(); + // Prevent CR/LF/Tab injection and normalize whitespace + const normalized = trimmed.replace(/[\r\n\t]+/g, ' '); + // Escape SQL wildcard and escape characters for LIKE operators. + // This makes sure user-supplied `%`, `_`, and `\\` are treated literally. + return normalized.replace(/[\\%_]/g, (char) => `\\${char}`); } - -export function enforceWhitelistedValue( - value: T | undefined, - allowlist: readonly T[], - fieldName: string, -): T | undefined { - if (value === undefined || value === null || value === '') { - return undefined; - } - - if (!allowlist.includes(value as T)) { - throw new Error( - `Invalid value for ${fieldName}: ${value}. Allowed values are ${allowlist.join(', ')}`, - ); - } - - return value as T; +export function enforceWhitelistedValue(value: T | undefined, allowlist: readonly T[], fieldName: string): T | undefined { + if (value === undefined || value === null || value === '') { + return undefined; + } + if (!allowlist.includes(value as T)) { + throw new Error(`Invalid value for ${fieldName}: ${value}. Allowed values are ${allowlist.join(', ')}`); + } + return value as T; } diff --git a/src/common/utils/user.utils.ts b/src/common/utils/user.utils.ts index e0955871..14bef029 100644 --- a/src/common/utils/user.utils.ts +++ b/src/common/utils/user.utils.ts @@ -1,11 +1,5 @@ -import { - NotFoundException, - UnauthorizedException, - BadRequestException, - ConflictException, -} from '@nestjs/common'; +import { NotFoundException, UnauthorizedException, BadRequestException, ConflictException, } from '@nestjs/common'; import { User } from '../../users/entities/user.entity'; - /** * Ensures a user exists, throwing a NotFoundException otherwise. * @param user The user object to check @@ -13,39 +7,33 @@ import { User } from '../../users/entities/user.entity'; * @returns The guaranteed non-null user object */ export function ensureUserExists(user: User | null | undefined, message = 'User not found'): User { - if (!user) { - throw new NotFoundException(message); - } - return user; + if (!user) { + throw new NotFoundException(message); + } + return user; } - /** * Ensures a user exists for authentication purposes, throwing an UnauthorizedException otherwise. * @param user The user object to check * @param message Optional custom error message * @returns The guaranteed non-null user object */ -export function ensureValidCredentials( - user: User | null | undefined, - message = 'Invalid credentials', -): User { - if (!user) { - throw new UnauthorizedException(message); - } - return user; +export function ensureValidCredentials(user: User | null | undefined, message = 'Invalid credentials'): User { + if (!user) { + throw new UnauthorizedException(message); + } + return user; } - /** * Ensures a user's account is active. * @param user The user object to check * @param message Optional custom error message */ export function ensureUserIsActive(user: User, message = 'Account is not active'): void { - if (user.status !== 'active') { - throw new UnauthorizedException(message); - } + if (user.status !== 'active') { + throw new UnauthorizedException(message); + } } - /** * Ensures a user has a valid and unexpired token for a specific field. * @param user The user object to check @@ -54,35 +42,24 @@ export function ensureUserIsActive(user: User, message = 'Account is not active' * @param message Optional custom error message * @returns The guaranteed non-null user object */ -export function ensureValidUserToken( - user: User | null | undefined, - tokenField: keyof User, - expiresField: keyof User, - message = 'Invalid or expired token', -): User { - if (!user || !user[tokenField] || !user[expiresField]) { - throw new BadRequestException(message); - } - - const expireDate = user[expiresField] as Date; - if (new Date() > expireDate) { - throw new BadRequestException(message); - } - - return user; +export function ensureValidUserToken(user: User | null | undefined, tokenField: keyof User, expiresField: keyof User, message = 'Invalid or expired token'): User { + if (!user || !user[tokenField] || !user[expiresField]) { + throw new BadRequestException(message); + } + const expireDate = user[expiresField] as Date; + if (new Date() > expireDate) { + throw new BadRequestException(message); + } + return user; } - /** * Ensures a user does not exist, throwing a ConflictException otherwise. * Useful for registration or email updates. * @param user The user object to check * @param message Optional custom error message */ -export function ensureUserDoesNotExist( - user: User | null | undefined, - message = 'User already exists', -): void { - if (user) { - throw new ConflictException(message); - } +export function ensureUserDoesNotExist(user: User | null | undefined, message = 'User already exists'): void { + if (user) { + throw new ConflictException(message); + } } diff --git a/src/common/utils/websocket.utils.ts b/src/common/utils/websocket.utils.ts index 24f8b2bb..dc54d6c5 100644 --- a/src/common/utils/websocket.utils.ts +++ b/src/common/utils/websocket.utils.ts @@ -1,108 +1,87 @@ import { Socket } from 'socket.io'; - type ConnectionMeta = { - userId: string; - lastSeen: number; - isAlive: boolean; + userId: string; + lastSeen: number; + isAlive: boolean; }; - class WebSocketManager { - private connections = new Map>(); // userId -> sockets - private meta = new Map(); // socketId -> meta - - private MAX_CONNECTIONS_PER_USER = 3; - private MAX_GLOBAL_CONNECTIONS = 5000; - private HEARTBEAT_INTERVAL = 30000; // 30s - private TIMEOUT = 60000; // 60s - - startHeartbeat(io: any) { - setInterval(() => { - io.sockets.sockets.forEach((socket: Socket) => { + private connections = new Map>(); // userId -> sockets + private meta = new Map(); // socketId -> meta + private MAX_CONNECTIONS_PER_USER = 3; + private MAX_GLOBAL_CONNECTIONS = 5000; + private HEARTBEAT_INTERVAL = 30000; // 30s + private TIMEOUT = 60000; // 60s + startHeartbeat(io: unknown) { + setInterval(() => { + io.sockets.sockets.forEach((socket: Socket) => { + const meta = this.meta.get(socket.id); + if (!meta) + return; + if (!meta.isAlive) { + socket.disconnect(true); + this.cleanupSocket(socket); + return; + } + meta.isAlive = false; + socket.emit('ping'); + }); + }, this.HEARTBEAT_INTERVAL); + } + handlePong(socket: Socket) { const meta = this.meta.get(socket.id); - - if (!meta) return; - - if (!meta.isAlive) { - socket.disconnect(true); - this.cleanupSocket(socket); - return; + if (meta) { + meta.isAlive = true; + meta.lastSeen = Date.now(); } - - meta.isAlive = false; - socket.emit('ping'); - }); - }, this.HEARTBEAT_INTERVAL); - } - - handlePong(socket: Socket) { - const meta = this.meta.get(socket.id); - if (meta) { - meta.isAlive = true; - meta.lastSeen = Date.now(); } - } - - registerConnection(userId: string, socket: Socket) { - if (!this.connections.has(userId)) { - this.connections.set(userId, new Set()); + registerConnection(userId: string, socket: Socket) { + if (!this.connections.has(userId)) { + this.connections.set(userId, new Set()); + } + const userConnections = this.connections.get(userId); + if (!userConnections) { + return false; + } + // enforce global connection limits + if (this.meta.size >= this.MAX_GLOBAL_CONNECTIONS) { + socket.emit('error', { message: 'Server is at maximum capacity' }); + socket.disconnect(true); + return false; + } + // enforce max connections + if (userConnections.size >= this.MAX_CONNECTIONS_PER_USER) { + const oldestSocket = [...userConnections][0]; + oldestSocket.disconnect(true); + this.cleanupSocket(oldestSocket); + } + userConnections.add(socket); + this.meta.set(socket.id, { + userId, + lastSeen: Date.now(), + isAlive: true, + }); + return true; } - - const userConnections = this.connections.get(userId); - if (!userConnections) { - return false; + cleanupSocket(socket: Socket) { + const meta = this.meta.get(socket.id); + if (!meta) + return; + const userConnections = this.connections.get(meta.userId); + userConnections?.delete(socket); + if (userConnections && userConnections.size === 0) { + this.connections.delete(meta.userId); + } + this.meta.delete(socket.id); } - - // enforce global connection limits - if (this.meta.size >= this.MAX_GLOBAL_CONNECTIONS) { - socket.emit('error', { message: 'Server is at maximum capacity' }); - socket.disconnect(true); - return false; + getActiveConnections(userId: string): number { + return this.connections.get(userId)?.size || 0; } - - // enforce max connections - if (userConnections.size >= this.MAX_CONNECTIONS_PER_USER) { - const oldestSocket = [...userConnections][0]; - oldestSocket.disconnect(true); - this.cleanupSocket(oldestSocket); + getTotalConnections(): number { + return this.meta.size; } - - userConnections.add(socket); - - this.meta.set(socket.id, { - userId, - lastSeen: Date.now(), - isAlive: true, - }); - return true; - } - - cleanupSocket(socket: Socket) { - const meta = this.meta.get(socket.id); - if (!meta) return; - - const userConnections = this.connections.get(meta.userId); - - userConnections?.delete(socket); - - if (userConnections && userConnections.size === 0) { - this.connections.delete(meta.userId); + forceCleanup() { + this.connections.clear(); + this.meta.clear(); } - - this.meta.delete(socket.id); - } - - getActiveConnections(userId: string): number { - return this.connections.get(userId)?.size || 0; - } - - getTotalConnections(): number { - return this.meta.size; - } - - forceCleanup() { - this.connections.clear(); - this.meta.clear(); - } } - export const wsManager = new WebSocketManager(); diff --git a/src/common/validators/is-strong-password.validator.ts b/src/common/validators/is-strong-password.validator.ts index 69f4227f..19d0faef 100644 --- a/src/common/validators/is-strong-password.validator.ts +++ b/src/common/validators/is-strong-password.validator.ts @@ -1,5 +1 @@ -export { - IsStrongPassword, - calculatePasswordStrength, - PasswordStrengthResult, -} from './password.validator'; +export { IsStrongPassword, calculatePasswordStrength, PasswordStrengthResult, } from './password.validator'; diff --git a/src/common/validators/password.validator.spec.ts b/src/common/validators/password.validator.spec.ts index 78c398fe..c4ce2af0 100644 --- a/src/common/validators/password.validator.spec.ts +++ b/src/common/validators/password.validator.spec.ts @@ -1,42 +1,34 @@ import { calculatePasswordStrength, PasswordConstraint } from './password.validator'; - describe('Password Validator', () => { - describe('calculatePasswordStrength', () => { - it('recognizes weak passwords', () => { - const result = calculatePasswordStrength('abc'); - expect(result.isValid).toBe(false); - expect(result.level).toBe('weak'); - expect(result.errors).toEqual( - expect.arrayContaining([ - 'Password must be at least 8 characters long', - 'Password must contain at least one uppercase letter', - 'Password must contain at least one number', - ]), - ); + describe('calculatePasswordStrength', () => { + it('recognizes weak passwords', () => { + const result = calculatePasswordStrength('abc'); + expect(result.isValid).toBe(false); + expect(result.level).toBe('weak'); + expect(result.errors).toEqual(expect.arrayContaining([ + 'Password must be at least 8 characters long', + 'Password must contain at least one uppercase letter', + 'Password must contain at least one number', + ])); + }); + it('recognizes strong passwords', () => { + const result = calculatePasswordStrength('StrongPass123!'); + expect(result.isValid).toBe(true); + expect(result.level).toBe('strong'); + expect(result.errors).toEqual([]); + }); }); - - it('recognizes strong passwords', () => { - const result = calculatePasswordStrength('StrongPass123!'); - expect(result.isValid).toBe(true); - expect(result.level).toBe('strong'); - expect(result.errors).toEqual([]); + describe('PasswordConstraint', () => { + const constraint = new PasswordConstraint(); + it('validates strong password as valid', () => { + expect(constraint.validate('StrongPass123!')).toBe(true); + }); + it('validates weak password as invalid', () => { + expect(constraint.validate('weak')).toBe(false); + }); + it('returns detailed message for weak password', () => { + const message = constraint.defaultMessage({ value: 'weak' } as unknown); + expect(message).toContain('Password must be at least 8 characters long'); + }); }); - }); - - describe('PasswordConstraint', () => { - const constraint = new PasswordConstraint(); - - it('validates strong password as valid', () => { - expect(constraint.validate('StrongPass123!')).toBe(true); - }); - - it('validates weak password as invalid', () => { - expect(constraint.validate('weak')).toBe(false); - }); - - it('returns detailed message for weak password', () => { - const message = constraint.defaultMessage({ value: 'weak' } as any); - expect(message).toContain('Password must be at least 8 characters long'); - }); - }); }); diff --git a/src/common/validators/password.validator.ts b/src/common/validators/password.validator.ts index e74d64e0..875a2e3b 100644 --- a/src/common/validators/password.validator.ts +++ b/src/common/validators/password.validator.ts @@ -1,86 +1,70 @@ -import { - registerDecorator, - ValidationOptions, - ValidatorConstraint, - ValidatorConstraintInterface, - ValidationArguments, -} from 'class-validator'; - +import { registerDecorator, ValidationOptions, ValidatorConstraint, ValidatorConstraintInterface, ValidationArguments, } from 'class-validator'; export interface PasswordStrengthResult { - isValid: boolean; - errors: string[]; - score: number; - level: 'weak' | 'medium' | 'strong'; + isValid: boolean; + errors: string[]; + score: number; + level: 'weak' | 'medium' | 'strong'; } - export const PASSWORD_REQUIREMENTS = { - minLength: 8, - uppercase: /[A-Z]/, - lowercase: /[a-z]/, - number: /\d/, - special: /[!@#$%^&*()_+\-=[\]{};':"\\|,.<>/?]/, + minLength: 8, + uppercase: /[A-Z]/, + lowercase: /[a-z]/, + number: /\d/, + special: /[!@#$%^&*()_+\-=[\]{};':"\\|,.<>/?]/, }; - export function calculatePasswordStrength(password: string): PasswordStrengthResult { - const errors: string[] = []; - - if (typeof password !== 'string') { - errors.push('Password must be a string'); - return { isValid: false, errors, score: 0, level: 'weak' }; - } - - if (password.length < PASSWORD_REQUIREMENTS.minLength) { - errors.push(`Password must be at least ${PASSWORD_REQUIREMENTS.minLength} characters long`); - } - if (!PASSWORD_REQUIREMENTS.uppercase.test(password)) { - errors.push('Password must contain at least one uppercase letter'); - } - if (!PASSWORD_REQUIREMENTS.lowercase.test(password)) { - errors.push('Password must contain at least one lowercase letter'); - } - if (!PASSWORD_REQUIREMENTS.number.test(password)) { - errors.push('Password must contain at least one number'); - } - if (!PASSWORD_REQUIREMENTS.special.test(password)) { - errors.push('Password must contain at least one special character'); - } - - const score = 5 - errors.length; - const level = score <= 2 ? 'weak' : score === 3 || score === 4 ? 'medium' : 'strong'; - - return { - isValid: errors.length === 0, - errors, - score: Math.max(0, score), - level, - }; + const errors: string[] = []; + if (typeof password !== 'string') { + errors.push('Password must be a string'); + return { isValid: false, errors, score: 0, level: 'weak' }; + } + if (password.length < PASSWORD_REQUIREMENTS.minLength) { + errors.push(`Password must be at least ${PASSWORD_REQUIREMENTS.minLength} characters long`); + } + if (!PASSWORD_REQUIREMENTS.uppercase.test(password)) { + errors.push('Password must contain at least one uppercase letter'); + } + if (!PASSWORD_REQUIREMENTS.lowercase.test(password)) { + errors.push('Password must contain at least one lowercase letter'); + } + if (!PASSWORD_REQUIREMENTS.number.test(password)) { + errors.push('Password must contain at least one number'); + } + if (!PASSWORD_REQUIREMENTS.special.test(password)) { + errors.push('Password must contain at least one special character'); + } + const score = 5 - errors.length; + const level = score <= 2 ? 'weak' : score === 3 || score === 4 ? 'medium' : 'strong'; + return { + isValid: errors.length === 0, + errors, + score: Math.max(0, score), + level, + }; } - @ValidatorConstraint({ name: 'password', async: false }) export class PasswordConstraint implements ValidatorConstraintInterface { - validate(password: string): boolean { - const result = calculatePasswordStrength(password); - return result.isValid; - } - - defaultMessage(args: ValidationArguments): string { - const password = args.value as string; - const result = calculatePasswordStrength(password); - if (result.errors.length === 0) { - return 'Password does not meet strength requirements'; + validate(password: string): boolean { + const result = calculatePasswordStrength(password); + return result.isValid; + } + defaultMessage(args: ValidationArguments): string { + const password = args.value as string; + const result = calculatePasswordStrength(password); + if (result.errors.length === 0) { + return 'Password does not meet strength requirements'; + } + return result.errors.join('; '); } - return result.errors.join('; '); - } } - export function IsStrongPassword(validationOptions?: ValidationOptions) { - return function (object: object, propertyName: string) { - registerDecorator({ - target: object.constructor, - propertyName, - options: validationOptions, - constraints: [], - validator: PasswordConstraint, - }); - }; + return function (object: object, propertyName: string) { + registerDecorator({ + target: object.constructor, + propertyName, + options: validationOptions, + constraints: [], + validator: PasswordConstraint, + }); + }; } diff --git a/src/config/cache.config.ts b/src/config/cache.config.ts index 7ccf1544..47b5f62b 100644 --- a/src/config/cache.config.ts +++ b/src/config/cache.config.ts @@ -1,26 +1,23 @@ import { redisStore } from 'cache-manager-redis-store'; - export const cacheConfig = { - isGlobal: true, - store: redisStore, - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379', 10), - ttl: parseInt(process.env.REDIS_TTL || '60', 10), + isGlobal: true, + store: redisStore, + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379', 10), + ttl: parseInt(process.env.REDIS_TTL || '60', 10), }; - export const sessionConfig = { - secret: process.env.SESSION_SECRET || 'teachlink-session-secret', - name: process.env.SESSION_COOKIE_NAME || 'teachlink.sid', - prefix: process.env.SESSION_PREFIX || 'sess:', - ttlSeconds: parseInt(process.env.SESSION_TTL_SECONDS || '604800', 10), - cookieMaxAgeMs: parseInt(process.env.SESSION_COOKIE_MAX_AGE_MS || '604800000', 10), - secureCookies: process.env.NODE_ENV === 'production', - stickySessionsRequired: (process.env.STICKY_SESSIONS_REQUIRED || 'true') === 'true', - trustProxy: (process.env.TRUST_PROXY || 'true') === 'true', + secret: process.env.SESSION_SECRET || 'teachlink-session-secret', + name: process.env.SESSION_COOKIE_NAME || 'teachlink.sid', + prefix: process.env.SESSION_PREFIX || 'sess:', + ttlSeconds: parseInt(process.env.SESSION_TTL_SECONDS || '604800', 10), + cookieMaxAgeMs: parseInt(process.env.SESSION_COOKIE_MAX_AGE_MS || '604800000', 10), + secureCookies: process.env.NODE_ENV === 'production', + stickySessionsRequired: (process.env.STICKY_SESSIONS_REQUIRED || 'true') === 'true', + trustProxy: (process.env.TRUST_PROXY || 'true') === 'true', }; - export const distributedLockConfig = { - ttlMs: parseInt(process.env.SESSION_LOCK_TTL_MS || '5000', 10), - maxRetries: parseInt(process.env.SESSION_LOCK_MAX_RETRIES || '5', 10), - retryDelayMs: parseInt(process.env.SESSION_LOCK_RETRY_DELAY_MS || '120', 10), + ttlMs: parseInt(process.env.SESSION_LOCK_TTL_MS || '5000', 10), + maxRetries: parseInt(process.env.SESSION_LOCK_MAX_RETRIES || '5', 10), + retryDelayMs: parseInt(process.env.SESSION_LOCK_RETRY_DELAY_MS || '120', 10), }; diff --git a/src/config/cors.config.ts b/src/config/cors.config.ts index 3db8edb0..b5b3e4cf 100644 --- a/src/config/cors.config.ts +++ b/src/config/cors.config.ts @@ -1,70 +1,59 @@ import { CorsOptions } from '@nestjs/common/interfaces/external/cors-options.interface'; - /** * CORS configuration for the TeachLink API. * This configuration restricts access to specific origins, methods, and headers * to enhance security, especially in production environments. */ export const corsConfig: CorsOptions = { - /** - * Defines the allowed origins for CORS requests. - * In production, this should be restricted to specific frontend domains. - * If CORS_ALLOWED_ORIGINS is not set, it defaults to common local development ports. - */ - origin: ( - origin: string | undefined, - callback: (err: Error | null, allow?: boolean) => void, - ): void => { - const rawOrigins = process.env.CORS_ALLOWED_ORIGINS; - - // Default allowed origins if none are configured - const defaultOrigins = ['http://localhost:3000', 'http://localhost:4000']; - - const allowedOrigins = rawOrigins - ? rawOrigins - .split(',') - .map((o) => o.trim()) - .filter((o) => o !== '') - : defaultOrigins; - - // Allow requests with no origin (like mobile apps or curl requests) - if (!origin) { - return callback(null, true); - } - - if (allowedOrigins.includes(origin) || allowedOrigins.includes('*')) { - callback(null, true); - } else { - callback(new Error('Not allowed by CORS')); - } - }, - - /** - * Restricts allowed HTTP methods. - */ - methods: ['GET', 'HEAD', 'PUT', 'PATCH', 'POST', 'DELETE', 'OPTIONS'], - - /** - * Restricts allowed headers. - * Includes common headers and custom correlation ID headers. - */ - allowedHeaders: [ - 'Content-Type', - 'Accept', - 'Authorization', - 'X-Requested-With', - 'X-HTTP-Method-Override', - 'x-request-id', - 'x-correlation-id', - ], - - /** - * Allows credentials (cookies, authorization headers) to be sent. - */ - credentials: true, - - /** - * Configures how long the results of a preflight request can be cached. - */ - maxAge: 3600, + /** + * Defines the allowed origins for CORS requests. + * In production, this should be restricted to specific frontend domains. + * If CORS_ALLOWED_ORIGINS is not set, it defaults to common local development ports. + */ + origin: (origin: string | undefined, callback: (err: Error | null, allow?: boolean) => void): void => { + const rawOrigins = process.env.CORS_ALLOWED_ORIGINS; + // Default allowed origins if none are configured + const defaultOrigins = ['http://localhost:3000', 'http://localhost:4000']; + const allowedOrigins = rawOrigins + ? rawOrigins + .split(',') + .map((o) => o.trim()) + .filter((o) => o !== '') + : defaultOrigins; + // Allow requests with no origin (like mobile apps or curl requests) + if (!origin) { + return callback(null, true); + } + if (allowedOrigins.includes(origin) || allowedOrigins.includes('*')) { + callback(null, true); + } + else { + callback(new Error('Not allowed by CORS')); + } + }, + /** + * Restricts allowed HTTP methods. + */ + methods: ['GET', 'HEAD', 'PUT', 'PATCH', 'POST', 'DELETE', 'OPTIONS'], + /** + * Restricts allowed headers. + * Includes common headers and custom correlation ID headers. + */ + allowedHeaders: [ + 'Content-Type', + 'Accept', + 'Authorization', + 'X-Requested-With', + 'X-HTTP-Method-Override', + 'x-request-id', + 'x-correlation-id', + ], + /** + * Allows credentials (cookies, authorization headers) to be sent. + */ + credentials: true, + /** + * Configures how long the results of a preflight request can be cached. + */ + maxAge: 3600, }; diff --git a/src/config/elasticsearch.config.ts b/src/config/elasticsearch.config.ts index ead56955..bf0655f8 100644 --- a/src/config/elasticsearch.config.ts +++ b/src/config/elasticsearch.config.ts @@ -1,34 +1,28 @@ import { ConfigService } from '@nestjs/config'; import { ElasticsearchModuleOptions } from '@nestjs/elasticsearch'; - -export const createElasticsearchConfig = ( - configService: ConfigService, -): ElasticsearchModuleOptions => { - const node = configService.get('ELASTICSEARCH_NODE') ?? 'http://localhost:9200'; - const username = configService.get('ELASTICSEARCH_USERNAME'); - const password = configService.get('ELASTICSEARCH_PASSWORD'); - const apiKey = configService.get('ELASTICSEARCH_API_KEY'); - const caFingerprint = configService.get('ELASTICSEARCH_CA_FINGERPRINT'); - const requestTimeout = configService.get('ELASTICSEARCH_REQUEST_TIMEOUT') ?? 30000; - const maxRetries = configService.get('ELASTICSEARCH_MAX_RETRIES') ?? 3; - - const options: ElasticsearchModuleOptions = { - node, - maxRetries, - requestTimeout, - sniffOnStart: false, - compression: true, - }; - - if (apiKey) { - options.auth = { apiKey }; - } else if (username && password) { - options.auth = { username, password }; - } - - if (caFingerprint) { - options.caFingerprint = caFingerprint; - } - - return options; +export const createElasticsearchConfig = (configService: ConfigService): ElasticsearchModuleOptions => { + const node = configService.get('ELASTICSEARCH_NODE') ?? 'http://localhost:9200'; + const username = configService.get('ELASTICSEARCH_USERNAME'); + const password = configService.get('ELASTICSEARCH_PASSWORD'); + const apiKey = configService.get('ELASTICSEARCH_API_KEY'); + const caFingerprint = configService.get('ELASTICSEARCH_CA_FINGERPRINT'); + const requestTimeout = configService.get('ELASTICSEARCH_REQUEST_TIMEOUT') ?? 30000; + const maxRetries = configService.get('ELASTICSEARCH_MAX_RETRIES') ?? 3; + const options: ElasticsearchModuleOptions = { + node, + maxRetries, + requestTimeout, + sniffOnStart: false, + compression: true, + }; + if (apiKey) { + options.auth = { apiKey }; + } + else if (username && password) { + options.auth = { username, password }; + } + if (caFingerprint) { + options.caFingerprint = caFingerprint; + } + return options; }; diff --git a/src/config/env.validation.ts b/src/config/env.validation.ts index 4cfac75f..c837dee1 100644 --- a/src/config/env.validation.ts +++ b/src/config/env.validation.ts @@ -1,137 +1,119 @@ import * as Joi from 'joi'; - export const envValidationSchema = Joi.object({ - // Node Environment - NODE_ENV: Joi.string().valid('development', 'production', 'test').default('development'), - PORT: Joi.number().default(3000), - - // Database Configuration - DATABASE_HOST: Joi.string().required(), - DATABASE_PORT: Joi.number().required(), - DATABASE_USER: Joi.string().required(), - DATABASE_PASSWORD: Joi.string().required(), - DATABASE_NAME: Joi.string().required(), - DATABASE_POOL_MAX: Joi.number().integer().min(1).default(30), - DATABASE_POOL_MIN: Joi.number().integer().min(0).default(5), - DATABASE_POOL_ACQUIRE_TIMEOUT_MS: Joi.number().integer().min(1000).default(10000), - DATABASE_POOL_IDLE_TIMEOUT_MS: Joi.number().integer().min(1000).default(30000), - - // Redis Configuration - REDIS_HOST: Joi.string().required(), - REDIS_PORT: Joi.number().required(), - - // JWT Configuration - JWT_SECRETS: Joi.string().optional(), - JWT_SECRET_CURRENT_VERSION: Joi.string().optional(), - JWT_SECRET: Joi.string() - .min(10) - .when('JWT_SECRETS', { is: Joi.exist(), then: Joi.optional(), otherwise: Joi.required() }), - JWT_EXPIRES_IN: Joi.string().default('15m'), - JWT_REFRESH_SECRET: Joi.string().min(10).required(), - JWT_REFRESH_EXPIRES_IN: Joi.string().default('7d'), - - // Encryption - ENCRYPTION_SECRET: Joi.string().min(32).required(), - - // Security Configuration - BCRYPT_ROUNDS: Joi.number().integer().min(4).max(15).default(10), - - // Stripe Configuration - STRIPE_SECRET_KEY: Joi.string().required(), - STRIPE_WEBHOOK_SECRET: Joi.string().required(), - STRIPE_HEALTH_URL: Joi.string().uri().optional(), - - // SMTP Email Configuration - SMTP_HOST: Joi.string().required(), - SMTP_PORT: Joi.number().required(), - SMTP_SECURE: Joi.boolean().default(false), - SMTP_USER: Joi.string().required(), - SMTP_PASS: Joi.string().required(), - EMAIL_FROM: Joi.string().email().required(), - EMAIL_FROM_NAME: Joi.string().default('TeachLink'), - - // AWS Configuration - AWS_ACCESS_KEY_ID: Joi.string().required(), - AWS_SECRET_ACCESS_KEY: Joi.string().required(), - AWS_REGION: Joi.string().default('us-east-1'), - AWS_S3_BUCKET: Joi.string().required(), - AWS_S3_BUCKET_NAME: Joi.string().optional(), - AWS_S3_BUCKET_SECONDARY: Joi.string().optional(), - AWS_KMS_KEY_ID: Joi.string().optional(), - AWS_CLOUDFRONT_DISTRIBUTION_ID: Joi.string().optional(), - AWS_HEALTH_URL: Joi.string().uri().optional(), - - // SendGrid Configuration - SENDGRID_API_KEY: Joi.string().required(), - SENDGRID_HEALTH_URL: Joi.string().uri().optional(), - - // Elasticsearch Configuration - ELASTICSEARCH_NODE: Joi.string().uri().default('http://localhost:9200'), - ELASTICSEARCH_USERNAME: Joi.string().optional(), - ELASTICSEARCH_PASSWORD: Joi.string().optional(), - ELASTICSEARCH_API_KEY: Joi.string().optional(), - ELASTICSEARCH_CA_FINGERPRINT: Joi.string().optional(), - ELASTICSEARCH_REQUEST_TIMEOUT: Joi.number().integer().default(30000), - ELASTICSEARCH_MAX_RETRIES: Joi.number().integer().default(3), - - // Rate Limiting - THROTTLE_TTL: Joi.number().default(60), - THROTTLE_LIMIT: Joi.number().default(10), - - // Session Configuration - SESSION_SECRET: Joi.string().min(10).required(), - SESSION_COOKIE_NAME: Joi.string().default('teachlink.sid'), - SESSION_PREFIX: Joi.string().default('sess:'), - SESSION_TTL_SECONDS: Joi.number().integer().default(604800), - SESSION_COOKIE_MAX_AGE_MS: Joi.number().integer().default(604800000), - SESSION_LOCK_TTL_MS: Joi.number().integer().default(5000), - SESSION_LOCK_MAX_RETRIES: Joi.number().integer().default(5), - SESSION_LOCK_RETRY_DELAY_MS: Joi.number().integer().default(120), - STICKY_SESSIONS_REQUIRED: Joi.boolean().default(true), - TRUST_PROXY: Joi.boolean().default(true), - - // Feature Flags - ENABLE_AUTH: Joi.boolean().default(true), - ENABLE_SESSION_MANAGEMENT: Joi.boolean().default(true), - ENABLE_PAYMENTS: Joi.boolean().default(true), - ENABLE_AB_TESTING: Joi.boolean().default(false), - ENABLE_DATA_WAREHOUSE: Joi.boolean().default(false), - ENABLE_COLLABORATION: Joi.boolean().default(true), - ENABLE_MEDIA_PROCESSING: Joi.boolean().default(true), - ENABLE_BACKUP: Joi.boolean().default(true), - ENABLE_GRAPHQL: Joi.boolean().default(false), - ENABLE_SYNC: Joi.boolean().default(true), - ENABLE_MIGRATIONS: Joi.boolean().default(true), - ENABLE_RATE_LIMITING: Joi.boolean().default(true), - ENABLE_OBSERVABILITY: Joi.boolean().default(true), - ENABLE_CACHING: Joi.boolean().default(true), - ENABLE_FEATURE_FLAGS: Joi.boolean().default(true), - ENABLE_SEARCH: Joi.boolean().default(true), - ENABLE_NOTIFICATIONS: Joi.boolean().default(true), - ENABLE_EMAIL_MARKETING: Joi.boolean().default(true), - ENABLE_GAMIFICATION: Joi.boolean().default(true), - ENABLE_ASSESSMENT: Joi.boolean().default(true), - ENABLE_LEARNING_PATHS: Joi.boolean().default(true), - ENABLE_MODERATION: Joi.boolean().default(true), - ENABLE_ORCHESTRATION: Joi.boolean().default(true), - ENABLE_SECURITY: Joi.boolean().default(true), - ENABLE_TENANCY: Joi.boolean().default(true), - ENABLE_CDN: Joi.boolean().default(true), - ENABLE_LOCALIZATION: Joi.boolean().default(true), - ENABLE_MALWARE_SCANNING: Joi.boolean().default(false), - - // i18n / localization - I18N_DEFAULT_LOCALE: Joi.string().default('en'), - I18N_SUPPORTED_LOCALES: Joi.string().default('en'), - I18N_CACHE_TTL_SECONDS: Joi.number().integer().min(0).default(300), - - // Cluster Mode - CLUSTER_MODE: Joi.boolean().default(false), - CLUSTER_WORKERS: Joi.number().integer().min(1).default(4), - - // Application URL - APP_URL: Joi.string().uri().default('http://localhost:3000'), - - // CORS Configuration - CORS_ALLOWED_ORIGINS: Joi.string().default('http://localhost:3000,http://localhost:4000'), + // Node Environment + NODE_ENV: Joi.string().valid('development', 'production', 'test').default('development'), + PORT: Joi.number().default(3000), + // Database Configuration + DATABASE_HOST: Joi.string().required(), + DATABASE_PORT: Joi.number().required(), + DATABASE_USER: Joi.string().required(), + DATABASE_PASSWORD: Joi.string().required(), + DATABASE_NAME: Joi.string().required(), + DATABASE_POOL_MAX: Joi.number().integer().min(1).default(30), + DATABASE_POOL_MIN: Joi.number().integer().min(0).default(5), + DATABASE_POOL_ACQUIRE_TIMEOUT_MS: Joi.number().integer().min(1000).default(10000), + DATABASE_POOL_IDLE_TIMEOUT_MS: Joi.number().integer().min(1000).default(30000), + // Redis Configuration + REDIS_HOST: Joi.string().required(), + REDIS_PORT: Joi.number().required(), + // JWT Configuration + JWT_SECRETS: Joi.string().optional(), + JWT_SECRET_CURRENT_VERSION: Joi.string().optional(), + JWT_SECRET: Joi.string() + .min(10) + .when('JWT_SECRETS', { is: Joi.exist(), then: Joi.optional(), otherwise: Joi.required() }), + JWT_EXPIRES_IN: Joi.string().default('15m'), + JWT_REFRESH_SECRET: Joi.string().min(10).required(), + JWT_REFRESH_EXPIRES_IN: Joi.string().default('7d'), + // Encryption + ENCRYPTION_SECRET: Joi.string().min(32).required(), + // Security Configuration + BCRYPT_ROUNDS: Joi.number().integer().min(4).max(15).default(10), + // Stripe Configuration + STRIPE_SECRET_KEY: Joi.string().required(), + STRIPE_WEBHOOK_SECRET: Joi.string().required(), + STRIPE_HEALTH_URL: Joi.string().uri().optional(), + // SMTP Email Configuration + SMTP_HOST: Joi.string().required(), + SMTP_PORT: Joi.number().required(), + SMTP_SECURE: Joi.boolean().default(false), + SMTP_USER: Joi.string().required(), + SMTP_PASS: Joi.string().required(), + EMAIL_FROM: Joi.string().email().required(), + EMAIL_FROM_NAME: Joi.string().default('TeachLink'), + // AWS Configuration + AWS_ACCESS_KEY_ID: Joi.string().required(), + AWS_SECRET_ACCESS_KEY: Joi.string().required(), + AWS_REGION: Joi.string().default('us-east-1'), + AWS_S3_BUCKET: Joi.string().required(), + AWS_S3_BUCKET_NAME: Joi.string().optional(), + AWS_S3_BUCKET_SECONDARY: Joi.string().optional(), + AWS_KMS_KEY_ID: Joi.string().optional(), + AWS_CLOUDFRONT_DISTRIBUTION_ID: Joi.string().optional(), + AWS_HEALTH_URL: Joi.string().uri().optional(), + // SendGrid Configuration + SENDGRID_API_KEY: Joi.string().required(), + SENDGRID_HEALTH_URL: Joi.string().uri().optional(), + // Elasticsearch Configuration + ELASTICSEARCH_NODE: Joi.string().uri().default('http://localhost:9200'), + ELASTICSEARCH_USERNAME: Joi.string().optional(), + ELASTICSEARCH_PASSWORD: Joi.string().optional(), + ELASTICSEARCH_API_KEY: Joi.string().optional(), + ELASTICSEARCH_CA_FINGERPRINT: Joi.string().optional(), + ELASTICSEARCH_REQUEST_TIMEOUT: Joi.number().integer().default(30000), + ELASTICSEARCH_MAX_RETRIES: Joi.number().integer().default(3), + // Rate Limiting + THROTTLE_TTL: Joi.number().default(60), + THROTTLE_LIMIT: Joi.number().default(10), + // Session Configuration + SESSION_SECRET: Joi.string().min(10).required(), + SESSION_COOKIE_NAME: Joi.string().default('teachlink.sid'), + SESSION_PREFIX: Joi.string().default('sess:'), + SESSION_TTL_SECONDS: Joi.number().integer().default(604800), + SESSION_COOKIE_MAX_AGE_MS: Joi.number().integer().default(604800000), + SESSION_LOCK_TTL_MS: Joi.number().integer().default(5000), + SESSION_LOCK_MAX_RETRIES: Joi.number().integer().default(5), + SESSION_LOCK_RETRY_DELAY_MS: Joi.number().integer().default(120), + STICKY_SESSIONS_REQUIRED: Joi.boolean().default(true), + TRUST_PROXY: Joi.boolean().default(true), + // Feature Flags + ENABLE_AUTH: Joi.boolean().default(true), + ENABLE_SESSION_MANAGEMENT: Joi.boolean().default(true), + ENABLE_PAYMENTS: Joi.boolean().default(true), + ENABLE_AB_TESTING: Joi.boolean().default(false), + ENABLE_DATA_WAREHOUSE: Joi.boolean().default(false), + ENABLE_COLLABORATION: Joi.boolean().default(true), + ENABLE_MEDIA_PROCESSING: Joi.boolean().default(true), + ENABLE_BACKUP: Joi.boolean().default(true), + ENABLE_GRAPHQL: Joi.boolean().default(false), + ENABLE_SYNC: Joi.boolean().default(true), + ENABLE_MIGRATIONS: Joi.boolean().default(true), + ENABLE_RATE_LIMITING: Joi.boolean().default(true), + ENABLE_OBSERVABILITY: Joi.boolean().default(true), + ENABLE_CACHING: Joi.boolean().default(true), + ENABLE_FEATURE_FLAGS: Joi.boolean().default(true), + ENABLE_SEARCH: Joi.boolean().default(true), + ENABLE_NOTIFICATIONS: Joi.boolean().default(true), + ENABLE_EMAIL_MARKETING: Joi.boolean().default(true), + ENABLE_GAMIFICATION: Joi.boolean().default(true), + ENABLE_ASSESSMENT: Joi.boolean().default(true), + ENABLE_LEARNING_PATHS: Joi.boolean().default(true), + ENABLE_MODERATION: Joi.boolean().default(true), + ENABLE_ORCHESTRATION: Joi.boolean().default(true), + ENABLE_SECURITY: Joi.boolean().default(true), + ENABLE_TENANCY: Joi.boolean().default(true), + ENABLE_CDN: Joi.boolean().default(true), + ENABLE_LOCALIZATION: Joi.boolean().default(true), + ENABLE_MALWARE_SCANNING: Joi.boolean().default(false), + // i18n / localization + I18N_DEFAULT_LOCALE: Joi.string().default('en'), + I18N_SUPPORTED_LOCALES: Joi.string().default('en'), + I18N_CACHE_TTL_SECONDS: Joi.number().integer().min(0).default(300), + // Cluster Mode + CLUSTER_MODE: Joi.boolean().default(false), + CLUSTER_WORKERS: Joi.number().integer().min(1).default(4), + // Application URL + APP_URL: Joi.string().uri().default('http://localhost:3000'), + // CORS Configuration + CORS_ALLOWED_ORIGINS: Joi.string().default('http://localhost:3000,http://localhost:4000'), }); diff --git a/src/config/feature-flags.config.ts b/src/config/feature-flags.config.ts index 2eb1a410..cb350ec7 100644 --- a/src/config/feature-flags.config.ts +++ b/src/config/feature-flags.config.ts @@ -2,203 +2,179 @@ * Feature flags configuration for conditional module loading * Used to control which modules are loaded at application startup */ - export interface FeatureFlagsConfig { - // Core feature flags - ENABLE_AUTH: boolean; - ENABLE_SESSION_MANAGEMENT: boolean; - - // Optional feature modules - ENABLE_PAYMENTS: boolean; - ENABLE_AB_TESTING: boolean; - ENABLE_DATA_WAREHOUSE: boolean; - ENABLE_COLLABORATION: boolean; - ENABLE_MEDIA_PROCESSING: boolean; - ENABLE_BACKUP: boolean; - ENABLE_GRAPHQL: boolean; - ENABLE_SYNC: boolean; - ENABLE_MIGRATIONS: boolean; - ENABLE_RATE_LIMITING: boolean; - ENABLE_OBSERVABILITY: boolean; - ENABLE_CACHING: boolean; - ENABLE_FEATURE_FLAGS: boolean; - ENABLE_SEARCH: boolean; - ENABLE_NOTIFICATIONS: boolean; - ENABLE_EMAIL_MARKETING: boolean; - ENABLE_GAMIFICATION: boolean; - ENABLE_ASSESSMENT: boolean; - ENABLE_LEARNING_PATHS: boolean; - ENABLE_MODERATION: boolean; - ENABLE_ORCHESTRATION: boolean; - ENABLE_SECURITY: boolean; - ENABLE_TENANCY: boolean; - ENABLE_CDN: boolean; - ENABLE_LOCALIZATION: boolean; + // Core feature flags + ENABLE_AUTH: boolean; + ENABLE_SESSION_MANAGEMENT: boolean; + // Optional feature modules + ENABLE_PAYMENTS: boolean; + ENABLE_AB_TESTING: boolean; + ENABLE_DATA_WAREHOUSE: boolean; + ENABLE_COLLABORATION: boolean; + ENABLE_MEDIA_PROCESSING: boolean; + ENABLE_BACKUP: boolean; + ENABLE_GRAPHQL: boolean; + ENABLE_SYNC: boolean; + ENABLE_MIGRATIONS: boolean; + ENABLE_RATE_LIMITING: boolean; + ENABLE_OBSERVABILITY: boolean; + ENABLE_CACHING: boolean; + ENABLE_FEATURE_FLAGS: boolean; + ENABLE_SEARCH: boolean; + ENABLE_NOTIFICATIONS: boolean; + ENABLE_EMAIL_MARKETING: boolean; + ENABLE_GAMIFICATION: boolean; + ENABLE_ASSESSMENT: boolean; + ENABLE_LEARNING_PATHS: boolean; + ENABLE_MODERATION: boolean; + ENABLE_ORCHESTRATION: boolean; + ENABLE_SECURITY: boolean; + ENABLE_TENANCY: boolean; + ENABLE_CDN: boolean; + ENABLE_LOCALIZATION: boolean; } - /** * Default feature flags - all features enabled by default */ export const defaultFeatureFlags: FeatureFlagsConfig = { - ENABLE_AUTH: true, - ENABLE_SESSION_MANAGEMENT: true, - ENABLE_PAYMENTS: true, - ENABLE_AB_TESTING: false, - ENABLE_DATA_WAREHOUSE: false, - ENABLE_COLLABORATION: true, - ENABLE_MEDIA_PROCESSING: true, - ENABLE_BACKUP: true, - ENABLE_GRAPHQL: false, - ENABLE_SYNC: true, - ENABLE_MIGRATIONS: true, - ENABLE_RATE_LIMITING: true, - ENABLE_OBSERVABILITY: true, - ENABLE_CACHING: true, - ENABLE_FEATURE_FLAGS: true, - ENABLE_SEARCH: true, - ENABLE_NOTIFICATIONS: true, - ENABLE_EMAIL_MARKETING: true, - ENABLE_GAMIFICATION: true, - ENABLE_ASSESSMENT: true, - ENABLE_LEARNING_PATHS: true, - ENABLE_MODERATION: true, - ENABLE_ORCHESTRATION: true, - ENABLE_SECURITY: true, - ENABLE_TENANCY: true, - ENABLE_CDN: true, - ENABLE_LOCALIZATION: true, + ENABLE_AUTH: true, + ENABLE_SESSION_MANAGEMENT: true, + ENABLE_PAYMENTS: true, + ENABLE_AB_TESTING: false, + ENABLE_DATA_WAREHOUSE: false, + ENABLE_COLLABORATION: true, + ENABLE_MEDIA_PROCESSING: true, + ENABLE_BACKUP: true, + ENABLE_GRAPHQL: false, + ENABLE_SYNC: true, + ENABLE_MIGRATIONS: true, + ENABLE_RATE_LIMITING: true, + ENABLE_OBSERVABILITY: true, + ENABLE_CACHING: true, + ENABLE_FEATURE_FLAGS: true, + ENABLE_SEARCH: true, + ENABLE_NOTIFICATIONS: true, + ENABLE_EMAIL_MARKETING: true, + ENABLE_GAMIFICATION: true, + ENABLE_ASSESSMENT: true, + ENABLE_LEARNING_PATHS: true, + ENABLE_MODERATION: true, + ENABLE_ORCHESTRATION: true, + ENABLE_SECURITY: true, + ENABLE_TENANCY: true, + ENABLE_CDN: true, + ENABLE_LOCALIZATION: true, }; - /** * Load feature flags from environment variables */ export function loadFeatureFlags(): FeatureFlagsConfig { - return { - ENABLE_AUTH: getBooleanEnv('ENABLE_AUTH', defaultFeatureFlags.ENABLE_AUTH), - ENABLE_SESSION_MANAGEMENT: getBooleanEnv( - 'ENABLE_SESSION_MANAGEMENT', - defaultFeatureFlags.ENABLE_SESSION_MANAGEMENT, - ), - ENABLE_PAYMENTS: getBooleanEnv('ENABLE_PAYMENTS', defaultFeatureFlags.ENABLE_PAYMENTS), - ENABLE_AB_TESTING: getBooleanEnv('ENABLE_AB_TESTING', defaultFeatureFlags.ENABLE_AB_TESTING), - ENABLE_DATA_WAREHOUSE: getBooleanEnv( - 'ENABLE_DATA_WAREHOUSE', - defaultFeatureFlags.ENABLE_DATA_WAREHOUSE, - ), - ENABLE_COLLABORATION: getBooleanEnv( - 'ENABLE_COLLABORATION', - defaultFeatureFlags.ENABLE_COLLABORATION, - ), - ENABLE_MEDIA_PROCESSING: getBooleanEnv( - 'ENABLE_MEDIA_PROCESSING', - defaultFeatureFlags.ENABLE_MEDIA_PROCESSING, - ), - ENABLE_BACKUP: getBooleanEnv('ENABLE_BACKUP', defaultFeatureFlags.ENABLE_BACKUP), - ENABLE_GRAPHQL: getBooleanEnv('ENABLE_GRAPHQL', defaultFeatureFlags.ENABLE_GRAPHQL), - ENABLE_SYNC: getBooleanEnv('ENABLE_SYNC', defaultFeatureFlags.ENABLE_SYNC), - ENABLE_MIGRATIONS: getBooleanEnv('ENABLE_MIGRATIONS', defaultFeatureFlags.ENABLE_MIGRATIONS), - ENABLE_RATE_LIMITING: getBooleanEnv( - 'ENABLE_RATE_LIMITING', - defaultFeatureFlags.ENABLE_RATE_LIMITING, - ), - ENABLE_OBSERVABILITY: getBooleanEnv( - 'ENABLE_OBSERVABILITY', - defaultFeatureFlags.ENABLE_OBSERVABILITY, - ), - ENABLE_CACHING: getBooleanEnv('ENABLE_CACHING', defaultFeatureFlags.ENABLE_CACHING), - ENABLE_FEATURE_FLAGS: getBooleanEnv( - 'ENABLE_FEATURE_FLAGS', - defaultFeatureFlags.ENABLE_FEATURE_FLAGS, - ), - ENABLE_SEARCH: getBooleanEnv('ENABLE_SEARCH', defaultFeatureFlags.ENABLE_SEARCH), - ENABLE_NOTIFICATIONS: getBooleanEnv( - 'ENABLE_NOTIFICATIONS', - defaultFeatureFlags.ENABLE_NOTIFICATIONS, - ), - ENABLE_EMAIL_MARKETING: getBooleanEnv( - 'ENABLE_EMAIL_MARKETING', - defaultFeatureFlags.ENABLE_EMAIL_MARKETING, - ), - ENABLE_GAMIFICATION: getBooleanEnv( - 'ENABLE_GAMIFICATION', - defaultFeatureFlags.ENABLE_GAMIFICATION, - ), - ENABLE_ASSESSMENT: getBooleanEnv('ENABLE_ASSESSMENT', defaultFeatureFlags.ENABLE_ASSESSMENT), - ENABLE_LEARNING_PATHS: getBooleanEnv( - 'ENABLE_LEARNING_PATHS', - defaultFeatureFlags.ENABLE_LEARNING_PATHS, - ), - ENABLE_MODERATION: getBooleanEnv('ENABLE_MODERATION', defaultFeatureFlags.ENABLE_MODERATION), - ENABLE_ORCHESTRATION: getBooleanEnv( - 'ENABLE_ORCHESTRATION', - defaultFeatureFlags.ENABLE_ORCHESTRATION, - ), - ENABLE_SECURITY: getBooleanEnv('ENABLE_SECURITY', defaultFeatureFlags.ENABLE_SECURITY), - ENABLE_TENANCY: getBooleanEnv('ENABLE_TENANCY', defaultFeatureFlags.ENABLE_TENANCY), - ENABLE_CDN: getBooleanEnv('ENABLE_CDN', defaultFeatureFlags.ENABLE_CDN), - ENABLE_LOCALIZATION: getBooleanEnv( - 'ENABLE_LOCALIZATION', - defaultFeatureFlags.ENABLE_LOCALIZATION, - ), - }; + return { + ENABLE_AUTH: getBooleanEnv('ENABLE_AUTH', defaultFeatureFlags.ENABLE_AUTH), + ENABLE_SESSION_MANAGEMENT: getBooleanEnv('ENABLE_SESSION_MANAGEMENT', defaultFeatureFlags.ENABLE_SESSION_MANAGEMENT), + ENABLE_PAYMENTS: getBooleanEnv('ENABLE_PAYMENTS', defaultFeatureFlags.ENABLE_PAYMENTS), + ENABLE_AB_TESTING: getBooleanEnv('ENABLE_AB_TESTING', defaultFeatureFlags.ENABLE_AB_TESTING), + ENABLE_DATA_WAREHOUSE: getBooleanEnv('ENABLE_DATA_WAREHOUSE', defaultFeatureFlags.ENABLE_DATA_WAREHOUSE), + ENABLE_COLLABORATION: getBooleanEnv('ENABLE_COLLABORATION', defaultFeatureFlags.ENABLE_COLLABORATION), + ENABLE_MEDIA_PROCESSING: getBooleanEnv('ENABLE_MEDIA_PROCESSING', defaultFeatureFlags.ENABLE_MEDIA_PROCESSING), + ENABLE_BACKUP: getBooleanEnv('ENABLE_BACKUP', defaultFeatureFlags.ENABLE_BACKUP), + ENABLE_GRAPHQL: getBooleanEnv('ENABLE_GRAPHQL', defaultFeatureFlags.ENABLE_GRAPHQL), + ENABLE_SYNC: getBooleanEnv('ENABLE_SYNC', defaultFeatureFlags.ENABLE_SYNC), + ENABLE_MIGRATIONS: getBooleanEnv('ENABLE_MIGRATIONS', defaultFeatureFlags.ENABLE_MIGRATIONS), + ENABLE_RATE_LIMITING: getBooleanEnv('ENABLE_RATE_LIMITING', defaultFeatureFlags.ENABLE_RATE_LIMITING), + ENABLE_OBSERVABILITY: getBooleanEnv('ENABLE_OBSERVABILITY', defaultFeatureFlags.ENABLE_OBSERVABILITY), + ENABLE_CACHING: getBooleanEnv('ENABLE_CACHING', defaultFeatureFlags.ENABLE_CACHING), + ENABLE_FEATURE_FLAGS: getBooleanEnv('ENABLE_FEATURE_FLAGS', defaultFeatureFlags.ENABLE_FEATURE_FLAGS), + ENABLE_SEARCH: getBooleanEnv('ENABLE_SEARCH', defaultFeatureFlags.ENABLE_SEARCH), + ENABLE_NOTIFICATIONS: getBooleanEnv('ENABLE_NOTIFICATIONS', defaultFeatureFlags.ENABLE_NOTIFICATIONS), + ENABLE_EMAIL_MARKETING: getBooleanEnv('ENABLE_EMAIL_MARKETING', defaultFeatureFlags.ENABLE_EMAIL_MARKETING), + ENABLE_GAMIFICATION: getBooleanEnv('ENABLE_GAMIFICATION', defaultFeatureFlags.ENABLE_GAMIFICATION), + ENABLE_ASSESSMENT: getBooleanEnv('ENABLE_ASSESSMENT', defaultFeatureFlags.ENABLE_ASSESSMENT), + ENABLE_LEARNING_PATHS: getBooleanEnv('ENABLE_LEARNING_PATHS', defaultFeatureFlags.ENABLE_LEARNING_PATHS), + ENABLE_MODERATION: getBooleanEnv('ENABLE_MODERATION', defaultFeatureFlags.ENABLE_MODERATION), + ENABLE_ORCHESTRATION: getBooleanEnv('ENABLE_ORCHESTRATION', defaultFeatureFlags.ENABLE_ORCHESTRATION), + ENABLE_SECURITY: getBooleanEnv('ENABLE_SECURITY', defaultFeatureFlags.ENABLE_SECURITY), + ENABLE_TENANCY: getBooleanEnv('ENABLE_TENANCY', defaultFeatureFlags.ENABLE_TENANCY), + ENABLE_CDN: getBooleanEnv('ENABLE_CDN', defaultFeatureFlags.ENABLE_CDN), + ENABLE_LOCALIZATION: getBooleanEnv('ENABLE_LOCALIZATION', defaultFeatureFlags.ENABLE_LOCALIZATION), + }; } - /** * Helper function to parse boolean environment variables */ function getBooleanEnv(key: string, defaultValue: boolean): boolean { - const value = process.env[key]; - if (value === undefined) { - return defaultValue; - } - return value.toLowerCase() === 'true' || value === '1'; + const value = process.env[key]; + if (value === undefined) { + return defaultValue; + } + return value.toLowerCase() === 'true' || value === '1'; } - /** * Get list of enabled modules based on feature flags */ export function getEnabledModules(flags: FeatureFlagsConfig): string[] { - const modules: string[] = []; - - if (flags.ENABLE_AUTH) modules.push('AuthModule'); - if (flags.ENABLE_PAYMENTS) modules.push('PaymentsModule'); - if (flags.ENABLE_AB_TESTING) modules.push('ABTestingModule'); - if (flags.ENABLE_DATA_WAREHOUSE) modules.push('DataWarehouseModule'); - if (flags.ENABLE_COLLABORATION) modules.push('CollaborationModule'); - if (flags.ENABLE_MEDIA_PROCESSING) modules.push('MediaModule'); - if (flags.ENABLE_BACKUP) modules.push('BackupModule'); - if (flags.ENABLE_GRAPHQL) modules.push('GraphQLModule'); - if (flags.ENABLE_SYNC) modules.push('SyncModule'); - if (flags.ENABLE_MIGRATIONS) modules.push('MigrationModule'); - if (flags.ENABLE_RATE_LIMITING) modules.push('RateLimitingModule'); - if (flags.ENABLE_OBSERVABILITY) modules.push('ObservabilityModule'); - if (flags.ENABLE_CACHING) modules.push('CachingModule'); - if (flags.ENABLE_FEATURE_FLAGS) modules.push('FeatureFlagsModule'); - if (flags.ENABLE_SEARCH) modules.push('SearchModule'); - if (flags.ENABLE_NOTIFICATIONS) modules.push('NotificationsModule'); - if (flags.ENABLE_EMAIL_MARKETING) modules.push('EmailMarketingModule'); - if (flags.ENABLE_GAMIFICATION) modules.push('GamificationModule'); - if (flags.ENABLE_ASSESSMENT) modules.push('AssessmentModule'); - if (flags.ENABLE_LEARNING_PATHS) modules.push('LearningPathsModule'); - if (flags.ENABLE_MODERATION) modules.push('ModerationModule'); - if (flags.ENABLE_ORCHESTRATION) modules.push('OrchestrationModule'); - if (flags.ENABLE_SECURITY) modules.push('SecurityModule'); - if (flags.ENABLE_TENANCY) modules.push('TenancyModule'); - if (flags.ENABLE_CDN) modules.push('CDNModule'); - if (flags.ENABLE_LOCALIZATION) modules.push('LocalizationModule'); - - return modules; + const modules: string[] = []; + if (flags.ENABLE_AUTH) + modules.push('AuthModule'); + if (flags.ENABLE_PAYMENTS) + modules.push('PaymentsModule'); + if (flags.ENABLE_AB_TESTING) + modules.push('ABTestingModule'); + if (flags.ENABLE_DATA_WAREHOUSE) + modules.push('DataWarehouseModule'); + if (flags.ENABLE_COLLABORATION) + modules.push('CollaborationModule'); + if (flags.ENABLE_MEDIA_PROCESSING) + modules.push('MediaModule'); + if (flags.ENABLE_BACKUP) + modules.push('BackupModule'); + if (flags.ENABLE_GRAPHQL) + modules.push('GraphQLModule'); + if (flags.ENABLE_SYNC) + modules.push('SyncModule'); + if (flags.ENABLE_MIGRATIONS) + modules.push('MigrationModule'); + if (flags.ENABLE_RATE_LIMITING) + modules.push('RateLimitingModule'); + if (flags.ENABLE_OBSERVABILITY) + modules.push('ObservabilityModule'); + if (flags.ENABLE_CACHING) + modules.push('CachingModule'); + if (flags.ENABLE_FEATURE_FLAGS) + modules.push('FeatureFlagsModule'); + if (flags.ENABLE_SEARCH) + modules.push('SearchModule'); + if (flags.ENABLE_NOTIFICATIONS) + modules.push('NotificationsModule'); + if (flags.ENABLE_EMAIL_MARKETING) + modules.push('EmailMarketingModule'); + if (flags.ENABLE_GAMIFICATION) + modules.push('GamificationModule'); + if (flags.ENABLE_ASSESSMENT) + modules.push('AssessmentModule'); + if (flags.ENABLE_LEARNING_PATHS) + modules.push('LearningPathsModule'); + if (flags.ENABLE_MODERATION) + modules.push('ModerationModule'); + if (flags.ENABLE_ORCHESTRATION) + modules.push('OrchestrationModule'); + if (flags.ENABLE_SECURITY) + modules.push('SecurityModule'); + if (flags.ENABLE_TENANCY) + modules.push('TenancyModule'); + if (flags.ENABLE_CDN) + modules.push('CDNModule'); + if (flags.ENABLE_LOCALIZATION) + modules.push('LocalizationModule'); + return modules; } - /** * Get list of disabled modules based on feature flags */ export function getDisabledModules(flags: FeatureFlagsConfig): string[] { - const allModules = Object.keys(defaultFeatureFlags) - .filter((key) => key.startsWith('ENABLE_')) - .map((key) => `${key.replace('ENABLE_', '')}Module`); - - const enabledModules = getEnabledModules(flags); - - return allModules.filter((module) => !enabledModules.includes(module)); + const allModules = Object.keys(defaultFeatureFlags) + .filter((key) => key.startsWith('ENABLE_')) + .map((key) => `${key.replace('ENABLE_', '')}Module`); + const enabledModules = getEnabledModules(flags); + return allModules.filter((module) => !enabledModules.includes(module)); } diff --git a/src/config/swagger.config.ts b/src/config/swagger.config.ts index b74acdf5..992f99b2 100644 --- a/src/config/swagger.config.ts +++ b/src/config/swagger.config.ts @@ -1,40 +1,34 @@ import { INestApplication } from '@nestjs/common'; import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger'; import { API_VERSIONING_DOCUMENTATION } from '../common/modules/api-versioning.module'; - export function setupSwagger(app: INestApplication): void { - const config = new DocumentBuilder() - .setTitle('TeachLink API') - .setDescription(`TeachLink backend API documentation. ${API_VERSIONING_DOCUMENTATION}`) - .setVersion('1.0') - .addBearerAuth( - { + const config = new DocumentBuilder() + .setTitle('TeachLink API') + .setDescription(`TeachLink backend API documentation. ${API_VERSIONING_DOCUMENTATION}`) + .setVersion('1.0') + .addBearerAuth({ type: 'http', scheme: 'bearer', bearerFormat: 'JWT', name: 'Authorization', description: 'Enter JWT token', in: 'header', - }, - 'access-token', - ) - .addTag('Auth', 'Authentication and authorization endpoints') - .addTag('Users', 'User management endpoints') - .addTag('Courses', 'Course management endpoints') - .addTag('Payments', 'Payment processing endpoints') - .addTag('Subscriptions', 'Subscription management endpoints') - .addServer('http://localhost:3000', 'Local Development') - .addServer('https://api.teachlink.com', 'Production') - .build(); - - const document = SwaggerModule.createDocument(app, config); - - SwaggerModule.setup('api/docs', app, document, { - swaggerOptions: { - persistAuthorization: true, - tagsSorter: 'alpha', - operationsSorter: 'alpha', - }, - customSiteTitle: 'TeachLink API Docs', - }); + }, 'access-token') + .addTag('Auth', 'Authentication and authorization endpoints') + .addTag('Users', 'User management endpoints') + .addTag('Courses', 'Course management endpoints') + .addTag('Payments', 'Payment processing endpoints') + .addTag('Subscriptions', 'Subscription management endpoints') + .addServer('http://localhost:3000', 'Local Development') + .addServer('https://api.teachlink.com', 'Production') + .build(); + const document = SwaggerModule.createDocument(app, config); + SwaggerModule.setup('api/docs', app, document, { + swaggerOptions: { + persistAuthorization: true, + tagsSorter: 'alpha', + operationsSorter: 'alpha', + }, + customSiteTitle: 'TeachLink API Docs', + }); } diff --git a/src/courses/courses.controller.ts b/src/courses/courses.controller.ts index 3febebd7..6a6155f0 100644 --- a/src/courses/courses.controller.ts +++ b/src/courses/courses.controller.ts @@ -1,15 +1,4 @@ -import { - Controller, - Get, - Post, - Body, - Patch, - Param, - Delete, - Query, - UseGuards, - Request, -} from '@nestjs/common'; +import { Controller, Get, Post, Body, Patch, Param, Delete, Query, UseGuards, Request, } from '@nestjs/common'; import { CoursesService } from './courses.service'; import { CreateCourseDto } from './dto/create-course.dto'; import { UpdateCourseDto } from './dto/update-course.dto'; @@ -20,95 +9,105 @@ import { LessonsService } from './lessons/lessons.service'; import { CreateLessonDto } from './dto/create-lesson.dto'; import { EnrollmentsService } from './enrollments/enrollments.service'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; - @Controller('courses') export class CoursesController { - constructor( - private readonly coursesService: CoursesService, - private readonly modulesService: ModulesService, - private readonly lessonsService: LessonsService, - private readonly enrollmentsService: EnrollmentsService, - ) {} - - @Post() - @UseGuards(JwtAuthGuard) - create(@Request() req, @Body() createCourseDto: CreateCourseDto) { - // Get user from JWT token - const user = req.user; - if (!user || !user.id) { - throw new Error('User not authenticated'); + constructor(private readonly coursesService: CoursesService, private readonly modulesService: ModulesService, private readonly lessonsService: LessonsService, private readonly enrollmentsService: EnrollmentsService) { } + @Post() + @UseGuards(JwtAuthGuard) + create( + @Request() + req, + @Body() + createCourseDto: CreateCourseDto) { + // Get user from JWT token + const user = req.user; + if (!user || !user.id) { + throw new Error('User not authenticated'); + } + return this.coursesService.create({ + ...createCourseDto, + instructorId: user.id, + }); } - return this.coursesService.create({ - ...createCourseDto, - instructorId: user.id, - }); - } - - @Get() - findAll(@Query() searchDto: CourseSearchDto) { - return this.coursesService.findAll(searchDto); - } - - @Get('cursor') - findAllWithCursor(@Query() searchDto: CursorCourseSearchDto) { - return this.coursesService.findAllWithCursor(searchDto); - } - - @Get('analytics') - @UseGuards(JwtAuthGuard) - getAnalytics(@Request() _req) { - return this.coursesService.getAnalytics(); - } - - @Get(':id') - findOne(@Param('id') id: string) { - return this.coursesService.findOne(id); - } - - @Patch(':id') - @UseGuards(JwtAuthGuard) - update(@Param('id') id: string, @Body() updateCourseDto: UpdateCourseDto) { - return this.coursesService.update(id, updateCourseDto); - } - - @Delete(':id') - @UseGuards(JwtAuthGuard) - remove(@Param('id') id: string) { - return this.coursesService.remove(id); - } - - // Modules - @Post(':id/modules') - @UseGuards(JwtAuthGuard) - createModule( - @Request() req, - @Param('id') courseId: string, - @Body() createModuleDto: CreateModuleDto, - ) { - createModuleDto.courseId = courseId; - return this.modulesService.create(createModuleDto); - } - - // Lessons - @Post('modules/:moduleId/lessons') - @UseGuards(JwtAuthGuard) - createLesson( - @Request() req, - @Param('moduleId') moduleId: string, - @Body() createLessonDto: CreateLessonDto, - ) { - createLessonDto.moduleId = moduleId; - return this.lessonsService.create(createLessonDto); - } - - // Enrollments - @Post(':id/enroll') - @UseGuards(JwtAuthGuard) - enroll(@Request() req, @Param('id') courseId: string) { - const userId = req.user.id; - if (!userId) { - throw new Error('User not authenticated'); + @Get() + findAll( + @Query() + searchDto: CourseSearchDto) { + return this.coursesService.findAll(searchDto); + } + @Get('cursor') + findAllWithCursor( + @Query() + searchDto: CursorCourseSearchDto) { + return this.coursesService.findAllWithCursor(searchDto); + } + @Get('analytics') + @UseGuards(JwtAuthGuard) + getAnalytics( + @Request() + _req) { + return this.coursesService.getAnalytics(); + } + @Get(':id') + findOne( + @Param('id') + id: string) { + return this.coursesService.findOne(id); + } + @Patch(':id') + @UseGuards(JwtAuthGuard) + update( + @Param('id') + id: string, + @Body() + updateCourseDto: UpdateCourseDto) { + return this.coursesService.update(id, updateCourseDto); + } + @Delete(':id') + @UseGuards(JwtAuthGuard) + remove( + @Param('id') + id: string) { + return this.coursesService.remove(id); + } + // Modules + @Post(':id/modules') + @UseGuards(JwtAuthGuard) + createModule( + @Request() + req, + @Param('id') + courseId: string, + @Body() + createModuleDto: CreateModuleDto) { + createModuleDto.courseId = courseId; + return this.modulesService.create(createModuleDto); + } + // Lessons + @Post('modules/:moduleId/lessons') + @UseGuards(JwtAuthGuard) + createLesson( + @Request() + req, + @Param('moduleId') + moduleId: string, + @Body() + createLessonDto: CreateLessonDto) { + createLessonDto.moduleId = moduleId; + return this.lessonsService.create(createLessonDto); + } + // Enrollments + @Post(':id/enroll') + @UseGuards(JwtAuthGuard) + enroll( + @Request() + req, + @Param('id') + courseId: string) { + const userId = req.user.id; + if (!userId) { + throw new Error('User not authenticated'); + } + return this.enrollmentsService.enroll(userId, courseId); } - return this.enrollmentsService.enroll(userId, courseId); - } } diff --git a/src/courses/courses.module.ts b/src/courses/courses.module.ts index bc885fbe..b4848f06 100644 --- a/src/courses/courses.module.ts +++ b/src/courses/courses.module.ts @@ -10,11 +10,11 @@ import { CourseModule as CourseModuleEntity } from './entities/course-module.ent import { Lesson } from './entities/lesson.entity'; import { Enrollment } from './entities/enrollment.entity'; import { User } from '../users/entities/user.entity'; - @Module({ - imports: [TypeOrmModule.forFeature([Course, CourseModuleEntity, Lesson, Enrollment, User])], - controllers: [CoursesController], - providers: [CoursesService, ModulesService, LessonsService, EnrollmentsService], - exports: [CoursesService], + imports: [TypeOrmModule.forFeature([Course, CourseModuleEntity, Lesson, Enrollment, User])], + controllers: [CoursesController], + providers: [CoursesService, ModulesService, LessonsService, EnrollmentsService], + exports: [CoursesService], }) -export class CoursesModule {} +export class CoursesModule { +} diff --git a/src/courses/courses.service.ts b/src/courses/courses.service.ts index 9d47065d..4e41d6dc 100644 --- a/src/courses/courses.service.ts +++ b/src/courses/courses.service.ts @@ -3,12 +3,7 @@ import { InjectRepository } from '@nestjs/typeorm'; import { In, Repository } from 'typeorm'; import { Course } from './entities/course.entity'; import { UpdateCourseDto } from './dto/update-course.dto'; -import { - paginate, - paginateWithCursor, - PaginatedResponse, - CursorPaginatedResponse, -} from '../common/utils/pagination.util'; +import { paginate, paginateWithCursor, PaginatedResponse, CursorPaginatedResponse, } from '../common/utils/pagination.util'; import { CourseSearchDto, CursorCourseSearchDto } from './dto/course-search.dto'; import { CachingService } from '../caching/caching.service'; import { CacheInvalidationService } from '../caching/invalidation/invalidation.service'; @@ -17,211 +12,157 @@ import { EventEmitter2 } from '@nestjs/event-emitter'; import { sanitizeSqlLike, enforceWhitelistedValue } from '../common/utils/sanitization.utils'; import { CourseModule } from './entities/course-module.entity'; import { Lesson } from './entities/lesson.entity'; - @Injectable() export class CoursesService { - constructor( + constructor( @InjectRepository(Course) - private coursesRepository: Repository, - private readonly cachingService: CachingService, - private readonly invalidationService: CacheInvalidationService, - private readonly eventEmitter: EventEmitter2, - ) {} - - async create(createCourseDto: any): Promise { - const course = this.coursesRepository.create({ - ...createCourseDto, - instructor: { id: createCourseDto.instructorId }, - }); - const saved = await this.coursesRepository.save(course); - return Array.isArray(saved) ? saved[0] : saved; - } - - async findAll(filter?: CourseSearchDto): Promise> { - const cacheKey = `${CACHE_PREFIXES.COURSES_LIST}:${JSON.stringify(filter || {})}`; - - return this.cachingService.getOrSet( - cacheKey, - async () => { - const query = this.coursesRepository.createQueryBuilder('course'); - - query.leftJoinAndSelect('course.instructor', 'instructor'); - - if (filter?.search) { - const safeSearch = sanitizeSqlLike(filter.search); - query.andWhere( - "(course.title ILIKE :search ESCAPE '\\' OR course.description ILIKE :search ESCAPE '\\')", - { search: `%${safeSearch}%` }, - ); - } - - if (filter?.status) { - const allowedStatuses = ['draft', 'published', 'archived'] as const; - const status = enforceWhitelistedValue(filter.status, allowedStatuses, 'status'); - query.andWhere('course.status = :status', { status }); - } - - if (filter?.instructorId) { - query.andWhere('course.instructorId = :instructorId', { - instructorId: filter.instructorId, - }); - } - - query.orderBy('course.createdAt', 'DESC'); - - return await paginate(query, filter); - }, - CACHE_TTL.COURSE_METADATA, - ); - } - - async findAllWithCursor( - filter?: CursorCourseSearchDto, - ): Promise> { - const cacheKey = `${CACHE_PREFIXES.COURSES_LIST}:cursor:${JSON.stringify(filter || {})}`; - - return this.cachingService.getOrSet( - cacheKey, - async () => { - const query = this.coursesRepository.createQueryBuilder('course'); - - query.leftJoinAndSelect('course.instructor', 'instructor'); - - if (filter?.search) { - query.andWhere('(course.title ILIKE :search OR course.description ILIKE :search)', { - search: `%${filter.search}%`, - }); - } - - if (filter?.status) { - query.andWhere('course.status = :status', { status: filter.status }); - } - - if (filter?.instructorId) { - query.andWhere('course.instructorId = :instructorId', { - instructorId: filter.instructorId, - }); - } - - if (filter?.minPrice !== undefined) { - query.andWhere('course.price >= :minPrice', { minPrice: filter.minPrice }); - } - - if (filter?.maxPrice !== undefined) { - query.andWhere('course.price <= :maxPrice', { maxPrice: filter.maxPrice }); + private coursesRepository: Repository, private readonly cachingService: CachingService, private readonly invalidationService: CacheInvalidationService, private readonly eventEmitter: EventEmitter2) { } + async create(createCourseDto: unknown): Promise { + const course = this.coursesRepository.create({ + ...createCourseDto, + instructor: { id: createCourseDto.instructorId }, + }); + const saved = await this.coursesRepository.save(course); + return Array.isArray(saved) ? saved[0] : saved; + } + async findAll(filter?: CourseSearchDto): Promise> { + const cacheKey = `${CACHE_PREFIXES.COURSES_LIST}:${JSON.stringify(filter || {})}`; + return this.cachingService.getOrSet(cacheKey, async () => { + const query = this.coursesRepository.createQueryBuilder('course'); + query.leftJoinAndSelect('course.instructor', 'instructor'); + if (filter?.search) { + const safeSearch = sanitizeSqlLike(filter.search); + query.andWhere("(course.title ILIKE :search ESCAPE '\\' OR course.description ILIKE :search ESCAPE '\\')", { search: `%${safeSearch}%` }); + } + if (filter?.status) { + const allowedStatuses = ['draft', 'published', 'archived'] as const; + const status = enforceWhitelistedValue(filter.status, allowedStatuses, 'status'); + query.andWhere('course.status = :status', { status }); + } + if (filter?.instructorId) { + query.andWhere('course.instructorId = :instructorId', { + instructorId: filter.instructorId, + }); + } + query.orderBy('course.createdAt', 'DESC'); + return await paginate(query, filter); + }, CACHE_TTL.COURSE_METADATA); + } + async findAllWithCursor(filter?: CursorCourseSearchDto): Promise> { + const cacheKey = `${CACHE_PREFIXES.COURSES_LIST}:cursor:${JSON.stringify(filter || {})}`; + return this.cachingService.getOrSet(cacheKey, async () => { + const query = this.coursesRepository.createQueryBuilder('course'); + query.leftJoinAndSelect('course.instructor', 'instructor'); + if (filter?.search) { + query.andWhere('(course.title ILIKE :search OR course.description ILIKE :search)', { + search: `%${filter.search}%`, + }); + } + if (filter?.status) { + query.andWhere('course.status = :status', { status: filter.status }); + } + if (filter?.instructorId) { + query.andWhere('course.instructorId = :instructorId', { + instructorId: filter.instructorId, + }); + } + if (filter?.minPrice !== undefined) { + query.andWhere('course.price >= :minPrice', { minPrice: filter.minPrice }); + } + if (filter?.maxPrice !== undefined) { + query.andWhere('course.price <= :maxPrice', { maxPrice: filter.maxPrice }); + } + return await paginateWithCursor(query, filter ?? {}); + }, CACHE_TTL.COURSE_METADATA); + } + async findByIds(ids: string[]): Promise { + if (ids.length === 0) + return []; + return await this.coursesRepository.findByIds(ids); + } + async findByInstructor(instructorId: string): Promise { + return await this.coursesRepository.find({ + where: { instructor: { id: instructorId } }, + relations: ['instructor'], + }); + } + async findByInstructorIds(instructorIds: string[]): Promise { + if (instructorIds.length === 0) + return []; + return await this.coursesRepository + .createQueryBuilder('course') + .leftJoinAndSelect('course.instructor', 'instructor') + .where('instructor.id IN (:...instructorIds)', { instructorIds }) + .getMany(); + } + async findOne(id: string): Promise { + const cacheKey = `${CACHE_PREFIXES.COURSE}:${id}`; + return this.cachingService.getOrSet(cacheKey, async () => { + const course = await this.coursesRepository.findOne({ + where: { id }, + relations: ['instructor', 'modules', 'modules.lessons'], + order: { + modules: { + order: 'ASC', + lessons: { + order: 'ASC', + }, + }, + } as unknown, + }); + if (!course) { + throw new NotFoundException(`Course with ID ${id} not found`); + } + return course; + }, CACHE_TTL.COURSE_DETAILS); + } + async update(id: string, updateCourseDto: UpdateCourseDto): Promise { + const course = await this.coursesRepository.findOne({ + where: { id }, + relations: ['instructor', 'modules', 'modules.lessons'], + }); + if (!course) { + throw new NotFoundException(`Course with ID ${id} not found`); } - - return await paginateWithCursor(query, filter ?? {}); - }, - CACHE_TTL.COURSE_METADATA, - ); - } - - async findByIds(ids: string[]): Promise { - if (ids.length === 0) return []; - return await this.coursesRepository.findByIds(ids); - } - - async findByInstructor(instructorId: string): Promise { - return await this.coursesRepository.find({ - where: { instructor: { id: instructorId } }, - relations: ['instructor'], - }); - } - - async findByInstructorIds(instructorIds: string[]): Promise { - if (instructorIds.length === 0) return []; - return await this.coursesRepository - .createQueryBuilder('course') - .leftJoinAndSelect('course.instructor', 'instructor') - .where('instructor.id IN (:...instructorIds)', { instructorIds }) - .getMany(); - } - - async findOne(id: string): Promise { - const cacheKey = `${CACHE_PREFIXES.COURSE}:${id}`; - - return this.cachingService.getOrSet( - cacheKey, - async () => { + Object.assign(course, updateCourseDto); + const saved = await this.coursesRepository.save(course); + // Invalidate cache after update + this.eventEmitter.emit(CACHE_EVENTS.COURSE_UPDATED, { courseId: id }); + return saved; + } + async remove(id: string): Promise { const course = await this.coursesRepository.findOne({ - where: { id }, - relations: ['instructor', 'modules', 'modules.lessons'], - order: { - modules: { - order: 'ASC', - lessons: { - order: 'ASC', - }, - }, - } as any, + where: { id }, + relations: ['modules'], }); if (!course) { - throw new NotFoundException(`Course with ID ${id} not found`); + throw new NotFoundException(`Course with ID ${id} not found`); } - return course; - }, - CACHE_TTL.COURSE_DETAILS, - ); - } - - async update(id: string, updateCourseDto: UpdateCourseDto): Promise { - const course = await this.coursesRepository.findOne({ - where: { id }, - relations: ['instructor', 'modules', 'modules.lessons'], - }); - if (!course) { - throw new NotFoundException(`Course with ID ${id} not found`); + await this.coursesRepository.manager.transaction(async (manager) => { + const moduleIds = course.modules.map((module) => module.id); + if (moduleIds.length > 0) { + await manager.getRepository(Lesson).softDelete({ moduleId: In(moduleIds) }); + } + await manager.getRepository(CourseModule).softDelete({ courseId: id }); + await manager.getRepository(Course).softDelete(id); + }); + // Invalidate cache after delete + this.eventEmitter.emit(CACHE_EVENTS.COURSE_DELETED, { courseId: id }); } - Object.assign(course, updateCourseDto); - const saved = await this.coursesRepository.save(course); - - // Invalidate cache after update - this.eventEmitter.emit(CACHE_EVENTS.COURSE_UPDATED, { courseId: id }); - - return saved; - } - - async remove(id: string): Promise { - const course = await this.coursesRepository.findOne({ - where: { id }, - relations: ['modules'], - }); - if (!course) { - throw new NotFoundException(`Course with ID ${id} not found`); + async getAnalytics(): Promise { + const totalCourses = await this.coursesRepository.count(); + const publishedCourses = await this.coursesRepository.count({ + where: { status: 'published' }, + }); + const { totalEnrollments } = await this.coursesRepository + .createQueryBuilder('course') + .leftJoin('course.enrollments', 'enrollment') + .select('COUNT(enrollment.id)', 'totalEnrollments') + .getRawOne(); + return { + totalCourses, + publishedCourses, + totalEnrollments: parseInt(totalEnrollments) || 0, + }; } - - await this.coursesRepository.manager.transaction(async (manager) => { - const moduleIds = course.modules.map((module) => module.id); - - if (moduleIds.length > 0) { - await manager.getRepository(Lesson).softDelete({ moduleId: In(moduleIds) }); - } - - await manager.getRepository(CourseModule).softDelete({ courseId: id }); - await manager.getRepository(Course).softDelete(id); - }); - - // Invalidate cache after delete - this.eventEmitter.emit(CACHE_EVENTS.COURSE_DELETED, { courseId: id }); - } - - async getAnalytics(): Promise { - const totalCourses = await this.coursesRepository.count(); - const publishedCourses = await this.coursesRepository.count({ - where: { status: 'published' }, - }); - - const { totalEnrollments } = await this.coursesRepository - .createQueryBuilder('course') - .leftJoin('course.enrollments', 'enrollment') - .select('COUNT(enrollment.id)', 'totalEnrollments') - .getRawOne(); - - return { - totalCourses, - publishedCourses, - totalEnrollments: parseInt(totalEnrollments) || 0, - }; - } } diff --git a/src/courses/dto/course-search.dto.ts b/src/courses/dto/course-search.dto.ts index 7d10d151..a2ecd458 100644 --- a/src/courses/dto/course-search.dto.ts +++ b/src/courses/dto/course-search.dto.ts @@ -1,45 +1,37 @@ import { IsString, IsOptional, IsNumber, IsUUID } from 'class-validator'; import { ApiPropertyOptional } from '@nestjs/swagger'; import { PaginationQueryDto, CursorPaginationQueryDto } from '../../common/dto/pagination.dto'; - export class CourseSearchDto extends PaginationQueryDto { - @ApiPropertyOptional() - @IsNumber() - @IsOptional() - minPrice?: number; - - @ApiPropertyOptional() - @IsNumber() - @IsOptional() - maxPrice?: number; - - @IsOptional() - @IsString() - status?: string; - - @IsOptional() - @IsString() - @IsUUID() - instructorId?: string; + @ApiPropertyOptional() + @IsNumber() + @IsOptional() + minPrice?: number; + @ApiPropertyOptional() + @IsNumber() + @IsOptional() + maxPrice?: number; + @IsOptional() + @IsString() + status?: string; + @IsOptional() + @IsString() + @IsUUID() + instructorId?: string; } - export class CursorCourseSearchDto extends CursorPaginationQueryDto { - @ApiPropertyOptional() - @IsNumber() - @IsOptional() - minPrice?: number; - - @ApiPropertyOptional() - @IsNumber() - @IsOptional() - maxPrice?: number; - - @IsOptional() - @IsString() - status?: string; - - @IsOptional() - @IsString() - @IsUUID() - instructorId?: string; + @ApiPropertyOptional() + @IsNumber() + @IsOptional() + minPrice?: number; + @ApiPropertyOptional() + @IsNumber() + @IsOptional() + maxPrice?: number; + @IsOptional() + @IsString() + status?: string; + @IsOptional() + @IsString() + @IsUUID() + instructorId?: string; } diff --git a/src/courses/dto/create-course.dto.ts b/src/courses/dto/create-course.dto.ts index 81d0100f..b76a5cb8 100644 --- a/src/courses/dto/create-course.dto.ts +++ b/src/courses/dto/create-course.dto.ts @@ -1,37 +1,25 @@ -import { - IsString, - IsNotEmpty, - IsNumber, - IsOptional, - MinLength, - MaxLength, - Min, -} from 'class-validator'; +import { IsString, IsNotEmpty, IsNumber, IsOptional, MinLength, MaxLength, Min, } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class CreateCourseDto { - @ApiProperty() - @IsString({ message: 'Title must be a string' }) - @IsNotEmpty({ message: 'Course title is required' }) - @MinLength(5, { message: 'Title must be at least 5 characters long' }) - @MaxLength(100, { message: 'Title cannot exceed 100 characters' }) - title: string; - - @ApiProperty() - @IsString({ message: 'Description must be a string' }) - @IsNotEmpty({ message: 'Course description is required' }) - @MinLength(20, { message: 'Description must be at least 20 characters long' }) - @MaxLength(5000, { message: 'Description is too long' }) - description: string; - - @ApiProperty({ required: false }) - @IsNumber({}, { message: 'Price must be a valid number' }) - @Min(0, { message: 'Price cannot be negative' }) - @IsOptional() - price?: number; - - @ApiProperty({ required: false }) - @IsString({ message: 'Thumbnail URL must be a valid string' }) - @IsOptional() - thumbnailUrl?: string; + @ApiProperty() + @IsString({ message: 'Title must be a string' }) + @IsNotEmpty({ message: 'Course title is required' }) + @MinLength(5, { message: 'Title must be at least 5 characters long' }) + @MaxLength(100, { message: 'Title cannot exceed 100 characters' }) + title: string; + @ApiProperty() + @IsString({ message: 'Description must be a string' }) + @IsNotEmpty({ message: 'Course description is required' }) + @MinLength(20, { message: 'Description must be at least 20 characters long' }) + @MaxLength(5000, { message: 'Description is too long' }) + description: string; + @ApiProperty({ required: false }) + @IsNumber({}, { message: 'Price must be a valid number' }) + @Min(0, { message: 'Price cannot be negative' }) + @IsOptional() + price?: number; + @ApiProperty({ required: false }) + @IsString({ message: 'Thumbnail URL must be a valid string' }) + @IsOptional() + thumbnailUrl?: string; } diff --git a/src/courses/dto/create-lesson.dto.ts b/src/courses/dto/create-lesson.dto.ts index faaf5aac..4bb61c33 100644 --- a/src/courses/dto/create-lesson.dto.ts +++ b/src/courses/dto/create-lesson.dto.ts @@ -1,38 +1,32 @@ import { IsString, IsNotEmpty, IsInt, IsOptional, IsUUID, IsUrl, IsNumber } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class CreateLessonDto { - @ApiProperty() - @IsString() - @IsNotEmpty() - title: string; - - @ApiProperty({ required: false }) - @IsString() - @IsOptional() - content?: string; - - @ApiProperty({ required: false }) - @IsString() - @IsOptional() - @IsUrl() - videoUrl?: string; - - @ApiProperty({ required: false }) - @IsInt() - @IsOptional() - @IsNumber() - order?: number; - - @ApiProperty({ required: false }) - @IsInt() - @IsOptional() - @IsNumber() - durationSeconds?: number; - - @ApiProperty() - @IsUUID() - @IsNotEmpty() - @IsString() - moduleId: string; + @ApiProperty() + @IsString() + @IsNotEmpty() + title: string; + @ApiProperty({ required: false }) + @IsString() + @IsOptional() + content?: string; + @ApiProperty({ required: false }) + @IsString() + @IsOptional() + @IsUrl() + videoUrl?: string; + @ApiProperty({ required: false }) + @IsInt() + @IsOptional() + @IsNumber() + order?: number; + @ApiProperty({ required: false }) + @IsInt() + @IsOptional() + @IsNumber() + durationSeconds?: number; + @ApiProperty() + @IsUUID() + @IsNotEmpty() + @IsString() + moduleId: string; } diff --git a/src/courses/dto/create-module.dto.ts b/src/courses/dto/create-module.dto.ts index 4d3b4c70..573b08ff 100644 --- a/src/courses/dto/create-module.dto.ts +++ b/src/courses/dto/create-module.dto.ts @@ -1,21 +1,18 @@ import { IsString, IsNotEmpty, IsInt, IsOptional, IsUUID, IsNumber } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class CreateModuleDto { - @ApiProperty() - @IsString() - @IsNotEmpty() - title: string; - - @ApiProperty({ required: false }) - @IsInt() - @IsOptional() - @IsNumber() - order?: number; - - @ApiProperty() - @IsUUID() - @IsNotEmpty() - @IsString() - courseId: string; + @ApiProperty() + @IsString() + @IsNotEmpty() + title: string; + @ApiProperty({ required: false }) + @IsInt() + @IsOptional() + @IsNumber() + order?: number; + @ApiProperty() + @IsUUID() + @IsNotEmpty() + @IsString() + courseId: string; } diff --git a/src/courses/dto/update-course.dto.ts b/src/courses/dto/update-course.dto.ts index 28f8c93c..deab1ef0 100644 --- a/src/courses/dto/update-course.dto.ts +++ b/src/courses/dto/update-course.dto.ts @@ -1,11 +1,10 @@ import { CreateCourseDto } from './create-course.dto'; import { PartialType, ApiPropertyOptional } from '@nestjs/swagger'; import { IsString, IsOptional, IsEnum } from 'class-validator'; - export class UpdateCourseDto extends PartialType(CreateCourseDto) { - @ApiPropertyOptional({ enum: ['draft', 'published', 'archived'] }) - @IsString() - @IsOptional() - @IsEnum(['draft', 'published', 'archived']) - status?: string; + @ApiPropertyOptional({ enum: ['draft', 'published', 'archived'] }) + @IsString() + @IsOptional() + @IsEnum(['draft', 'published', 'archived']) + status?: string; } diff --git a/src/courses/enrollments/enrollments.service.ts b/src/courses/enrollments/enrollments.service.ts index 443a771a..ee3c7b21 100644 --- a/src/courses/enrollments/enrollments.service.ts +++ b/src/courses/enrollments/enrollments.service.ts @@ -4,56 +4,50 @@ import { Repository } from 'typeorm'; import { Enrollment } from '../entities/enrollment.entity'; import { Course } from '../entities/course.entity'; import { User } from '../../users/entities/user.entity'; - @Injectable() export class EnrollmentsService { - constructor( + constructor( @InjectRepository(Enrollment) - private enrollmentsRepository: Repository, + private enrollmentsRepository: Repository, @InjectRepository(Course) - private coursesRepository: Repository, + private coursesRepository: Repository, @InjectRepository(User) - private usersRepository: Repository, - ) {} - - async enroll(userId: string, courseId: string): Promise { - const existing = await this.enrollmentsRepository.findOne({ - where: { user: { id: userId }, course: { id: courseId } }, - }); - if (existing) { - throw new ConflictException('User is already enrolled in this course'); + private usersRepository: Repository) { } + async enroll(userId: string, courseId: string): Promise { + const existing = await this.enrollmentsRepository.findOne({ + where: { user: { id: userId }, course: { id: courseId } }, + }); + if (existing) { + throw new ConflictException('User is already enrolled in this course'); + } + const user = await this.usersRepository.findOneBy({ id: userId }); + if (!user) + throw new NotFoundException('User not found'); + const course = await this.coursesRepository.findOneBy({ id: courseId }); + if (!course) + throw new NotFoundException('Course not found'); + const enrollment = this.enrollmentsRepository.create({ + user, + course, + progress: 0, + status: 'active', + }); + return this.enrollmentsRepository.save(enrollment); } - - const user = await this.usersRepository.findOneBy({ id: userId }); - if (!user) throw new NotFoundException('User not found'); - - const course = await this.coursesRepository.findOneBy({ id: courseId }); - if (!course) throw new NotFoundException('Course not found'); - - const enrollment = this.enrollmentsRepository.create({ - user, - course, - progress: 0, - status: 'active', - }); - return this.enrollmentsRepository.save(enrollment); - } - - async findUserEnrollments(userId: string): Promise { - return this.enrollmentsRepository.find({ - where: { user: { id: userId } }, - relations: ['course'], - }); - } - - async updateProgress(enrollmentId: string, progress: number): Promise { - const enrollment = await this.enrollmentsRepository.findOneBy({ id: enrollmentId }); - if (!enrollment) throw new NotFoundException('Enrollment not found'); - - enrollment.progress = progress; - if (progress >= 100) { - enrollment.status = 'completed'; + async findUserEnrollments(userId: string): Promise { + return this.enrollmentsRepository.find({ + where: { user: { id: userId } }, + relations: ['course'], + }); + } + async updateProgress(enrollmentId: string, progress: number): Promise { + const enrollment = await this.enrollmentsRepository.findOneBy({ id: enrollmentId }); + if (!enrollment) + throw new NotFoundException('Enrollment not found'); + enrollment.progress = progress; + if (progress >= 100) { + enrollment.status = 'completed'; + } + return this.enrollmentsRepository.save(enrollment); } - return this.enrollmentsRepository.save(enrollment); - } } diff --git a/src/courses/entities/course-module.entity.ts b/src/courses/entities/course-module.entity.ts index ef52a3e9..c28018db 100644 --- a/src/courses/entities/course-module.entity.ts +++ b/src/courses/entities/course-module.entity.ts @@ -1,36 +1,21 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - OneToMany, - Index, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, Index, DeleteDateColumn, } from 'typeorm'; import { Course } from './course.entity'; import { Lesson } from './lesson.entity'; - @Entity() export class CourseModule { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - title: string; - - @Column({ type: 'int', default: 0 }) - order: number; - - @ManyToOne(() => Course, (course) => course.modules, { onDelete: 'CASCADE' }) - course: Course; - - @Column({ name: 'course_id' }) - @Index() - courseId: string; - - @OneToMany(() => Lesson, (lesson) => lesson.module) - lessons: Lesson[]; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + title: string; + @Column({ type: 'int', default: 0 }) + order: number; + @ManyToOne(() => Course, (course) => course.modules, { onDelete: 'CASCADE' }) + course: Course; + @Column({ name: 'course_id' }) + @Index() + courseId: string; + @OneToMany(() => Lesson, (lesson) => lesson.module) + lessons: Lesson[]; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/courses/entities/course.entity.ts b/src/courses/entities/course.entity.ts index d00ac7a6..ca9a26e1 100644 --- a/src/courses/entities/course.entity.ts +++ b/src/courses/entities/course.entity.ts @@ -1,59 +1,36 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - OneToMany, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, OneToMany, Index, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; import { CourseModule } from './course-module.entity'; import { Enrollment } from './enrollment.entity'; - @Entity() export class Course { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - title: string; - - @Column('text') - description: string; - - @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) - price: number; - - @Column({ default: 'draft' }) // draft, published, archived - @Index() - status: string; - - @Column({ nullable: true }) - thumbnailUrl: string; - - @ManyToOne(() => User, (user) => user.courses) - instructor: User; - - @Column({ name: 'instructor_id' }) - @Index() - instructorId: string; - - @OneToMany(() => CourseModule, (module) => module.course) - modules: CourseModule[]; - - @OneToMany(() => Enrollment, (enrollment) => enrollment.course) - enrollments: Enrollment[]; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + title: string; + @Column('text') + description: string; + @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) + price: number; + @Column({ default: 'draft' }) // draft, published, archived + @Index() + status: string; + @Column({ nullable: true }) + thumbnailUrl: string; + @ManyToOne(() => User, (user) => user.courses) + instructor: User; + @Column({ name: 'instructor_id' }) + @Index() + instructorId: string; + @OneToMany(() => CourseModule, (module) => module.course) + modules: CourseModule[]; + @OneToMany(() => Enrollment, (enrollment) => enrollment.course) + enrollments: Enrollment[]; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/courses/entities/enrollment.entity.ts b/src/courses/entities/enrollment.entity.ts index a0a83bc2..12eec689 100644 --- a/src/courses/entities/enrollment.entity.ts +++ b/src/courses/entities/enrollment.entity.ts @@ -1,46 +1,29 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - CreateDateColumn, - UpdateDateColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, CreateDateColumn, UpdateDateColumn, Index, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; import { Course } from './course.entity'; - @Entity() @Index(['userId', 'status']) @Index(['courseId', 'status']) export class Enrollment { - @PrimaryGeneratedColumn('uuid') - id: string; - - @ManyToOne(() => User, (user) => user.enrollments, { onDelete: 'CASCADE' }) - user: User; - - @Column({ name: 'user_id' }) - @Index() - userId: string; - - @ManyToOne(() => Course, (course) => course.enrollments, { onDelete: 'CASCADE' }) - course: Course; - - @Column({ name: 'course_id' }) - @Index() - courseId: string; - - @Column({ type: 'float', default: 0 }) - progress: number; // 0 to 100 - - @Column({ default: 'active' }) // active, completed, dropped - @Index() - status: string; - - @CreateDateColumn() - enrolledAt: Date; - - @UpdateDateColumn() - lastAccessedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @ManyToOne(() => User, (user) => user.enrollments, { onDelete: 'CASCADE' }) + user: User; + @Column({ name: 'user_id' }) + @Index() + userId: string; + @ManyToOne(() => Course, (course) => course.enrollments, { onDelete: 'CASCADE' }) + course: Course; + @Column({ name: 'course_id' }) + @Index() + courseId: string; + @Column({ type: 'float', default: 0 }) + progress: number; // 0 to 100 + @Column({ default: 'active' }) // active, completed, dropped + @Index() + status: string; + @CreateDateColumn() + enrolledAt: Date; + @UpdateDateColumn() + lastAccessedAt: Date; } diff --git a/src/courses/entities/lesson.entity.ts b/src/courses/entities/lesson.entity.ts index 564d1bae..5f0603b3 100644 --- a/src/courses/entities/lesson.entity.ts +++ b/src/courses/entities/lesson.entity.ts @@ -1,40 +1,24 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - Index, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, Index, DeleteDateColumn, } from 'typeorm'; import { CourseModule } from './course-module.entity'; - @Entity() export class Lesson { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - title: string; - - @Column('text', { nullable: true }) - content: string; - - @Column({ nullable: true }) - videoUrl: string; - - @Column({ type: 'int', default: 0 }) - order: number; - - @Column({ type: 'int', default: 0 }) - durationSeconds: number; - - @ManyToOne(() => CourseModule, (module) => module.lessons, { onDelete: 'CASCADE' }) - module: CourseModule; - - @Column({ name: 'module_id' }) - @Index() - moduleId: string; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + title: string; + @Column('text', { nullable: true }) + content: string; + @Column({ nullable: true }) + videoUrl: string; + @Column({ type: 'int', default: 0 }) + order: number; + @Column({ type: 'int', default: 0 }) + durationSeconds: number; + @ManyToOne(() => CourseModule, (module) => module.lessons, { onDelete: 'CASCADE' }) + module: CourseModule; + @Column({ name: 'module_id' }) + @Index() + moduleId: string; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/courses/guards/ws-jwt-auth.guard.ts b/src/courses/guards/ws-jwt-auth.guard.ts index 0ff404e6..d825d23f 100644 --- a/src/courses/guards/ws-jwt-auth.guard.ts +++ b/src/courses/guards/ws-jwt-auth.guard.ts @@ -1,28 +1,24 @@ import { CanActivate, ExecutionContext, Injectable } from '@nestjs/common'; import { JwtService } from '@nestjs/jwt'; import { Socket } from 'socket.io'; - @Injectable() export class WsJwtAuthGuard implements CanActivate { - constructor(private readonly jwtService: JwtService) {} - - async canActivate(context: ExecutionContext): Promise { - const client: Socket = context.switchToWs().getClient(); - const token = - client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; - - if (!token) { - client.disconnect(true); - return false; + constructor(private readonly jwtService: JwtService) { } + async canActivate(context: ExecutionContext): Promise { + const client: Socket = context.switchToWs().getClient(); + const token = client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; + if (!token) { + client.disconnect(true); + return false; + } + try { + const payload = await this.jwtService.verifyAsync(token); + (client as unknown).user = payload; // attach user context + return true; + } + catch (_err) { + client.disconnect(true); + return false; + } } - - try { - const payload = await this.jwtService.verifyAsync(token); - (client as any).user = payload; // attach user context - return true; - } catch (_err) { - client.disconnect(true); - return false; - } - } } diff --git a/src/courses/lessons/lessons.service.ts b/src/courses/lessons/lessons.service.ts index 233879df..b65ea770 100644 --- a/src/courses/lessons/lessons.service.ts +++ b/src/courses/lessons/lessons.service.ts @@ -4,45 +4,38 @@ import { Repository } from 'typeorm'; import { Lesson } from '../entities/lesson.entity'; import { CreateLessonDto } from '../dto/create-lesson.dto'; import { CourseModule } from '../entities/course-module.entity'; - @Injectable() export class LessonsService { - constructor( + constructor( @InjectRepository(Lesson) - private lessonsRepository: Repository, + private lessonsRepository: Repository, @InjectRepository(CourseModule) - private modulesRepository: Repository, - ) {} - - async create(createLessonDto: CreateLessonDto): Promise { - const module = await this.modulesRepository.findOneBy({ id: createLessonDto.moduleId }); - if (!module) { - throw new NotFoundException(`Module with ID ${createLessonDto.moduleId} not found`); + private modulesRepository: Repository) { } + async create(createLessonDto: CreateLessonDto): Promise { + const module = await this.modulesRepository.findOneBy({ id: createLessonDto.moduleId }); + if (!module) { + throw new NotFoundException(`Module with ID ${createLessonDto.moduleId} not found`); + } + const lesson = this.lessonsRepository.create({ + ...createLessonDto, + module, + }); + return this.lessonsRepository.save(lesson); } - - const lesson = this.lessonsRepository.create({ - ...createLessonDto, - module, - }); - return this.lessonsRepository.save(lesson); - } - - async findOne(id: string): Promise { - const lesson = await this.lessonsRepository.findOneBy({ id }); - if (!lesson) { - throw new NotFoundException(`Lesson with ID ${id} not found`); + async findOne(id: string): Promise { + const lesson = await this.lessonsRepository.findOneBy({ id }); + if (!lesson) { + throw new NotFoundException(`Lesson with ID ${id} not found`); + } + return lesson; + } + async update(id: string, updateData: Partial): Promise { + const lesson = await this.findOne(id); + Object.assign(lesson, updateData); + return this.lessonsRepository.save(lesson); + } + async remove(id: string): Promise { + await this.findOne(id); + await this.lessonsRepository.softDelete(id); } - return lesson; - } - - async update(id: string, updateData: Partial): Promise { - const lesson = await this.findOne(id); - Object.assign(lesson, updateData); - return this.lessonsRepository.save(lesson); - } - - async remove(id: string): Promise { - await this.findOne(id); - await this.lessonsRepository.softDelete(id); - } } diff --git a/src/courses/modules/modules.service.ts b/src/courses/modules/modules.service.ts index 46ad69fe..20345bd3 100644 --- a/src/courses/modules/modules.service.ts +++ b/src/courses/modules/modules.service.ts @@ -5,56 +5,49 @@ import { CourseModule } from '../entities/course-module.entity'; import { CreateModuleDto } from '../dto/create-module.dto'; import { Course } from '../entities/course.entity'; import { Lesson } from '../entities/lesson.entity'; - @Injectable() export class ModulesService { - constructor( + constructor( @InjectRepository(CourseModule) - private modulesRepository: Repository, + private modulesRepository: Repository, @InjectRepository(Course) - private coursesRepository: Repository, - ) {} - - async create(createModuleDto: CreateModuleDto): Promise { - const course = await this.coursesRepository.findOneBy({ id: createModuleDto.courseId }); - if (!course) { - throw new NotFoundException(`Course with ID ${createModuleDto.courseId} not found`); + private coursesRepository: Repository) { } + async create(createModuleDto: CreateModuleDto): Promise { + const course = await this.coursesRepository.findOneBy({ id: createModuleDto.courseId }); + if (!course) { + throw new NotFoundException(`Course with ID ${createModuleDto.courseId} not found`); + } + const module = this.modulesRepository.create({ + ...createModuleDto, + course, + }); + return this.modulesRepository.save(module); } - - const module = this.modulesRepository.create({ - ...createModuleDto, - course, - }); - return this.modulesRepository.save(module); - } - - async findOne(id: string): Promise { - const module = await this.modulesRepository.findOne({ - where: { id }, - relations: ['lessons'], - order: { - lessons: { - order: 'ASC', - } as any, - }, - }); - if (!module) { - throw new NotFoundException(`Module with ID ${id} not found`); + async findOne(id: string): Promise { + const module = await this.modulesRepository.findOne({ + where: { id }, + relations: ['lessons'], + order: { + lessons: { + order: 'ASC', + } as unknown, + }, + }); + if (!module) { + throw new NotFoundException(`Module with ID ${id} not found`); + } + return module; + } + async update(id: string, updateData: Partial): Promise { + const module = await this.findOne(id); + Object.assign(module, updateData); + return this.modulesRepository.save(module); + } + async remove(id: string): Promise { + const module = await this.findOne(id); + await this.modulesRepository.manager.transaction(async (manager) => { + await manager.getRepository(Lesson).softDelete({ moduleId: module.id }); + await manager.getRepository(CourseModule).softDelete(module.id); + }); } - return module; - } - - async update(id: string, updateData: Partial): Promise { - const module = await this.findOne(id); - Object.assign(module, updateData); - return this.modulesRepository.save(module); - } - - async remove(id: string): Promise { - const module = await this.findOne(id); - await this.modulesRepository.manager.transaction(async (manager) => { - await manager.getRepository(Lesson).softDelete({ moduleId: module.id }); - await manager.getRepository(CourseModule).softDelete(module.id); - }); - } } diff --git a/src/data-warehouse/data-warehouse.controller.ts b/src/data-warehouse/data-warehouse.controller.ts index b4923c22..d0588677 100644 --- a/src/data-warehouse/data-warehouse.controller.ts +++ b/src/data-warehouse/data-warehouse.controller.ts @@ -6,213 +6,242 @@ import { DataLineageService } from './lineage/data-lineage.service'; import { IncrementalLoaderService } from './loading/incremental-loader.service'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RolesGuard } from '../auth/guards/roles.guard'; - @Controller('data-warehouse') @UseGuards(JwtAuthGuard, RolesGuard) export class DataWarehouseController { - constructor( - private readonly etlService: ETLPipelineService, - private readonly modelingService: DimensionalModelingService, - private readonly qualityService: DataQualityService, - private readonly lineageService: DataLineageService, - private readonly loaderService: IncrementalLoaderService, - ) {} - - // ETL Pipeline endpoints - @Post('etl/pipeline') - async createETLPipeline(@Body() config: any, @Request() _req): Promise { - const pipeline = await this.etlService.createPipeline(config); - return { success: true, pipeline }; - } - - @Get('etl/pipeline/:id') - async getETLPipeline(@Param('id') id: string): Promise { - const pipeline = await this.etlService.getJobStatus(id); - return { success: true, pipeline }; - } - - @Get('etl/pipelines') - async getAllETLPipelines(): Promise { - const pipelines = await this.etlService.getAllJobs(); - return { success: true, pipelines }; - } - - // Dimensional Modeling endpoints - @Post('modeling/star-schema') - async createStarSchema(@Body() body: { name: string; factTable: any; dimensionTables: any[] }): Promise { - const model = await this.modelingService.createStarSchema( - body.name, - body.factTable, - body.dimensionTables, - ); - return { success: true, model }; - } - - @Post('modeling/snowflake-schema') - async createSnowflakeSchema( - @Body() body: { name: string; factTable: any; dimensionTables: any[]; subDimensions: any }, - ): Promise { - const model = await this.modelingService.createSnowflakeSchema( - body.name, - body.factTable, - body.dimensionTables, - body.subDimensions, - ); - return { success: true, model }; - } - - @Get('modeling/models') - async getAllModels(): Promise { - const models = await this.modelingService.getAllModels(); - return { success: true, models }; - } - - @Get('modeling/model/:id') - async getModel(@Param('id') id: string): Promise { - const model = await this.modelingService.getModel(id); - return { success: true, model }; - } - - @Post('modeling/query') - async createQuery(@Body() queryConfig: any): Promise { - const query = await this.modelingService.createQuery(queryConfig); - return { success: true, query }; - } - - @Post('modeling/query/:id/execute') - async executeQuery(@Param('id') id: string, @Body() body: { parameters?: any }): Promise { - const results = await this.modelingService.executeQuery(id, body.parameters); - return { success: true, results }; - } - - // Data Quality endpoints - @Post('quality/profile') - async createQualityProfile(@Body() profileConfig: any): Promise { - const profile = await this.qualityService.createProfile(profileConfig); - return { success: true, profile }; - } - - @Post('quality/profiles/standard') - async createStandardProfiles(): Promise { - const profiles = await this.qualityService.createStandardProfiles(); - return { success: true, profiles }; - } - - @Post('quality/check/:profileId') - async runQualityCheck(@Param('profileId') profileId: string, @Body() body: { data: any[] }): Promise { - const check = await this.qualityService.runQualityChecks(profileId, body.data); - return { success: true, check }; - } - - @Get('quality/checks/:profileId') - async getQualityChecks(@Param('profileId') profileId: string): Promise { - const checks = await this.qualityService.getChecksForProfile(profileId); - return { success: true, checks }; - } - - @Get('quality/issues') - async getQualityIssues( - @Query('profileId') profileId?: string, - @Query('severity') severity?: string, - @Query('resolved') resolved?: string, - ): Promise { - const resolvedBool = resolved === 'true' ? true : resolved === 'false' ? false : undefined; - const issues = await this.qualityService.getQualityIssues(profileId, severity, resolvedBool); - return { success: true, issues }; - } - - // Data Lineage endpoints - @Post('lineage/graph') - async createLineageGraph(@Body() graphConfig: any): Promise { - const graph = await this.lineageService.createGraph(graphConfig); - return { success: true, graph }; - } - - @Post('lineage/graphs/standard') - async createStandardLineage(): Promise { - const graph = await this.lineageService.createStandardLineage(); - return { success: true, graph }; - } - - @Get('lineage/graphs') - async getAllLineageGraphs(): Promise { - const graphs = await this.lineageService.getAllGraphs(); - return { success: true, graphs }; - } - - @Post('lineage/graph/:id/node') - async addLineageNode(@Param('id') graphId: string, @Body() nodeConfig: any): Promise { - const node = await this.lineageService.addNode(graphId, nodeConfig); - return { success: true, node }; - } - - @Post('lineage/graph/:id/edge') - async addLineageEdge(@Param('id') graphId: string, @Body() edgeConfig: any): Promise { - const edge = await this.lineageService.addEdge(graphId, edgeConfig); - return { success: true, edge }; - } - - @Post('lineage/graph/:id/trace/:nodeId') - async traceLineage( - @Param('id') graphId: string, - @Param('nodeId') nodeId: string, - @Body() body: { traceType?: 'upstream' | 'downstream' | 'complete' }, - ): Promise { - const traceType = body.traceType || 'complete'; - const trace = await this.lineageService.traceLineage(graphId, nodeId, traceType); - return { success: true, trace }; - } - - @Post('lineage/graph/:id/impact/:nodeId') - async analyzeImpact(@Param('id') graphId: string, @Param('nodeId') nodeId: string): Promise { - const analysis = await this.lineageService.analyzeImpact(graphId, nodeId); - return { success: true, analysis }; - } - - // Incremental Loading endpoints - @Post('loading/job') - async createLoadJob(@Body() body: { config: any; source: any; target: any }): Promise { - const job = await this.loaderService.createLoadJob(body.config, body.source, body.target); - return { success: true, job }; - } - - @Post('loading/job/:id/execute') - async executeLoadJob( - @Param('id') jobId: string, - @Body() body: { sourceTable: string; targetTable: string }, - ): Promise { - const job = await this.loaderService.executeLoad(jobId, body.sourceTable, body.targetTable); - return { success: true, job }; - } - - @Get('loading/jobs') - async getAllLoadJobs(): Promise { - const jobs = await this.loaderService.getAllJobs(); - return { success: true, jobs }; - } - - @Get('loading/job/:id') - async getLoadJob(@Param('id') id: string): Promise { - const job = await this.loaderService.getJobStatus(id); - return { success: true, job }; - } - - @Post('loading/watermark') - async setWatermark(@Body() body: { tableName: string; columnName: string; value: any }): Promise { - const watermark = await this.loaderService.setWatermark( - body.tableName, - body.columnName, - body.value, - ); - return { success: true, watermark }; - } - - @Get('loading/watermark/:tableName/:columnName') - async getWatermark( - @Param('tableName') tableName: string, - @Param('columnName') columnName: string, - ): Promise { - const watermark = await this.loaderService.getWatermark(tableName, columnName); - return { success: true, watermark }; - } + constructor(private readonly etlService: ETLPipelineService, private readonly modelingService: DimensionalModelingService, private readonly qualityService: DataQualityService, private readonly lineageService: DataLineageService, private readonly loaderService: IncrementalLoaderService) { } + // ETL Pipeline endpoints + @Post('etl/pipeline') + async createETLPipeline( + @Body() + config: unknown, + @Request() + _req): Promise { + const pipeline = await this.etlService.createPipeline(config); + return { success: true, pipeline }; + } + @Get('etl/pipeline/:id') + async getETLPipeline( + @Param('id') + id: string): Promise { + const pipeline = await this.etlService.getJobStatus(id); + return { success: true, pipeline }; + } + @Get('etl/pipelines') + async getAllETLPipelines(): Promise { + const pipelines = await this.etlService.getAllJobs(); + return { success: true, pipelines }; + } + // Dimensional Modeling endpoints + @Post('modeling/star-schema') + async createStarSchema( + @Body() + body: { + name: string; + factTable: unknown; + dimensionTables: unknown[]; + }): Promise { + const model = await this.modelingService.createStarSchema(body.name, body.factTable, body.dimensionTables); + return { success: true, model }; + } + @Post('modeling/snowflake-schema') + async createSnowflakeSchema( + @Body() + body: { + name: string; + factTable: unknown; + dimensionTables: unknown[]; + subDimensions: unknown; + }): Promise { + const model = await this.modelingService.createSnowflakeSchema(body.name, body.factTable, body.dimensionTables, body.subDimensions); + return { success: true, model }; + } + @Get('modeling/models') + async getAllModels(): Promise { + const models = await this.modelingService.getAllModels(); + return { success: true, models }; + } + @Get('modeling/model/:id') + async getModel( + @Param('id') + id: string): Promise { + const model = await this.modelingService.getModel(id); + return { success: true, model }; + } + @Post('modeling/query') + async createQuery( + @Body() + queryConfig: unknown): Promise { + const query = await this.modelingService.createQuery(queryConfig); + return { success: true, query }; + } + @Post('modeling/query/:id/execute') + async executeQuery( + @Param('id') + id: string, + @Body() + body: { + parameters?: unknown; + }): Promise { + const results = await this.modelingService.executeQuery(id, body.parameters); + return { success: true, results }; + } + // Data Quality endpoints + @Post('quality/profile') + async createQualityProfile( + @Body() + profileConfig: unknown): Promise { + const profile = await this.qualityService.createProfile(profileConfig); + return { success: true, profile }; + } + @Post('quality/profiles/standard') + async createStandardProfiles(): Promise { + const profiles = await this.qualityService.createStandardProfiles(); + return { success: true, profiles }; + } + @Post('quality/check/:profileId') + async runQualityCheck( + @Param('profileId') + profileId: string, + @Body() + body: { + data: unknown[]; + }): Promise { + const check = await this.qualityService.runQualityChecks(profileId, body.data); + return { success: true, check }; + } + @Get('quality/checks/:profileId') + async getQualityChecks( + @Param('profileId') + profileId: string): Promise { + const checks = await this.qualityService.getChecksForProfile(profileId); + return { success: true, checks }; + } + @Get('quality/issues') + async getQualityIssues( + @Query('profileId') + profileId?: string, + @Query('severity') + severity?: string, + @Query('resolved') + resolved?: string): Promise { + const resolvedBool = resolved === 'true' ? true : resolved === 'false' ? false : undefined; + const issues = await this.qualityService.getQualityIssues(profileId, severity, resolvedBool); + return { success: true, issues }; + } + // Data Lineage endpoints + @Post('lineage/graph') + async createLineageGraph( + @Body() + graphConfig: unknown): Promise { + const graph = await this.lineageService.createGraph(graphConfig); + return { success: true, graph }; + } + @Post('lineage/graphs/standard') + async createStandardLineage(): Promise { + const graph = await this.lineageService.createStandardLineage(); + return { success: true, graph }; + } + @Get('lineage/graphs') + async getAllLineageGraphs(): Promise { + const graphs = await this.lineageService.getAllGraphs(); + return { success: true, graphs }; + } + @Post('lineage/graph/:id/node') + async addLineageNode( + @Param('id') + graphId: string, + @Body() + nodeConfig: unknown): Promise { + const node = await this.lineageService.addNode(graphId, nodeConfig); + return { success: true, node }; + } + @Post('lineage/graph/:id/edge') + async addLineageEdge( + @Param('id') + graphId: string, + @Body() + edgeConfig: unknown): Promise { + const edge = await this.lineageService.addEdge(graphId, edgeConfig); + return { success: true, edge }; + } + @Post('lineage/graph/:id/trace/:nodeId') + async traceLineage( + @Param('id') + graphId: string, + @Param('nodeId') + nodeId: string, + @Body() + body: { + traceType?: 'upstream' | 'downstream' | 'complete'; + }): Promise { + const traceType = body.traceType || 'complete'; + const trace = await this.lineageService.traceLineage(graphId, nodeId, traceType); + return { success: true, trace }; + } + @Post('lineage/graph/:id/impact/:nodeId') + async analyzeImpact( + @Param('id') + graphId: string, + @Param('nodeId') + nodeId: string): Promise { + const analysis = await this.lineageService.analyzeImpact(graphId, nodeId); + return { success: true, analysis }; + } + // Incremental Loading endpoints + @Post('loading/job') + async createLoadJob( + @Body() + body: { + config: unknown; + source: unknown; + target: unknown; + }): Promise { + const job = await this.loaderService.createLoadJob(body.config, body.source, body.target); + return { success: true, job }; + } + @Post('loading/job/:id/execute') + async executeLoadJob( + @Param('id') + jobId: string, + @Body() + body: { + sourceTable: string; + targetTable: string; + }): Promise { + const job = await this.loaderService.executeLoad(jobId, body.sourceTable, body.targetTable); + return { success: true, job }; + } + @Get('loading/jobs') + async getAllLoadJobs(): Promise { + const jobs = await this.loaderService.getAllJobs(); + return { success: true, jobs }; + } + @Get('loading/job/:id') + async getLoadJob( + @Param('id') + id: string): Promise { + const job = await this.loaderService.getJobStatus(id); + return { success: true, job }; + } + @Post('loading/watermark') + async setWatermark( + @Body() + body: { + tableName: string; + columnName: string; + value: unknown; + }): Promise { + const watermark = await this.loaderService.setWatermark(body.tableName, body.columnName, body.value); + return { success: true, watermark }; + } + @Get('loading/watermark/:tableName/:columnName') + async getWatermark( + @Param('tableName') + tableName: string, + @Param('columnName') + columnName: string): Promise { + const watermark = await this.loaderService.getWatermark(tableName, columnName); + return { success: true, watermark }; + } } diff --git a/src/data-warehouse/data-warehouse.module.ts b/src/data-warehouse/data-warehouse.module.ts index 841c2207..031998e5 100644 --- a/src/data-warehouse/data-warehouse.module.ts +++ b/src/data-warehouse/data-warehouse.module.ts @@ -5,23 +5,23 @@ import { DataQualityService } from './quality/data-quality.service'; import { DataLineageService } from './lineage/data-lineage.service'; import { IncrementalLoaderService } from './loading/incremental-loader.service'; import { DataWarehouseController } from './data-warehouse.controller'; - @Module({ - imports: [], - controllers: [DataWarehouseController], - providers: [ - ETLPipelineService, - DimensionalModelingService, - DataQualityService, - DataLineageService, - IncrementalLoaderService, - ], - exports: [ - ETLPipelineService, - DimensionalModelingService, - DataQualityService, - DataLineageService, - IncrementalLoaderService, - ], + imports: [], + controllers: [DataWarehouseController], + providers: [ + ETLPipelineService, + DimensionalModelingService, + DataQualityService, + DataLineageService, + IncrementalLoaderService, + ], + exports: [ + ETLPipelineService, + DimensionalModelingService, + DataQualityService, + DataLineageService, + IncrementalLoaderService, + ], }) -export class DataWarehouseModule {} +export class DataWarehouseModule { +} diff --git a/src/data-warehouse/etl/etl-pipeline.service.ts b/src/data-warehouse/etl/etl-pipeline.service.ts index 37d90923..84d1abcb 100644 --- a/src/data-warehouse/etl/etl-pipeline.service.ts +++ b/src/data-warehouse/etl/etl-pipeline.service.ts @@ -1,374 +1,323 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface ETLJob { - id: string; - name: string; - source: string; - target: string; - status: 'pending' | 'running' | 'completed' | 'failed'; - startTime: Date; - endTime?: Date; - duration?: number; - recordsProcessed: number; - recordsFailed: number; - config: ETLConfig; + id: string; + name: string; + source: string; + target: string; + status: 'pending' | 'running' | 'completed' | 'failed'; + startTime: Date; + endTime?: Date; + duration?: number; + recordsProcessed: number; + recordsFailed: number; + config: ETLConfig; } - export interface ETLConfig { - sourceConnection: DataSourceConfig; - targetConnection: DataSourceConfig; - transformations: TransformationRule[]; - schedule?: string; - incremental?: boolean; - batchSize?: number; + sourceConnection: DataSourceConfig; + targetConnection: DataSourceConfig; + transformations: TransformationRule[]; + schedule?: string; + incremental?: boolean; + batchSize?: number; } - export interface DataSourceConfig { - type: 'postgres' | 'mysql' | 'mongodb' | 'api' | 'file'; - host?: string; - port?: number; - database?: string; - username?: string; - password?: string; - collection?: string; - endpoint?: string; - filePath?: string; - query?: string; + type: 'postgres' | 'mysql' | 'mongodb' | 'api' | 'file'; + host?: string; + port?: number; + database?: string; + username?: string; + password?: string; + collection?: string; + endpoint?: string; + filePath?: string; + query?: string; } - export interface TransformationRule { - id: string; - sourceField: string; - targetField: string; - transformationType: 'map' | 'filter' | 'aggregate' | 'calculate' | 'format'; - config: any; + id: string; + sourceField: string; + targetField: string; + transformationType: 'map' | 'filter' | 'aggregate' | 'calculate' | 'format'; + config: unknown; } - export interface ExtractedData { - data: any[]; - metadata: { - source: string; - timestamp: Date; - recordCount: number; - }; + data: unknown[]; + metadata: { + source: string; + timestamp: Date; + recordCount: number; + }; } - export interface TransformedData { - data: any[]; - metadata: { - transformationsApplied: string[]; - timestamp: Date; - recordCount: number; - }; + data: unknown[]; + metadata: { + transformationsApplied: string[]; + timestamp: Date; + recordCount: number; + }; } - @Injectable() export class ETLPipelineService { - private readonly logger = new Logger(ETLPipelineService.name); - private jobs: Map = new Map(); - - /** - * Create and execute an ETL pipeline - */ - async createPipeline(config: ETLConfig): Promise { - const jobId = uuidv4(); - const job: ETLJob = { - id: jobId, - name: `ETL_Pipeline_${new Date().toISOString()}`, - source: config.sourceConnection.type, - target: config.targetConnection.type, - status: 'pending', - startTime: new Date(), - recordsProcessed: 0, - recordsFailed: 0, - config, - }; - - this.jobs.set(jobId, job); - this.logger.log(`Created ETL pipeline job ${jobId}`); - - // Start the pipeline execution - this.executePipeline(jobId); - - return job; - } - - /** - * Execute the ETL pipeline - */ - private async executePipeline(jobId: string): Promise { - const job = this.jobs.get(jobId); - if (!job) { - throw new Error(`Job ${jobId} not found`); + private readonly logger = new Logger(ETLPipelineService.name); + private jobs: Map = new Map(); + /** + * Create and execute an ETL pipeline + */ + async createPipeline(config: ETLConfig): Promise { + const jobId = uuidv4(); + const job: ETLJob = { + id: jobId, + name: `ETL_Pipeline_${new Date().toISOString()}`, + source: config.sourceConnection.type, + target: config.targetConnection.type, + status: 'pending', + startTime: new Date(), + recordsProcessed: 0, + recordsFailed: 0, + config, + }; + this.jobs.set(jobId, job); + this.logger.log(`Created ETL pipeline job ${jobId}`); + // Start the pipeline execution + this.executePipeline(jobId); + return job; } - - job.status = 'running'; - job.startTime = new Date(); - - try { - // Extract phase - this.logger.log(`Starting extraction for job ${jobId}`); - const extractedData = await this.extract(job.config.sourceConnection); - - // Transform phase - this.logger.log(`Starting transformation for job ${jobId}`); - const transformedData = await this.transform(extractedData, job.config.transformations); - - // Load phase - this.logger.log(`Starting loading for job ${jobId}`); - await this.load(transformedData, job.config.targetConnection); - - // Update job status - job.status = 'completed'; - job.endTime = new Date(); - job.duration = job.endTime.getTime() - job.startTime.getTime(); - job.recordsProcessed = transformedData.data.length; - - this.logger.log(`ETL pipeline ${jobId} completed successfully`); - } catch (error) { - this.logger.error(`ETL pipeline ${jobId} failed: ${error.message}`); - job.status = 'failed'; - job.endTime = new Date(); - job.duration = job.endTime.getTime() - job.startTime.getTime(); - job.recordsFailed = 1; + /** + * Execute the ETL pipeline + */ + private async executePipeline(jobId: string): Promise { + const job = this.jobs.get(jobId); + if (!job) { + throw new Error(`Job ${jobId} not found`); + } + job.status = 'running'; + job.startTime = new Date(); + try { + // Extract phase + this.logger.log(`Starting extraction for job ${jobId}`); + const extractedData = await this.extract(job.config.sourceConnection); + // Transform phase + this.logger.log(`Starting transformation for job ${jobId}`); + const transformedData = await this.transform(extractedData, job.config.transformations); + // Load phase + this.logger.log(`Starting loading for job ${jobId}`); + await this.load(transformedData, job.config.targetConnection); + // Update job status + job.status = 'completed'; + job.endTime = new Date(); + job.duration = job.endTime.getTime() - job.startTime.getTime(); + job.recordsProcessed = transformedData.data.length; + this.logger.log(`ETL pipeline ${jobId} completed successfully`); + } + catch (error) { + this.logger.error(`ETL pipeline ${jobId} failed: ${error.message}`); + job.status = 'failed'; + job.endTime = new Date(); + job.duration = job.endTime.getTime() - job.startTime.getTime(); + job.recordsFailed = 1; + } } - } - - /** - * Extract data from source - */ - private async extract(sourceConfig: DataSourceConfig): Promise { - // This is a simplified implementation - // In a real system, this would connect to various data sources - - let data: any[] = []; - - switch (sourceConfig.type) { - case 'postgres': - // Connect to PostgreSQL and execute query - data = await this.extractFromPostgres(sourceConfig); - break; - case 'mysql': - // Connect to MySQL and execute query - data = await this.extractFromMysql(sourceConfig); - break; - case 'mongodb': - // Connect to MongoDB and fetch documents - data = await this.extractFromMongoDB(sourceConfig); - break; - case 'api': - // Call external API - data = await this.extractFromAPI(sourceConfig); - break; - case 'file': - // Read from file - data = await this.extractFromFile(sourceConfig); - break; - default: - throw new Error(`Unsupported source type: ${sourceConfig.type}`); + /** + * Extract data from source + */ + private async extract(sourceConfig: DataSourceConfig): Promise { + // This is a simplified implementation + // In a real system, this would connect to various data sources + let data: unknown[] = []; + switch (sourceConfig.type) { + case 'postgres': + // Connect to PostgreSQL and execute query + data = await this.extractFromPostgres(sourceConfig); + break; + case 'mysql': + // Connect to MySQL and execute query + data = await this.extractFromMysql(sourceConfig); + break; + case 'mongodb': + // Connect to MongoDB and fetch documents + data = await this.extractFromMongoDB(sourceConfig); + break; + case 'api': + // Call external API + data = await this.extractFromAPI(sourceConfig); + break; + case 'file': + // Read from file + data = await this.extractFromFile(sourceConfig); + break; + default: + throw new Error(`Unsupported source type: ${sourceConfig.type}`); + } + return { + data, + metadata: { + source: sourceConfig.type, + timestamp: new Date(), + recordCount: data.length, + }, + }; } - - return { - data, - metadata: { - source: sourceConfig.type, - timestamp: new Date(), - recordCount: data.length, - }, - }; - } - - /** - * Transform extracted data - */ - private async transform( - extractedData: ExtractedData, - transformations: TransformationRule[], - ): Promise { - let transformedData = [...extractedData.data]; - const appliedTransformations: string[] = []; - - for (const rule of transformations) { - switch (rule.transformationType) { - case 'map': - transformedData = transformedData.map((item) => ({ - ...item, - [rule.targetField]: this.applyMapping(item[rule.sourceField], rule.config), - })); - break; - - case 'filter': - transformedData = transformedData.filter((item) => - this.applyFilter(item[rule.sourceField], rule.config), - ); - break; - - case 'calculate': - transformedData = transformedData.map((item) => ({ - ...item, - [rule.targetField]: this.applyCalculation(item, rule.config), - })); - break; - - case 'format': - transformedData = transformedData.map((item) => ({ - ...item, - [rule.targetField]: this.applyFormatting(item[rule.sourceField], rule.config), - })); - break; - } - - appliedTransformations.push( - `${rule.transformationType}:${rule.sourceField}->${rule.targetField}`, - ); + /** + * Transform extracted data + */ + private async transform(extractedData: ExtractedData, transformations: TransformationRule[]): Promise { + let transformedData = [...extractedData.data]; + const appliedTransformations: string[] = []; + for (const rule of transformations) { + switch (rule.transformationType) { + case 'map': + transformedData = transformedData.map((item) => ({ + ...item, + [rule.targetField]: this.applyMapping(item[rule.sourceField], rule.config), + })); + break; + case 'filter': + transformedData = transformedData.filter((item) => this.applyFilter(item[rule.sourceField], rule.config)); + break; + case 'calculate': + transformedData = transformedData.map((item) => ({ + ...item, + [rule.targetField]: this.applyCalculation(item, rule.config), + })); + break; + case 'format': + transformedData = transformedData.map((item) => ({ + ...item, + [rule.targetField]: this.applyFormatting(item[rule.sourceField], rule.config), + })); + break; + } + appliedTransformations.push(`${rule.transformationType}:${rule.sourceField}->${rule.targetField}`); + } + return { + data: transformedData, + metadata: { + transformationsApplied: appliedTransformations, + timestamp: new Date(), + recordCount: transformedData.length, + }, + }; } - - return { - data: transformedData, - metadata: { - transformationsApplied: appliedTransformations, - timestamp: new Date(), - recordCount: transformedData.length, - }, - }; - } - - /** - * Load transformed data to target - */ - private async load( - transformedData: TransformedData, - targetConfig: DataSourceConfig, - ): Promise { - // This is a simplified implementation - // In a real system, this would connect to the target data warehouse - - switch (targetConfig.type) { - case 'postgres': - await this.loadToPostgres(transformedData, targetConfig); - break; - case 'mysql': - await this.loadToMysql(transformedData, targetConfig); - break; - case 'mongodb': - await this.loadToMongoDB(transformedData, targetConfig); - break; - default: - throw new Error(`Unsupported target type: ${targetConfig.type}`); + /** + * Load transformed data to target + */ + private async load(transformedData: TransformedData, targetConfig: DataSourceConfig): Promise { + // This is a simplified implementation + // In a real system, this would connect to the target data warehouse + switch (targetConfig.type) { + case 'postgres': + await this.loadToPostgres(transformedData, targetConfig); + break; + case 'mysql': + await this.loadToMysql(transformedData, targetConfig); + break; + case 'mongodb': + await this.loadToMongoDB(transformedData, targetConfig); + break; + default: + throw new Error(`Unsupported target type: ${targetConfig.type}`); + } + } + /** + * Get job status + */ + async getJobStatus(jobId: string): Promise { + return this.jobs.get(jobId) || null; + } + /** + * Get all jobs + */ + async getAllJobs(): Promise { + return Array.from(this.jobs.values()); + } + /** + * Cancel a running job + */ + async cancelJob(jobId: string): Promise { + const job = this.jobs.get(jobId); + if (!job || job.status !== 'running') { + return false; + } + job.status = 'failed'; + job.endTime = new Date(); + return true; + } + // Helper methods for data source operations + private async extractFromPostgres(config: DataSourceConfig): Promise { + // Implementation would use a PostgreSQL client + this.logger.log(`Extracting from PostgreSQL: ${config.database}`); + return []; // Placeholder + } + private async extractFromMysql(config: DataSourceConfig): Promise { + // Implementation would use a MySQL client + this.logger.log(`Extracting from MySQL: ${config.database}`); + return []; // Placeholder + } + private async extractFromMongoDB(config: DataSourceConfig): Promise { + // Implementation would use MongoDB client + this.logger.log(`Extracting from MongoDB: ${config.database}`); + return []; // Placeholder + } + private async extractFromAPI(config: DataSourceConfig): Promise { + // Implementation would make HTTP requests + this.logger.log(`Extracting from API: ${config.endpoint}`); + return []; // Placeholder + } + private async extractFromFile(config: DataSourceConfig): Promise { + // Implementation would read from file system + this.logger.log(`Extracting from file: ${config.filePath}`); + return []; // Placeholder + } + private async loadToPostgres(data: TransformedData, config: DataSourceConfig): Promise { + // Implementation would use a PostgreSQL client + this.logger.log(`Loading to PostgreSQL: ${config.database}`); + } + private async loadToMysql(data: TransformedData, config: DataSourceConfig): Promise { + // Implementation would use a MySQL client + this.logger.log(`Loading to MySQL: ${config.database}`); } - } - - /** - * Get job status - */ - async getJobStatus(jobId: string): Promise { - return this.jobs.get(jobId) || null; - } - - /** - * Get all jobs - */ - async getAllJobs(): Promise { - return Array.from(this.jobs.values()); - } - - /** - * Cancel a running job - */ - async cancelJob(jobId: string): Promise { - const job = this.jobs.get(jobId); - if (!job || job.status !== 'running') { - return false; + private async loadToMongoDB(data: TransformedData, config: DataSourceConfig): Promise { + // Implementation would use MongoDB client + this.logger.log(`Loading to MongoDB: ${config.database}`); } - - job.status = 'failed'; - job.endTime = new Date(); - return true; - } - - // Helper methods for data source operations - private async extractFromPostgres(config: DataSourceConfig): Promise { - // Implementation would use a PostgreSQL client - this.logger.log(`Extracting from PostgreSQL: ${config.database}`); - return []; // Placeholder - } - - private async extractFromMysql(config: DataSourceConfig): Promise { - // Implementation would use a MySQL client - this.logger.log(`Extracting from MySQL: ${config.database}`); - return []; // Placeholder - } - - private async extractFromMongoDB(config: DataSourceConfig): Promise { - // Implementation would use MongoDB client - this.logger.log(`Extracting from MongoDB: ${config.database}`); - return []; // Placeholder - } - - private async extractFromAPI(config: DataSourceConfig): Promise { - // Implementation would make HTTP requests - this.logger.log(`Extracting from API: ${config.endpoint}`); - return []; // Placeholder - } - - private async extractFromFile(config: DataSourceConfig): Promise { - // Implementation would read from file system - this.logger.log(`Extracting from file: ${config.filePath}`); - return []; // Placeholder - } - - private async loadToPostgres(data: TransformedData, config: DataSourceConfig): Promise { - // Implementation would use a PostgreSQL client - this.logger.log(`Loading to PostgreSQL: ${config.database}`); - } - - private async loadToMysql(data: TransformedData, config: DataSourceConfig): Promise { - // Implementation would use a MySQL client - this.logger.log(`Loading to MySQL: ${config.database}`); - } - - private async loadToMongoDB(data: TransformedData, config: DataSourceConfig): Promise { - // Implementation would use MongoDB client - this.logger.log(`Loading to MongoDB: ${config.database}`); - } - - // Transformation helper methods - private applyMapping(value: any, config: any): any { - if (config.mapping && config.mapping[value] !== undefined) { - return config.mapping[value]; + // Transformation helper methods + private applyMapping(value: unknown, config: unknown): unknown { + if (config.mapping && config.mapping[value] !== undefined) { + return config.mapping[value]; + } + return value; } - return value; - } - - private applyFilter(value: any, config: any): boolean { - if (config.operator === 'equals') { - return value === config.value; - } else if (config.operator === 'greaterThan') { - return value > config.value; - } else if (config.operator === 'lessThan') { - return value < config.value; + private applyFilter(value: unknown, config: unknown): boolean { + if (config.operator === 'equals') { + return value === config.value; + } + else if (config.operator === 'greaterThan') { + return value > config.value; + } + else if (config.operator === 'lessThan') { + return value < config.value; + } + return true; } - return true; - } - - private applyCalculation(item: any, config: any): any { - if (config.operation === 'sum') { - return (item[config.field1] || 0) + (item[config.field2] || 0); - } else if (config.operation === 'multiply') { - return (item[config.field1] || 0) * (item[config.field2] || 0); + private applyCalculation(item: unknown, config: unknown): unknown { + if (config.operation === 'sum') { + return (item[config.field1] || 0) + (item[config.field2] || 0); + } + else if (config.operation === 'multiply') { + return (item[config.field1] || 0) * (item[config.field2] || 0); + } + return item[config.field1]; } - return item[config.field1]; - } - - private applyFormatting(value: any, config: any): any { - if (config.format === 'date') { - return new Date(value).toISOString(); - } else if (config.format === 'uppercase') { - return String(value).toUpperCase(); - } else if (config.format === 'lowercase') { - return String(value).toLowerCase(); + private applyFormatting(value: unknown, config: unknown): unknown { + if (config.format === 'date') { + return new Date(value).toISOString(); + } + else if (config.format === 'uppercase') { + return String(value).toUpperCase(); + } + else if (config.format === 'lowercase') { + return String(value).toLowerCase(); + } + return value; } - return value; - } } diff --git a/src/data-warehouse/lineage/data-lineage.service.ts b/src/data-warehouse/lineage/data-lineage.service.ts index d6f88fe0..1dfa7182 100644 --- a/src/data-warehouse/lineage/data-lineage.service.ts +++ b/src/data-warehouse/lineage/data-lineage.service.ts @@ -1,452 +1,371 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface DataLineageNode { - id: string; - name: string; - type: 'source' | 'transformation' | 'target' | 'process'; - description: string; - system: string; - table?: string; - column?: string; - metadata: any; - createdAt: Date; + id: string; + name: string; + type: 'source' | 'transformation' | 'target' | 'process'; + description: string; + system: string; + table?: string; + column?: string; + metadata: unknown; + createdAt: Date; } - export interface DataLineageEdge { - id: string; - sourceId: string; - targetId: string; - transformation?: string; - description: string; - timestamp: Date; - metadata: any; + id: string; + sourceId: string; + targetId: string; + transformation?: string; + description: string; + timestamp: Date; + metadata: unknown; } - export interface DataLineageGraph { - id: string; - name: string; - description: string; - nodes: DataLineageNode[]; - edges: DataLineageEdge[]; - createdAt: Date; - updatedAt: Date; + id: string; + name: string; + description: string; + nodes: DataLineageNode[]; + edges: DataLineageEdge[]; + createdAt: Date; + updatedAt: Date; } - export interface LineageTrace { - id: string; - nodeId: string; - traceType: 'upstream' | 'downstream' | 'complete'; - path: LineagePath[]; - createdAt: Date; + id: string; + nodeId: string; + traceType: 'upstream' | 'downstream' | 'complete'; + path: LineagePath[]; + createdAt: Date; } - export interface LineagePath { - fromNode: string; - toNode: string; - transformation: string; - timestamp: Date; + fromNode: string; + toNode: string; + transformation: string; + timestamp: Date; } - export interface ImpactAnalysis { - id: string; - nodeId: string; - affectedNodes: string[]; - impactLevel: 'high' | 'medium' | 'low'; - analysis: string; - timestamp: Date; + id: string; + nodeId: string; + affectedNodes: string[]; + impactLevel: 'high' | 'medium' | 'low'; + analysis: string; + timestamp: Date; } - @Injectable() export class DataLineageService { - private readonly logger = new Logger(DataLineageService.name); - private graphs: Map = new Map(); - private traces: Map = new Map(); - private impactAnalyses: Map = new Map(); - - /** - * Create a new lineage graph - */ - async createGraph( - graphConfig: Omit, - ): Promise { - const graphId = uuidv4(); - const graph: DataLineageGraph = { - id: graphId, - ...graphConfig, - nodes: [], - edges: [], - createdAt: new Date(), - updatedAt: new Date(), - }; - - this.graphs.set(graphId, graph); - this.logger.log(`Created lineage graph ${graphId}: ${graph.name}`); - - return graph; - } - - /** - * Get a lineage graph - */ - async getGraph(graphId: string): Promise { - return this.graphs.get(graphId) || null; - } - - /** - * Get all lineage graphs - */ - async getAllGraphs(): Promise { - return Array.from(this.graphs.values()); - } - - /** - * Add a node to a lineage graph - */ - async addNode( - graphId: string, - nodeConfig: Omit, - ): Promise { - const graph = this.graphs.get(graphId); - if (!graph) { - throw new Error(`Graph ${graphId} not found`); + private readonly logger = new Logger(DataLineageService.name); + private graphs: Map = new Map(); + private traces: Map = new Map(); + private impactAnalyses: Map = new Map(); + /** + * Create a new lineage graph + */ + async createGraph(graphConfig: Omit): Promise { + const graphId = uuidv4(); + const graph: DataLineageGraph = { + id: graphId, + ...graphConfig, + nodes: [], + edges: [], + createdAt: new Date(), + updatedAt: new Date(), + }; + this.graphs.set(graphId, graph); + this.logger.log(`Created lineage graph ${graphId}: ${graph.name}`); + return graph; } - - const node: DataLineageNode = { - id: uuidv4(), - ...nodeConfig, - createdAt: new Date(), - }; - - graph.nodes.push(node); - graph.updatedAt = new Date(); - - this.logger.log(`Added node ${node.id} to graph ${graphId}`); - - return node; - } - - /** - * Add an edge to a lineage graph - */ - async addEdge( - graphId: string, - edgeConfig: Omit, - ): Promise { - const graph = this.graphs.get(graphId); - if (!graph) { - throw new Error(`Graph ${graphId} not found`); + /** + * Get a lineage graph + */ + async getGraph(graphId: string): Promise { + return this.graphs.get(graphId) || null; } - - // Validate that source and target nodes exist - const sourceExists = graph.nodes.some((node) => node.id === edgeConfig.sourceId); - const targetExists = graph.nodes.some((node) => node.id === edgeConfig.targetId); - - if (!sourceExists || !targetExists) { - throw new Error('Source or target node not found in graph'); + /** + * Get all lineage graphs + */ + async getAllGraphs(): Promise { + return Array.from(this.graphs.values()); } - - const edge: DataLineageEdge = { - id: uuidv4(), - ...edgeConfig, - timestamp: new Date(), - }; - - graph.edges.push(edge); - graph.updatedAt = new Date(); - - this.logger.log(`Added edge ${edge.id} to graph ${graphId}`); - - return edge; - } - - /** - * Trace data lineage upstream or downstream - */ - async traceLineage( - graphId: string, - nodeId: string, - traceType: 'upstream' | 'downstream' | 'complete' = 'complete', - ): Promise { - const graph = this.graphs.get(graphId); - if (!graph) { - throw new Error(`Graph ${graphId} not found`); + /** + * Add a node to a lineage graph + */ + async addNode(graphId: string, nodeConfig: Omit): Promise { + const graph = this.graphs.get(graphId); + if (!graph) { + throw new Error(`Graph ${graphId} not found`); + } + const node: DataLineageNode = { + id: uuidv4(), + ...nodeConfig, + createdAt: new Date(), + }; + graph.nodes.push(node); + graph.updatedAt = new Date(); + this.logger.log(`Added node ${node.id} to graph ${graphId}`); + return node; } - - const traceId = uuidv4(); - const path: LineagePath[] = []; - - if (traceType === 'upstream' || traceType === 'complete') { - this.traceUpstream(graph, nodeId, path); + /** + * Add an edge to a lineage graph + */ + async addEdge(graphId: string, edgeConfig: Omit): Promise { + const graph = this.graphs.get(graphId); + if (!graph) { + throw new Error(`Graph ${graphId} not found`); + } + // Validate that source and target nodes exist + const sourceExists = graph.nodes.some((node) => node.id === edgeConfig.sourceId); + const targetExists = graph.nodes.some((node) => node.id === edgeConfig.targetId); + if (!sourceExists || !targetExists) { + throw new Error('Source or target node not found in graph'); + } + const edge: DataLineageEdge = { + id: uuidv4(), + ...edgeConfig, + timestamp: new Date(), + }; + graph.edges.push(edge); + graph.updatedAt = new Date(); + this.logger.log(`Added edge ${edge.id} to graph ${graphId}`); + return edge; + } + /** + * Trace data lineage upstream or downstream + */ + async traceLineage(graphId: string, nodeId: string, traceType: 'upstream' | 'downstream' | 'complete' = 'complete'): Promise { + const graph = this.graphs.get(graphId); + if (!graph) { + throw new Error(`Graph ${graphId} not found`); + } + const traceId = uuidv4(); + const path: LineagePath[] = []; + if (traceType === 'upstream' || traceType === 'complete') { + this.traceUpstream(graph, nodeId, path); + } + if (traceType === 'downstream' || traceType === 'complete') { + this.traceDownstream(graph, nodeId, path); + } + const trace: LineageTrace = { + id: traceId, + nodeId, + traceType, + path, + createdAt: new Date(), + }; + this.traces.set(traceId, trace); + this.logger.log(`Created lineage trace ${traceId} for node ${nodeId}`); + return trace; + } + /** + * Perform impact analysis + */ + async analyzeImpact(graphId: string, nodeId: string): Promise { + const graph = this.graphs.get(graphId); + if (!graph) { + throw new Error(`Graph ${graphId} not found`); + } + const analysisId = uuidv4(); + const affectedNodes = this.findAffectedNodes(graph, nodeId); + const impactLevel = this.calculateImpactLevel(affectedNodes.length, graph.nodes.length); + const analysis: ImpactAnalysis = { + id: analysisId, + nodeId, + affectedNodes, + impactLevel, + analysis: `Node ${nodeId} affects ${affectedNodes.length} other nodes`, + timestamp: new Date(), + }; + this.impactAnalyses.set(analysisId, analysis); + this.logger.log(`Created impact analysis ${analysisId} for node ${nodeId}`); + return analysis; } - - if (traceType === 'downstream' || traceType === 'complete') { - this.traceDownstream(graph, nodeId, path); + /** + * Get lineage trace + */ + async getTrace(traceId: string): Promise { + return this.traces.get(traceId) || null; } - - const trace: LineageTrace = { - id: traceId, - nodeId, - traceType, - path, - createdAt: new Date(), - }; - - this.traces.set(traceId, trace); - this.logger.log(`Created lineage trace ${traceId} for node ${nodeId}`); - - return trace; - } - - /** - * Perform impact analysis - */ - async analyzeImpact(graphId: string, nodeId: string): Promise { - const graph = this.graphs.get(graphId); - if (!graph) { - throw new Error(`Graph ${graphId} not found`); + /** + * Get impact analysis + */ + async getImpactAnalysis(analysisId: string): Promise { + return this.impactAnalyses.get(analysisId) || null; } - - const analysisId = uuidv4(); - const affectedNodes = this.findAffectedNodes(graph, nodeId); - const impactLevel = this.calculateImpactLevel(affectedNodes.length, graph.nodes.length); - - const analysis: ImpactAnalysis = { - id: analysisId, - nodeId, - affectedNodes, - impactLevel, - analysis: `Node ${nodeId} affects ${affectedNodes.length} other nodes`, - timestamp: new Date(), - }; - - this.impactAnalyses.set(analysisId, analysis); - this.logger.log(`Created impact analysis ${analysisId} for node ${nodeId}`); - - return analysis; - } - - /** - * Get lineage trace - */ - async getTrace(traceId: string): Promise { - return this.traces.get(traceId) || null; - } - - /** - * Get impact analysis - */ - async getImpactAnalysis(analysisId: string): Promise { - return this.impactAnalyses.get(analysisId) || null; - } - - /** - * Get all traces for a graph - */ - async getTracesForGraph(graphId: string): Promise { - const traces = Array.from(this.traces.values()); - return traces.filter((trace) => { - const graph = this.graphs.get(graphId); - return graph && graph.nodes.some((node) => node.id === trace.nodeId); - }); - } - - /** - * Get all impact analyses for a graph - */ - async getImpactAnalysesForGraph(graphId: string): Promise { - const analyses = Array.from(this.impactAnalyses.values()); - return analyses.filter((analysis) => { - const graph = this.graphs.get(graphId); - return graph && graph.nodes.some((node) => node.id === analysis.nodeId); - }); - } - - /** - * Create standard lineage for common data flows - */ - async createStandardLineage(): Promise { - const graph = await this.createGraph({ - name: 'Standard Data Flow', - description: 'Standard lineage for user and post data flow', - }); - - // Add source nodes - const userSource = await this.addNode(graph.id, { - name: 'User Source System', - type: 'source', - description: 'Source system containing user data', - system: 'User Service', - metadata: { systemType: 'microservice' }, - }); - - const postSource = await this.addNode(graph.id, { - name: 'Post Source System', - type: 'source', - description: 'Source system containing post data', - system: 'Post Service', - metadata: { systemType: 'microservice' }, - }); - - // Add transformation nodes - const userTransform = await this.addNode(graph.id, { - name: 'User Data Transformation', - type: 'transformation', - description: 'ETL transformation for user data', - system: 'ETL Pipeline', - metadata: { transformationType: 'cleanse_enrich' }, - }); - - const postTransform = await this.addNode(graph.id, { - name: 'Post Data Transformation', - type: 'transformation', - description: 'ETL transformation for post data', - system: 'ETL Pipeline', - metadata: { transformationType: 'cleanse_enrich' }, - }); - - // Add target nodes - const dataWarehouse = await this.addNode(graph.id, { - name: 'Data Warehouse', - type: 'target', - description: 'Central data warehouse', - system: 'Snowflake', - table: 'dim_users', - metadata: { schema: 'analytics' }, - }); - - const postWarehouse = await this.addNode(graph.id, { - name: 'Post Data Warehouse', - type: 'target', - description: 'Post data in warehouse', - system: 'Snowflake', - table: 'fact_posts', - metadata: { schema: 'analytics' }, - }); - - // Add edges - await this.addEdge(graph.id, { - sourceId: userSource.id, - targetId: userTransform.id, - description: 'User data extraction', - transformation: 'extract', - metadata: { frequency: 'hourly' }, - }); - - await this.addEdge(graph.id, { - sourceId: postSource.id, - targetId: postTransform.id, - description: 'Post data extraction', - transformation: 'extract', - metadata: { frequency: 'hourly' }, - }); - - await this.addEdge(graph.id, { - sourceId: userTransform.id, - targetId: dataWarehouse.id, - description: 'Load transformed user data', - transformation: 'load', - metadata: { method: 'incremental' }, - }); - - await this.addEdge(graph.id, { - sourceId: postTransform.id, - targetId: postWarehouse.id, - description: 'Load transformed post data', - transformation: 'load', - metadata: { method: 'incremental' }, - }); - - return graph; - } - - /** - * Search for nodes in lineage graphs - */ - async searchNodes(searchTerm: string, graphId?: string): Promise { - let nodes: DataLineageNode[] = []; - - if (graphId) { - const graph = this.graphs.get(graphId); - if (graph) { - nodes = graph.nodes; - } - } else { - // Search across all graphs - for (const graph of this.graphs.values()) { - nodes.push(...graph.nodes); - } + /** + * Get all traces for a graph + */ + async getTracesForGraph(graphId: string): Promise { + const traces = Array.from(this.traces.values()); + return traces.filter((trace) => { + const graph = this.graphs.get(graphId); + return graph && graph.nodes.some((node) => node.id === trace.nodeId); + }); } - - // Filter by search term - return nodes.filter( - (node) => - node.name.toLowerCase().includes(searchTerm.toLowerCase()) || - node.description.toLowerCase().includes(searchTerm.toLowerCase()) || - node.system.toLowerCase().includes(searchTerm.toLowerCase()), - ); - } - - // Helper methods - private traceUpstream(graph: DataLineageGraph, nodeId: string, path: LineagePath[]): void { - const incomingEdges = graph.edges.filter((edge) => edge.targetId === nodeId); - - for (const edge of incomingEdges) { - path.push({ - fromNode: edge.sourceId, - toNode: edge.targetId, - transformation: edge.transformation || '', - timestamp: edge.timestamp, - }); - - this.traceUpstream(graph, edge.sourceId, path); + /** + * Get all impact analyses for a graph + */ + async getImpactAnalysesForGraph(graphId: string): Promise { + const analyses = Array.from(this.impactAnalyses.values()); + return analyses.filter((analysis) => { + const graph = this.graphs.get(graphId); + return graph && graph.nodes.some((node) => node.id === analysis.nodeId); + }); } - } - - private traceDownstream(graph: DataLineageGraph, nodeId: string, path: LineagePath[]): void { - const outgoingEdges = graph.edges.filter((edge) => edge.sourceId === nodeId); - - for (const edge of outgoingEdges) { - path.push({ - fromNode: edge.sourceId, - toNode: edge.targetId, - transformation: edge.transformation || '', - timestamp: edge.timestamp, - }); - - this.traceDownstream(graph, edge.targetId, path); + /** + * Create standard lineage for common data flows + */ + async createStandardLineage(): Promise { + const graph = await this.createGraph({ + name: 'Standard Data Flow', + description: 'Standard lineage for user and post data flow', + }); + // Add source nodes + const userSource = await this.addNode(graph.id, { + name: 'User Source System', + type: 'source', + description: 'Source system containing user data', + system: 'User Service', + metadata: { systemType: 'microservice' }, + }); + const postSource = await this.addNode(graph.id, { + name: 'Post Source System', + type: 'source', + description: 'Source system containing post data', + system: 'Post Service', + metadata: { systemType: 'microservice' }, + }); + // Add transformation nodes + const userTransform = await this.addNode(graph.id, { + name: 'User Data Transformation', + type: 'transformation', + description: 'ETL transformation for user data', + system: 'ETL Pipeline', + metadata: { transformationType: 'cleanse_enrich' }, + }); + const postTransform = await this.addNode(graph.id, { + name: 'Post Data Transformation', + type: 'transformation', + description: 'ETL transformation for post data', + system: 'ETL Pipeline', + metadata: { transformationType: 'cleanse_enrich' }, + }); + // Add target nodes + const dataWarehouse = await this.addNode(graph.id, { + name: 'Data Warehouse', + type: 'target', + description: 'Central data warehouse', + system: 'Snowflake', + table: 'dim_users', + metadata: { schema: 'analytics' }, + }); + const postWarehouse = await this.addNode(graph.id, { + name: 'Post Data Warehouse', + type: 'target', + description: 'Post data in warehouse', + system: 'Snowflake', + table: 'fact_posts', + metadata: { schema: 'analytics' }, + }); + // Add edges + await this.addEdge(graph.id, { + sourceId: userSource.id, + targetId: userTransform.id, + description: 'User data extraction', + transformation: 'extract', + metadata: { frequency: 'hourly' }, + }); + await this.addEdge(graph.id, { + sourceId: postSource.id, + targetId: postTransform.id, + description: 'Post data extraction', + transformation: 'extract', + metadata: { frequency: 'hourly' }, + }); + await this.addEdge(graph.id, { + sourceId: userTransform.id, + targetId: dataWarehouse.id, + description: 'Load transformed user data', + transformation: 'load', + metadata: { method: 'incremental' }, + }); + await this.addEdge(graph.id, { + sourceId: postTransform.id, + targetId: postWarehouse.id, + description: 'Load transformed post data', + transformation: 'load', + metadata: { method: 'incremental' }, + }); + return graph; } - } - - private findAffectedNodes(graph: DataLineageGraph, nodeId: string): string[] { - const affectedNodes: string[] = []; - const visited = new Set(); - - const traverse = (currentNodeId: string) => { - if (visited.has(currentNodeId)) return; - visited.add(currentNodeId); - - const outgoingEdges = graph.edges.filter((edge) => edge.sourceId === currentNodeId); - for (const edge of outgoingEdges) { - if (!affectedNodes.includes(edge.targetId)) { - affectedNodes.push(edge.targetId); + /** + * Search for nodes in lineage graphs + */ + async searchNodes(searchTerm: string, graphId?: string): Promise { + let nodes: DataLineageNode[] = []; + if (graphId) { + const graph = this.graphs.get(graphId); + if (graph) { + nodes = graph.nodes; + } } - traverse(edge.targetId); - } - }; - - traverse(nodeId); - return affectedNodes; - } - - private calculateImpactLevel( - affectedCount: number, - totalCount: number, - ): 'high' | 'medium' | 'low' { - const percentage = (affectedCount / totalCount) * 100; - - if (percentage >= 50) return 'high'; - if (percentage >= 20) return 'medium'; - return 'low'; - } + else { + // Search across all graphs + for (const graph of this.graphs.values()) { + nodes.push(...graph.nodes); + } + } + // Filter by search term + return nodes.filter((node) => node.name.toLowerCase().includes(searchTerm.toLowerCase()) || + node.description.toLowerCase().includes(searchTerm.toLowerCase()) || + node.system.toLowerCase().includes(searchTerm.toLowerCase())); + } + // Helper methods + private traceUpstream(graph: DataLineageGraph, nodeId: string, path: LineagePath[]): void { + const incomingEdges = graph.edges.filter((edge) => edge.targetId === nodeId); + for (const edge of incomingEdges) { + path.push({ + fromNode: edge.sourceId, + toNode: edge.targetId, + transformation: edge.transformation || '', + timestamp: edge.timestamp, + }); + this.traceUpstream(graph, edge.sourceId, path); + } + } + private traceDownstream(graph: DataLineageGraph, nodeId: string, path: LineagePath[]): void { + const outgoingEdges = graph.edges.filter((edge) => edge.sourceId === nodeId); + for (const edge of outgoingEdges) { + path.push({ + fromNode: edge.sourceId, + toNode: edge.targetId, + transformation: edge.transformation || '', + timestamp: edge.timestamp, + }); + this.traceDownstream(graph, edge.targetId, path); + } + } + private findAffectedNodes(graph: DataLineageGraph, nodeId: string): string[] { + const affectedNodes: string[] = []; + const visited = new Set(); + const traverse = (currentNodeId: string) => { + if (visited.has(currentNodeId)) + return; + visited.add(currentNodeId); + const outgoingEdges = graph.edges.filter((edge) => edge.sourceId === currentNodeId); + for (const edge of outgoingEdges) { + if (!affectedNodes.includes(edge.targetId)) { + affectedNodes.push(edge.targetId); + } + traverse(edge.targetId); + } + }; + traverse(nodeId); + return affectedNodes; + } + private calculateImpactLevel(affectedCount: number, totalCount: number): 'high' | 'medium' | 'low' { + const percentage = (affectedCount / totalCount) * 100; + if (percentage >= 50) + return 'high'; + if (percentage >= 20) + return 'medium'; + return 'low'; + } } diff --git a/src/data-warehouse/loading/incremental-loader.service.ts b/src/data-warehouse/loading/incremental-loader.service.ts index 0039f100..2f8d05cd 100644 --- a/src/data-warehouse/loading/incremental-loader.service.ts +++ b/src/data-warehouse/loading/incremental-loader.service.ts @@ -1,516 +1,388 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface IncrementalLoadJob { - id: string; - name: string; - sourceTable: string; - targetTable: string; - lastProcessedId?: number; - lastProcessedTimestamp?: Date; - status: 'pending' | 'running' | 'completed' | 'failed'; - startTime: Date; - endTime?: Date; - duration?: number; - recordsProcessed: number; - recordsInserted: number; - recordsUpdated: number; - recordsDeleted: number; - config: IncrementalLoadConfig; + id: string; + name: string; + sourceTable: string; + targetTable: string; + lastProcessedId?: number; + lastProcessedTimestamp?: Date; + status: 'pending' | 'running' | 'completed' | 'failed'; + startTime: Date; + endTime?: Date; + duration?: number; + recordsProcessed: number; + recordsInserted: number; + recordsUpdated: number; + recordsDeleted: number; + config: IncrementalLoadConfig; } - export interface IncrementalLoadConfig { - loadType: 'timestamp' | 'sequence' | 'cdc' | 'watermark'; - sourceConnection: DataSourceConfig; - targetConnection: DataSourceConfig; - batchSize: number; - maxRetries: number; - retryDelay: number; - watermarkColumn?: string; - timestampColumn?: string; - sequenceColumn?: string; - primaryKey: string[]; - incrementalColumns: string[]; + loadType: 'timestamp' | 'sequence' | 'cdc' | 'watermark'; + sourceConnection: DataSourceConfig; + targetConnection: DataSourceConfig; + batchSize: number; + maxRetries: number; + retryDelay: number; + watermarkColumn?: string; + timestampColumn?: string; + sequenceColumn?: string; + primaryKey: string[]; + incrementalColumns: string[]; } - export interface DataSourceConfig { - type: 'postgres' | 'mysql' | 'mongodb' | 'snowflake'; - host?: string; - port?: number; - database?: string; - username?: string; - password?: string; - schema?: string; - warehouse?: string; + type: 'postgres' | 'mysql' | 'mongodb' | 'snowflake'; + host?: string; + port?: number; + database?: string; + username?: string; + password?: string; + schema?: string; + warehouse?: string; } - export interface Watermark { - id: string; - tableName: string; - columnName: string; - lastValue: any; - lastUpdated: Date; + id: string; + tableName: string; + columnName: string; + lastValue: unknown; + lastUpdated: Date; } - export interface CDCEvent { - id: string; - tableName: string; - operation: 'INSERT' | 'UPDATE' | 'DELETE'; - primaryKey: { [key: string]: any }; - oldValues?: { [key: string]: any }; - newValues?: { [key: string]: any }; - timestamp: Date; - transactionId?: string; + id: string; + tableName: string; + operation: 'INSERT' | 'UPDATE' | 'DELETE'; + primaryKey: { + [key: string]: unknown; + }; + oldValues?: { + [key: string]: unknown; + }; + newValues?: { + [key: string]: unknown; + }; + timestamp: Date; + transactionId?: string; } - @Injectable() export class IncrementalLoaderService { - private readonly logger = new Logger(IncrementalLoaderService.name); - private jobs: Map = new Map(); - private watermarks: Map = new Map(); - private cdcEvents: Map = new Map(); - - /** - * Create an incremental load job - */ - async createLoadJob( - config: Omit, - sourceConfig: DataSourceConfig, - targetConfig: DataSourceConfig, - ): Promise { - const jobId = uuidv4(); - const jobName = `Incremental_Load_${config.loadType}_${new Date().toISOString()}`; - - const job: IncrementalLoadJob = { - id: jobId, - name: jobName, - sourceTable: '', - targetTable: '', - status: 'pending', - startTime: new Date(), - recordsProcessed: 0, - recordsInserted: 0, - recordsUpdated: 0, - recordsDeleted: 0, - config: { - ...config, - sourceConnection: sourceConfig, - targetConnection: targetConfig, - }, - }; - - this.jobs.set(jobId, job); - this.logger.log(`Created incremental load job ${jobId}: ${jobName}`); - - return job; - } - - /** - * Execute incremental load - */ - async executeLoad( - jobId: string, - sourceTable: string, - targetTable: string, - ): Promise { - const job = this.jobs.get(jobId); - if (!job) { - throw new Error(`Job ${jobId} not found`); + private readonly logger = new Logger(IncrementalLoaderService.name); + private jobs: Map = new Map(); + private watermarks: Map = new Map(); + private cdcEvents: Map = new Map(); + /** + * Create an incremental load job + */ + async createLoadJob(config: Omit, sourceConfig: DataSourceConfig, targetConfig: DataSourceConfig): Promise { + const jobId = uuidv4(); + const jobName = `Incremental_Load_${config.loadType}_${new Date().toISOString()}`; + const job: IncrementalLoadJob = { + id: jobId, + name: jobName, + sourceTable: '', + targetTable: '', + status: 'pending', + startTime: new Date(), + recordsProcessed: 0, + recordsInserted: 0, + recordsUpdated: 0, + recordsDeleted: 0, + config: { + ...config, + sourceConnection: sourceConfig, + targetConnection: targetConfig, + }, + }; + this.jobs.set(jobId, job); + this.logger.log(`Created incremental load job ${jobId}: ${jobName}`); + return job; } - - job.sourceTable = sourceTable; - job.targetTable = targetTable; - job.status = 'running'; - job.startTime = new Date(); - - try { - this.logger.log( - `Starting incremental load for job ${jobId}: ${sourceTable} -> ${targetTable}`, - ); - - let recordsProcessed = 0; - let recordsInserted = 0; - let recordsUpdated = 0; - let recordsDeleted = 0; - - switch (job.config.loadType) { - case 'timestamp': { - const timestampResult = await this.loadByTimestamp(job); - recordsProcessed = timestampResult.processed; - recordsInserted = timestampResult.inserted; - recordsUpdated = timestampResult.updated; - break; + /** + * Execute incremental load + */ + async executeLoad(jobId: string, sourceTable: string, targetTable: string): Promise { + const job = this.jobs.get(jobId); + if (!job) { + throw new Error(`Job ${jobId} not found`); } - - case 'sequence': { - const sequenceResult = await this.loadBySequence(job); - recordsProcessed = sequenceResult.processed; - recordsInserted = sequenceResult.inserted; - recordsUpdated = sequenceResult.updated; - break; + job.sourceTable = sourceTable; + job.targetTable = targetTable; + job.status = 'running'; + job.startTime = new Date(); + try { + this.logger.log(`Starting incremental load for job ${jobId}: ${sourceTable} -> ${targetTable}`); + let recordsProcessed = 0; + let recordsInserted = 0; + let recordsUpdated = 0; + let recordsDeleted = 0; + switch (job.config.loadType) { + case 'timestamp': { + const timestampResult = await this.loadByTimestamp(job); + recordsProcessed = timestampResult.processed; + recordsInserted = timestampResult.inserted; + recordsUpdated = timestampResult.updated; + break; + } + case 'sequence': { + const sequenceResult = await this.loadBySequence(job); + recordsProcessed = sequenceResult.processed; + recordsInserted = sequenceResult.inserted; + recordsUpdated = sequenceResult.updated; + break; + } + case 'watermark': { + const watermarkResult = await this.loadByWatermark(job); + recordsProcessed = watermarkResult.processed; + recordsInserted = watermarkResult.inserted; + recordsUpdated = watermarkResult.updated; + break; + } + case 'cdc': { + const cdcResult = await this.loadByCDC(job); + recordsProcessed = cdcResult.processed; + recordsInserted = cdcResult.inserted; + recordsUpdated = cdcResult.updated; + recordsDeleted = cdcResult.deleted; + break; + } + } + job.status = 'completed'; + job.endTime = new Date(); + job.duration = job.endTime.getTime() - job.startTime.getTime(); + job.recordsProcessed = recordsProcessed; + job.recordsInserted = recordsInserted; + job.recordsUpdated = recordsUpdated; + job.recordsDeleted = recordsDeleted; + this.logger.log(`Incremental load job ${jobId} completed successfully`); + this.logger.log(`Records processed: ${recordsProcessed}, Inserted: ${recordsInserted}, Updated: ${recordsUpdated}, Deleted: ${recordsDeleted}`); } - - case 'watermark': { - const watermarkResult = await this.loadByWatermark(job); - recordsProcessed = watermarkResult.processed; - recordsInserted = watermarkResult.inserted; - recordsUpdated = watermarkResult.updated; - break; + catch (error) { + this.logger.error(`Incremental load job ${jobId} failed: ${error.message}`); + job.status = 'failed'; + job.endTime = new Date(); + job.duration = job.endTime.getTime() - job.startTime.getTime(); } - - case 'cdc': { - const cdcResult = await this.loadByCDC(job); - recordsProcessed = cdcResult.processed; - recordsInserted = cdcResult.inserted; - recordsUpdated = cdcResult.updated; - recordsDeleted = cdcResult.deleted; - break; + return job; + } + /** + * Load data using timestamp-based approach + */ + private async loadByTimestamp(job: IncrementalLoadJob): Promise<{ + processed: number; + inserted: number; + updated: number; + }> { + const timestampColumn = job.config.timestampColumn || 'updated_at'; + const lastTimestamp = job.lastProcessedTimestamp || new Date(0); + this.logger.log(`Loading data newer than ${lastTimestamp.toISOString()}`); + // Get incremental data from source + const incrementalData = await this.getSourceDataSince(job.config.sourceConnection, job.sourceTable, timestampColumn, lastTimestamp); + // Apply changes to target + const result = await this.applyChangesToTarget(job.config.targetConnection, job.targetTable, incrementalData, job.config.primaryKey, 'timestamp'); + // Update last processed timestamp + if (incrementalData.length > 0) { + const maxTimestamp = Math.max(...incrementalData.map((row) => new Date(row[timestampColumn]).getTime())); + job.lastProcessedTimestamp = new Date(maxTimestamp); } - } - - job.status = 'completed'; - job.endTime = new Date(); - job.duration = job.endTime.getTime() - job.startTime.getTime(); - job.recordsProcessed = recordsProcessed; - job.recordsInserted = recordsInserted; - job.recordsUpdated = recordsUpdated; - job.recordsDeleted = recordsDeleted; - - this.logger.log(`Incremental load job ${jobId} completed successfully`); - this.logger.log( - `Records processed: ${recordsProcessed}, Inserted: ${recordsInserted}, Updated: ${recordsUpdated}, Deleted: ${recordsDeleted}`, - ); - } catch (error) { - this.logger.error(`Incremental load job ${jobId} failed: ${error.message}`); - job.status = 'failed'; - job.endTime = new Date(); - job.duration = job.endTime.getTime() - job.startTime.getTime(); + return result; } - - return job; - } - - /** - * Load data using timestamp-based approach - */ - private async loadByTimestamp(job: IncrementalLoadJob): Promise<{ - processed: number; - inserted: number; - updated: number; - }> { - const timestampColumn = job.config.timestampColumn || 'updated_at'; - const lastTimestamp = job.lastProcessedTimestamp || new Date(0); - - this.logger.log(`Loading data newer than ${lastTimestamp.toISOString()}`); - - // Get incremental data from source - const incrementalData = await this.getSourceDataSince( - job.config.sourceConnection, - job.sourceTable, - timestampColumn, - lastTimestamp, - ); - - // Apply changes to target - const result = await this.applyChangesToTarget( - job.config.targetConnection, - job.targetTable, - incrementalData, - job.config.primaryKey, - 'timestamp', - ); - - // Update last processed timestamp - if (incrementalData.length > 0) { - const maxTimestamp = Math.max( - ...incrementalData.map((row) => new Date(row[timestampColumn]).getTime()), - ); - job.lastProcessedTimestamp = new Date(maxTimestamp); + /** + * Load data using sequence-based approach + */ + private async loadBySequence(job: IncrementalLoadJob): Promise<{ + processed: number; + inserted: number; + updated: number; + }> { + const sequenceColumn = job.config.sequenceColumn || 'id'; + const lastId = job.lastProcessedId || 0; + this.logger.log(`Loading data with ${sequenceColumn} > ${lastId}`); + // Get incremental data from source + const incrementalData = await this.getSourceDataAfter(job.config.sourceConnection, job.sourceTable, sequenceColumn, lastId); + // Apply changes to target + const result = await this.applyChangesToTarget(job.config.targetConnection, job.targetTable, incrementalData, job.config.primaryKey, 'sequence'); + // Update last processed ID + if (incrementalData.length > 0) { + const maxId = Math.max(...incrementalData.map((row) => row[sequenceColumn])); + job.lastProcessedId = maxId; + } + return result; } - - return result; - } - - /** - * Load data using sequence-based approach - */ - private async loadBySequence(job: IncrementalLoadJob): Promise<{ - processed: number; - inserted: number; - updated: number; - }> { - const sequenceColumn = job.config.sequenceColumn || 'id'; - const lastId = job.lastProcessedId || 0; - - this.logger.log(`Loading data with ${sequenceColumn} > ${lastId}`); - - // Get incremental data from source - const incrementalData = await this.getSourceDataAfter( - job.config.sourceConnection, - job.sourceTable, - sequenceColumn, - lastId, - ); - - // Apply changes to target - const result = await this.applyChangesToTarget( - job.config.targetConnection, - job.targetTable, - incrementalData, - job.config.primaryKey, - 'sequence', - ); - - // Update last processed ID - if (incrementalData.length > 0) { - const maxId = Math.max(...incrementalData.map((row) => row[sequenceColumn])); - job.lastProcessedId = maxId; + /** + * Load data using watermark approach + */ + private async loadByWatermark(job: IncrementalLoadJob): Promise<{ + processed: number; + inserted: number; + updated: number; + }> { + const watermarkColumn = job.config.watermarkColumn || 'updated_at'; + const watermarkKey = `${job.sourceTable}_${watermarkColumn}`; + const watermark = this.watermarks.get(watermarkKey); + const lastValue = watermark?.lastValue || 0; + this.logger.log(`Loading data with ${watermarkColumn} > ${lastValue}`); + // Get incremental data from source + const incrementalData = await this.getSourceDataAfter(job.config.sourceConnection, job.sourceTable, watermarkColumn, lastValue); + // Apply changes to target + const result = await this.applyChangesToTarget(job.config.targetConnection, job.targetTable, incrementalData, job.config.primaryKey, 'watermark'); + // Update watermark + if (incrementalData.length > 0) { + const maxValue = Math.max(...incrementalData.map((row) => row[watermarkColumn])); + const newWatermark: Watermark = { + id: uuidv4(), + tableName: job.sourceTable, + columnName: watermarkColumn, + lastValue: maxValue, + lastUpdated: new Date(), + }; + this.watermarks.set(watermarkKey, newWatermark); + } + return result; } - - return result; - } - - /** - * Load data using watermark approach - */ - private async loadByWatermark(job: IncrementalLoadJob): Promise<{ - processed: number; - inserted: number; - updated: number; - }> { - const watermarkColumn = job.config.watermarkColumn || 'updated_at'; - const watermarkKey = `${job.sourceTable}_${watermarkColumn}`; - const watermark = this.watermarks.get(watermarkKey); - const lastValue = watermark?.lastValue || 0; - - this.logger.log(`Loading data with ${watermarkColumn} > ${lastValue}`); - - // Get incremental data from source - const incrementalData = await this.getSourceDataAfter( - job.config.sourceConnection, - job.sourceTable, - watermarkColumn, - lastValue, - ); - - // Apply changes to target - const result = await this.applyChangesToTarget( - job.config.targetConnection, - job.targetTable, - incrementalData, - job.config.primaryKey, - 'watermark', - ); - - // Update watermark - if (incrementalData.length > 0) { - const maxValue = Math.max(...incrementalData.map((row) => row[watermarkColumn])); - const newWatermark: Watermark = { - id: uuidv4(), - tableName: job.sourceTable, - columnName: watermarkColumn, - lastValue: maxValue, - lastUpdated: new Date(), - }; - this.watermarks.set(watermarkKey, newWatermark); + /** + * Load data using CDC (Change Data Capture) + */ + private async loadByCDC(job: IncrementalLoadJob): Promise<{ + processed: number; + inserted: number; + updated: number; + deleted: number; + }> { + const cdcKey = `${job.sourceTable}_cdc`; + const events = this.cdcEvents.get(cdcKey) || []; + this.logger.log(`Processing ${events.length} CDC events`); + let inserted = 0; + let updated = 0; + let deleted = 0; + // Process each CDC event + for (const event of events) { + switch (event.operation) { + case 'INSERT': + await this.insertRecord(job.config.targetConnection, job.targetTable, event.newValues); + inserted++; + break; + case 'UPDATE': + await this.updateRecord(job.config.targetConnection, job.targetTable, event.primaryKey, event.newValues); + updated++; + break; + case 'DELETE': + await this.deleteRecord(job.config.targetConnection, job.targetTable, event.primaryKey); + deleted++; + break; + } + } + // Clear processed events + this.cdcEvents.delete(cdcKey); + return { + processed: events.length, + inserted, + updated, + deleted, + }; } - - return result; - } - - /** - * Load data using CDC (Change Data Capture) - */ - private async loadByCDC(job: IncrementalLoadJob): Promise<{ - processed: number; - inserted: number; - updated: number; - deleted: number; - }> { - const cdcKey = `${job.sourceTable}_cdc`; - const events = this.cdcEvents.get(cdcKey) || []; - - this.logger.log(`Processing ${events.length} CDC events`); - - let inserted = 0; - let updated = 0; - let deleted = 0; - - // Process each CDC event - for (const event of events) { - switch (event.operation) { - case 'INSERT': - await this.insertRecord(job.config.targetConnection, job.targetTable, event.newValues); - inserted++; - break; - - case 'UPDATE': - await this.updateRecord( - job.config.targetConnection, - job.targetTable, - event.primaryKey, - event.newValues, - ); - updated++; - break; - - case 'DELETE': - await this.deleteRecord(job.config.targetConnection, job.targetTable, event.primaryKey); - deleted++; - break; - } + /** + * Get job status + */ + async getJobStatus(jobId: string): Promise { + return this.jobs.get(jobId) || null; + } + /** + * Get all jobs + */ + async getAllJobs(): Promise { + return Array.from(this.jobs.values()); + } + /** + * Get watermark for a table and column + */ + async getWatermark(tableName: string, columnName: string): Promise { + const key = `${tableName}_${columnName}`; + return this.watermarks.get(key) || null; + } + /** + * Set watermark manually + */ + async setWatermark(tableName: string, columnName: string, value: unknown): Promise { + const key = `${tableName}_${columnName}`; + const watermark: Watermark = { + id: uuidv4(), + tableName, + columnName, + lastValue: value, + lastUpdated: new Date(), + }; + this.watermarks.set(key, watermark); + this.logger.log(`Set watermark for ${tableName}.${columnName} = ${value}`); + return watermark; + } + /** + * Add CDC event + */ + async addCDCEvent(event: Omit): Promise { + const cdcEvent: CDCEvent = { + id: uuidv4(), + ...event, + timestamp: new Date(), + }; + const key = `${event.tableName}_cdc`; + const events = this.cdcEvents.get(key) || []; + events.push(cdcEvent); + this.cdcEvents.set(key, events); + this.logger.log(`Added CDC event for ${event.tableName}: ${event.operation}`); + return cdcEvent; + } + /** + * Get CDC events for a table + */ + async getCDCEvents(tableName: string): Promise { + const key = `${tableName}_cdc`; + return this.cdcEvents.get(key) || []; + } + // Helper methods for data operations + private async getSourceDataSince(_connection: DataSourceConfig, table: string, column: string, timestamp: Date): Promise { + // Implementation would connect to source database and query + this.logger.log(`Querying ${table} where ${column} > ${timestamp.toISOString()}`); + return []; // Placeholder + } + private async getSourceDataAfter(_connection: DataSourceConfig, table: string, column: string, value: unknown): Promise { + // Implementation would connect to source database and query + this.logger.log(`Querying ${table} where ${column} > ${value}`); + return []; // Placeholder + } + private async applyChangesToTarget(_connection: DataSourceConfig, table: string, data: unknown[], _primaryKey: string[], loadType: string): Promise<{ + processed: number; + inserted: number; + updated: number; + }> { + // Implementation would apply changes to target database + this.logger.log(`Applying ${data.length} records to ${table} using ${loadType} strategy`); + return { + processed: data.length, + inserted: data.length, // Simplified logic + updated: 0, + }; + } + private async insertRecord(_connection: DataSourceConfig, table: string, _values: { + [key: string]: unknown; + }): Promise { + // Implementation would insert record into target + this.logger.log(`Inserting record into ${table}`); + } + private async validateRecord(_values: unknown): Promise { + // Implementation would validate record + return true; + } + private async updateRecord(_connection: DataSourceConfig, table: string, primaryKey: { + [key: string]: unknown; + }, _values: { + [key: string]: unknown; + }): Promise { + // Implementation would update record in target + this.logger.log(`Updating record in ${table} where ${JSON.stringify(primaryKey)}`); + } + private async deleteRecord(_connection: DataSourceConfig, table: string, primaryKey: { + [key: string]: unknown; + }): Promise { + // Implementation would delete record from target + this.logger.log(`Deleting record from ${table} where ${JSON.stringify(primaryKey)}`); } - - // Clear processed events - this.cdcEvents.delete(cdcKey); - - return { - processed: events.length, - inserted, - updated, - deleted, - }; - } - - /** - * Get job status - */ - async getJobStatus(jobId: string): Promise { - return this.jobs.get(jobId) || null; - } - - /** - * Get all jobs - */ - async getAllJobs(): Promise { - return Array.from(this.jobs.values()); - } - - /** - * Get watermark for a table and column - */ - async getWatermark(tableName: string, columnName: string): Promise { - const key = `${tableName}_${columnName}`; - return this.watermarks.get(key) || null; - } - - /** - * Set watermark manually - */ - async setWatermark(tableName: string, columnName: string, value: any): Promise { - const key = `${tableName}_${columnName}`; - const watermark: Watermark = { - id: uuidv4(), - tableName, - columnName, - lastValue: value, - lastUpdated: new Date(), - }; - - this.watermarks.set(key, watermark); - this.logger.log(`Set watermark for ${tableName}.${columnName} = ${value}`); - - return watermark; - } - - /** - * Add CDC event - */ - async addCDCEvent(event: Omit): Promise { - const cdcEvent: CDCEvent = { - id: uuidv4(), - ...event, - timestamp: new Date(), - }; - - const key = `${event.tableName}_cdc`; - const events = this.cdcEvents.get(key) || []; - events.push(cdcEvent); - this.cdcEvents.set(key, events); - - this.logger.log(`Added CDC event for ${event.tableName}: ${event.operation}`); - - return cdcEvent; - } - - /** - * Get CDC events for a table - */ - async getCDCEvents(tableName: string): Promise { - const key = `${tableName}_cdc`; - return this.cdcEvents.get(key) || []; - } - - // Helper methods for data operations - private async getSourceDataSince( - _connection: DataSourceConfig, - table: string, - column: string, - timestamp: Date, - ): Promise { - // Implementation would connect to source database and query - this.logger.log(`Querying ${table} where ${column} > ${timestamp.toISOString()}`); - return []; // Placeholder - } - - private async getSourceDataAfter( - _connection: DataSourceConfig, - table: string, - column: string, - value: any, - ): Promise { - // Implementation would connect to source database and query - this.logger.log(`Querying ${table} where ${column} > ${value}`); - return []; // Placeholder - } - - private async applyChangesToTarget( - _connection: DataSourceConfig, - table: string, - data: any[], - _primaryKey: string[], - loadType: string, - ): Promise<{ processed: number; inserted: number; updated: number }> { - // Implementation would apply changes to target database - this.logger.log(`Applying ${data.length} records to ${table} using ${loadType} strategy`); - - return { - processed: data.length, - inserted: data.length, // Simplified logic - updated: 0, - }; - } - - private async insertRecord( - _connection: DataSourceConfig, - table: string, - _values: { [key: string]: any }, - ): Promise { - // Implementation would insert record into target - this.logger.log(`Inserting record into ${table}`); - } - - private async validateRecord(_values: any): Promise { - // Implementation would validate record - return true; - } - - private async updateRecord( - _connection: DataSourceConfig, - table: string, - primaryKey: { [key: string]: any }, - _values: { [key: string]: any }, - ): Promise { - // Implementation would update record in target - this.logger.log(`Updating record in ${table} where ${JSON.stringify(primaryKey)}`); - } - - private async deleteRecord( - _connection: DataSourceConfig, - table: string, - primaryKey: { [key: string]: any }, - ): Promise { - // Implementation would delete record from target - this.logger.log(`Deleting record from ${table} where ${JSON.stringify(primaryKey)}`); - } } diff --git a/src/data-warehouse/modeling/dimensional-modeling.service.ts b/src/data-warehouse/modeling/dimensional-modeling.service.ts index e3b25a34..c2fd3ca7 100644 --- a/src/data-warehouse/modeling/dimensional-modeling.service.ts +++ b/src/data-warehouse/modeling/dimensional-modeling.service.ts @@ -1,445 +1,371 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface DimensionalModel { - id: string; - name: string; - type: 'star' | 'snowflake' | 'galaxy'; - factTables: FactTable[]; - dimensionTables: DimensionTable[]; - relationships: Relationship[]; - createdAt: Date; - updatedAt: Date; + id: string; + name: string; + type: 'star' | 'snowflake' | 'galaxy'; + factTables: FactTable[]; + dimensionTables: DimensionTable[]; + relationships: Relationship[]; + createdAt: Date; + updatedAt: Date; } - export interface FactTable { - id: string; - name: string; - description: string; - measures: Measure[]; - foreignKeys: ForeignKey[]; - granularity: string; + id: string; + name: string; + description: string; + measures: Measure[]; + foreignKeys: ForeignKey[]; + granularity: string; } - export interface DimensionTable { - id: string; - name: string; - description: string; - attributes: DimensionAttribute[]; - hierarchy?: DimensionHierarchy; - type: 'conformed' | 'degenerate' | 'junk' | 'role-playing'; + id: string; + name: string; + description: string; + attributes: DimensionAttribute[]; + hierarchy?: DimensionHierarchy; + type: 'conformed' | 'degenerate' | 'junk' | 'role-playing'; } - export interface Measure { - id: string; - name: string; - description: string; - dataType: string; - aggregationType: 'sum' | 'count' | 'avg' | 'min' | 'max'; - formula?: string; + id: string; + name: string; + description: string; + dataType: string; + aggregationType: 'sum' | 'count' | 'avg' | 'min' | 'max'; + formula?: string; } - export interface DimensionAttribute { - id: string; - name: string; - description: string; - dataType: string; - isKey: boolean; - isNullable: boolean; + id: string; + name: string; + description: string; + dataType: string; + isKey: boolean; + isNullable: boolean; } - export interface ForeignKey { - id: string; - name: string; - referencedTable: string; - referencedColumn: string; + id: string; + name: string; + referencedTable: string; + referencedColumn: string; } - export interface Relationship { - id: string; - fromTable: string; - toTable: string; - relationshipType: 'one-to-one' | 'one-to-many' | 'many-to-many'; - joinCondition: string; + id: string; + fromTable: string; + toTable: string; + relationshipType: 'one-to-one' | 'one-to-many' | 'many-to-many'; + joinCondition: string; } - export interface DimensionHierarchy { - levels: HierarchyLevel[]; - rollupPaths: string[][]; + levels: HierarchyLevel[]; + rollupPaths: string[][]; } - export interface HierarchyLevel { - id: string; - name: string; - level: number; - attributes: string[]; + id: string; + name: string; + level: number; + attributes: string[]; } - export interface AnalyticsQuery { - id: string; - name: string; - description: string; - modelId: string; - query: string; - parameters: QueryParameter[]; - metrics: string[]; - dimensions: string[]; - filters: QueryFilter[]; + id: string; + name: string; + description: string; + modelId: string; + query: string; + parameters: QueryParameter[]; + metrics: string[]; + dimensions: string[]; + filters: QueryFilter[]; } - export interface QueryParameter { - name: string; - type: string; - defaultValue?: any; - required: boolean; + name: string; + type: string; + defaultValue?: unknown; + required: boolean; } - export interface QueryFilter { - field: string; - operator: string; - value: any; + field: string; + operator: string; + value: unknown; } - @Injectable() export class DimensionalModelingService { - private readonly logger = new Logger(DimensionalModelingService.name); - private models: Map = new Map(); - private queries: Map = new Map(); - - /** - * Create a new dimensional model - */ - async createModel( - modelConfig: Omit, - ): Promise { - const modelId = uuidv4(); - const model: DimensionalModel = { - id: modelId, - ...modelConfig, - createdAt: new Date(), - updatedAt: new Date(), - }; - - this.models.set(modelId, model); - this.logger.log(`Created dimensional model ${modelId}: ${model.name}`); - - return model; - } - - /** - * Get a dimensional model - */ - async getModel(modelId: string): Promise { - return this.models.get(modelId) || null; - } - - /** - * Get all dimensional models - */ - async getAllModels(): Promise { - return Array.from(this.models.values()); - } - - /** - * Update a dimensional model - */ - async updateModel( - modelId: string, - updates: Partial, - ): Promise { - const model = this.models.get(modelId); - if (!model) { - return null; + private readonly logger = new Logger(DimensionalModelingService.name); + private models: Map = new Map(); + private queries: Map = new Map(); + /** + * Create a new dimensional model + */ + async createModel(modelConfig: Omit): Promise { + const modelId = uuidv4(); + const model: DimensionalModel = { + id: modelId, + ...modelConfig, + createdAt: new Date(), + updatedAt: new Date(), + }; + this.models.set(modelId, model); + this.logger.log(`Created dimensional model ${modelId}: ${model.name}`); + return model; } - - const updatedModel = { - ...model, - ...updates, - updatedAt: new Date(), - }; - - this.models.set(modelId, updatedModel); - this.logger.log(`Updated dimensional model ${modelId}`); - - return updatedModel; - } - - /** - * Delete a dimensional model - */ - async deleteModel(modelId: string): Promise { - const exists = this.models.has(modelId); - if (exists) { - this.models.delete(modelId); - this.logger.log(`Deleted dimensional model ${modelId}`); + /** + * Get a dimensional model + */ + async getModel(modelId: string): Promise { + return this.models.get(modelId) || null; } - return exists; - } - - /** - * Create a star schema model - */ - async createStarSchema( - name: string, - factTable: Omit, - dimensionTables: Array>, - ): Promise { - const factTableWithId: FactTable = { - ...factTable, - id: uuidv4(), - }; - - const dimensionTablesWithIds: DimensionTable[] = dimensionTables.map((dim) => ({ - ...dim, - id: uuidv4(), - })); - - // Create foreign keys for each dimension - const foreignKeys: ForeignKey[] = dimensionTablesWithIds.map((dim) => ({ - id: uuidv4(), - name: `${dim.name}_id`, - referencedTable: dim.name, - referencedColumn: 'id', - })); - - factTableWithId.foreignKeys = foreignKeys; - - const model = await this.createModel({ - name, - type: 'star', - factTables: [factTableWithId], - dimensionTables: dimensionTablesWithIds, - relationships: this.createStarRelationships(factTableWithId, dimensionTablesWithIds), - }); - - return model; - } - - /** - * Create a snowflake schema model - */ - async createSnowflakeSchema( - name: string, - factTable: Omit, - dimensionTables: Array>, - subDimensions: { [key: string]: Array> }, - ): Promise { - const factTableWithId: FactTable = { - ...factTable, - id: uuidv4(), - }; - - const dimensionTablesWithIds: DimensionTable[] = dimensionTables.map((dim) => ({ - ...dim, - id: uuidv4(), - })); - - // Process sub-dimensions - const allDimensions: DimensionTable[] = [...dimensionTablesWithIds]; - const relationships: Relationship[] = []; - - for (const [parentDimName, subDims] of Object.entries(subDimensions)) { - const parentDim = dimensionTablesWithIds.find((d) => d.name === parentDimName); - if (parentDim) { - const subDimWithIds = subDims.map((sub) => ({ - ...sub, - id: uuidv4(), + /** + * Get all dimensional models + */ + async getAllModels(): Promise { + return Array.from(this.models.values()); + } + /** + * Update a dimensional model + */ + async updateModel(modelId: string, updates: Partial): Promise { + const model = this.models.get(modelId); + if (!model) { + return null; + } + const updatedModel = { + ...model, + ...updates, + updatedAt: new Date(), + }; + this.models.set(modelId, updatedModel); + this.logger.log(`Updated dimensional model ${modelId}`); + return updatedModel; + } + /** + * Delete a dimensional model + */ + async deleteModel(modelId: string): Promise { + const exists = this.models.has(modelId); + if (exists) { + this.models.delete(modelId); + this.logger.log(`Deleted dimensional model ${modelId}`); + } + return exists; + } + /** + * Create a star schema model + */ + async createStarSchema(name: string, factTable: Omit, dimensionTables: Array>): Promise { + const factTableWithId: FactTable = { + ...factTable, + id: uuidv4(), + }; + const dimensionTablesWithIds: DimensionTable[] = dimensionTables.map((dim) => ({ + ...dim, + id: uuidv4(), })); - - allDimensions.push(...subDimWithIds); - - // Create relationships between parent and sub-dimensions - subDimWithIds.forEach((subDim) => { - relationships.push({ + // Create foreign keys for each dimension + const foreignKeys: ForeignKey[] = dimensionTablesWithIds.map((dim) => ({ id: uuidv4(), - fromTable: parentDim.name, - toTable: subDim.name, - relationshipType: 'one-to-many', - joinCondition: `${parentDim.name}.id = ${subDim.name}.${parentDim.name}_id`, - }); + name: `${dim.name}_id`, + referencedTable: dim.name, + referencedColumn: 'id', + })); + factTableWithId.foreignKeys = foreignKeys; + const model = await this.createModel({ + name, + type: 'star', + factTables: [factTableWithId], + dimensionTables: dimensionTablesWithIds, + relationships: this.createStarRelationships(factTableWithId, dimensionTablesWithIds), + }); + return model; + } + /** + * Create a snowflake schema model + */ + async createSnowflakeSchema(name: string, factTable: Omit, dimensionTables: Array>, subDimensions: { + [key: string]: Array>; + }): Promise { + const factTableWithId: FactTable = { + ...factTable, + id: uuidv4(), + }; + const dimensionTablesWithIds: DimensionTable[] = dimensionTables.map((dim) => ({ + ...dim, + id: uuidv4(), + })); + // Process sub-dimensions + const allDimensions: DimensionTable[] = [...dimensionTablesWithIds]; + const relationships: Relationship[] = []; + for (const [parentDimName, subDims] of Object.entries(subDimensions)) { + const parentDim = dimensionTablesWithIds.find((d) => d.name === parentDimName); + if (parentDim) { + const subDimWithIds = subDims.map((sub) => ({ + ...sub, + id: uuidv4(), + })); + allDimensions.push(...subDimWithIds); + // Create relationships between parent and sub-dimensions + subDimWithIds.forEach((subDim) => { + relationships.push({ + id: uuidv4(), + fromTable: parentDim.name, + toTable: subDim.name, + relationshipType: 'one-to-many', + joinCondition: `${parentDim.name}.id = ${subDim.name}.${parentDim.name}_id`, + }); + }); + } + } + // Create foreign keys for fact table + const foreignKeys: ForeignKey[] = allDimensions.map((dim) => ({ + id: uuidv4(), + name: `${dim.name}_id`, + referencedTable: dim.name, + referencedColumn: 'id', + })); + factTableWithId.foreignKeys = foreignKeys; + const model = await this.createModel({ + name, + type: 'snowflake', + factTables: [factTableWithId], + dimensionTables: allDimensions, + relationships: [ + ...this.createStarRelationships(factTableWithId, allDimensions), + ...relationships, + ], }); - } + return model; } - - // Create foreign keys for fact table - const foreignKeys: ForeignKey[] = allDimensions.map((dim) => ({ - id: uuidv4(), - name: `${dim.name}_id`, - referencedTable: dim.name, - referencedColumn: 'id', - })); - - factTableWithId.foreignKeys = foreignKeys; - - const model = await this.createModel({ - name, - type: 'snowflake', - factTables: [factTableWithId], - dimensionTables: allDimensions, - relationships: [ - ...this.createStarRelationships(factTableWithId, allDimensions), - ...relationships, - ], - }); - - return model; - } - - /** - * Create an analytics query - */ - async createQuery(queryConfig: Omit): Promise { - const queryId = uuidv4(); - const query: AnalyticsQuery = { - id: queryId, - ...queryConfig, - }; - - this.queries.set(queryId, query); - this.logger.log(`Created analytics query ${queryId}: ${query.name}`); - - return query; - } - - /** - * Execute an analytics query - */ - async executeQuery(queryId: string, parameters: { [key: string]: any } = {}): Promise { - const query = this.queries.get(queryId); - if (!query) { - throw new Error(`Query ${queryId} not found`); + /** + * Create an analytics query + */ + async createQuery(queryConfig: Omit): Promise { + const queryId = uuidv4(); + const query: AnalyticsQuery = { + id: queryId, + ...queryConfig, + }; + this.queries.set(queryId, query); + this.logger.log(`Created analytics query ${queryId}: ${query.name}`); + return query; } - - // Validate required parameters - for (const param of query.parameters) { - if (param.required && parameters[param.name] === undefined) { - throw new Error(`Required parameter ${param.name} is missing`); - } + /** + * Execute an analytics query + */ + async executeQuery(queryId: string, parameters: { + [key: string]: unknown; + } = {}): Promise { + const query = this.queries.get(queryId); + if (!query) { + throw new Error(`Query ${queryId} not found`); + } + // Validate required parameters + for (const param of query.parameters) { + if (param.required && parameters[param.name] === undefined) { + throw new Error(`Required parameter ${param.name} is missing`); + } + } + // In a real implementation, this would execute against the data warehouse + this.logger.log(`Executing query ${queryId} with parameters:`, parameters); + // Return mock data for demonstration + return this.generateMockResults(query, parameters); } - - // In a real implementation, this would execute against the data warehouse - this.logger.log(`Executing query ${queryId} with parameters:`, parameters); - - // Return mock data for demonstration - return this.generateMockResults(query, parameters); - } - - /** - * Get query results with pagination - */ - async getQueryResults( - queryId: string, - parameters: { [key: string]: any } = {}, - page: number = 1, - limit: number = 100, - ): Promise<{ data: any[]; total: number; page: number; limit: number }> { - const results = await this.executeQuery(queryId, parameters); - const total = results.length; - const startIndex = (page - 1) * limit; - const endIndex = startIndex + limit; - const paginatedData = results.slice(startIndex, endIndex); - - return { - data: paginatedData, - total, - page, - limit, - }; - } - - /** - * Get all queries for a model - */ - async getQueriesForModel(modelId: string): Promise { - const queries = Array.from(this.queries.values()); - return queries.filter((query) => query.modelId === modelId); - } - - /** - * Validate model integrity - */ - async validateModel(modelId: string): Promise<{ valid: boolean; errors: string[] }> { - const model = this.models.get(modelId); - if (!model) { - return { valid: false, errors: [`Model ${modelId} not found`] }; + /** + * Get query results with pagination + */ + async getQueryResults(queryId: string, parameters: { + [key: string]: unknown; + } = {}, page: number = 1, limit: number = 100): Promise<{ + data: unknown[]; + total: number; + page: number; + limit: number; + }> { + const results = await this.executeQuery(queryId, parameters); + const total = results.length; + const startIndex = (page - 1) * limit; + const endIndex = startIndex + limit; + const paginatedData = results.slice(startIndex, endIndex); + return { + data: paginatedData, + total, + page, + limit, + }; } - - const errors: string[] = []; - - // Check for required components - if (!model.factTables || model.factTables.length === 0) { - errors.push('Model must have at least one fact table'); + /** + * Get all queries for a model + */ + async getQueriesForModel(modelId: string): Promise { + const queries = Array.from(this.queries.values()); + return queries.filter((query) => query.modelId === modelId); } - - if (!model.dimensionTables || model.dimensionTables.length === 0) { - errors.push('Model must have at least one dimension table'); + /** + * Validate model integrity + */ + async validateModel(modelId: string): Promise<{ + valid: boolean; + errors: string[]; + }> { + const model = this.models.get(modelId); + if (!model) { + return { valid: false, errors: [`Model ${modelId} not found`] }; + } + const errors: string[] = []; + // Check for required components + if (!model.factTables || model.factTables.length === 0) { + errors.push('Model must have at least one fact table'); + } + if (!model.dimensionTables || model.dimensionTables.length === 0) { + errors.push('Model must have at least one dimension table'); + } + // Validate relationships + for (const relationship of model.relationships) { + const fromTable = this.findTableByName(model, relationship.fromTable); + const toTable = this.findTableByName(model, relationship.toTable); + if (!fromTable) { + errors.push(`Relationship references non-existent table: ${relationship.fromTable}`); + } + if (!toTable) { + errors.push(`Relationship references non-existent table: ${relationship.toTable}`); + } + } + return { + valid: errors.length === 0, + errors, + }; + } + // Helper methods + private createStarRelationships(factTable: FactTable, dimensionTables: DimensionTable[]): Relationship[] { + return dimensionTables.map((dim) => ({ + id: uuidv4(), + fromTable: factTable.name, + toTable: dim.name, + relationshipType: 'one-to-many', + joinCondition: `${factTable.name}.${dim.name}_id = ${dim.name}.id`, + })); } - - // Validate relationships - for (const relationship of model.relationships) { - const fromTable = this.findTableByName(model, relationship.fromTable); - const toTable = this.findTableByName(model, relationship.toTable); - - if (!fromTable) { - errors.push(`Relationship references non-existent table: ${relationship.fromTable}`); - } - - if (!toTable) { - errors.push(`Relationship references non-existent table: ${relationship.toTable}`); - } + private findTableByName(model: DimensionalModel, tableName: string): FactTable | DimensionTable | undefined { + const factTable = model.factTables.find((ft) => ft.name === tableName); + if (factTable) + return factTable; + return model.dimensionTables.find((dt) => dt.name === tableName); } - - return { - valid: errors.length === 0, - errors, - }; - } - - // Helper methods - private createStarRelationships( - factTable: FactTable, - dimensionTables: DimensionTable[], - ): Relationship[] { - return dimensionTables.map((dim) => ({ - id: uuidv4(), - fromTable: factTable.name, - toTable: dim.name, - relationshipType: 'one-to-many', - joinCondition: `${factTable.name}.${dim.name}_id = ${dim.name}.id`, - })); - } - - private findTableByName( - model: DimensionalModel, - tableName: string, - ): FactTable | DimensionTable | undefined { - const factTable = model.factTables.find((ft) => ft.name === tableName); - if (factTable) return factTable; - - return model.dimensionTables.find((dt) => dt.name === tableName); - } - - private generateMockResults(query: AnalyticsQuery, _parameters: { [key: string]: any }): any[] { - // Generate mock data based on query configuration - const results: any[] = []; - const rowCount = Math.floor(Math.random() * 100) + 10; // 10-110 rows - - for (let i = 0; i < rowCount; i++) { - const row: any = {}; - - // Add metrics - query.metrics.forEach((metric) => { - row[metric] = Math.floor(Math.random() * 10000); - }); - - // Add dimensions - query.dimensions.forEach((dimension) => { - row[dimension] = `Value_${dimension}_${i}`; - }); - - results.push(row); + private generateMockResults(query: AnalyticsQuery, _parameters: { + [key: string]: unknown; + }): unknown[] { + // Generate mock data based on query configuration + const results: unknown[] = []; + const rowCount = Math.floor(Math.random() * 100) + 10; // 10-110 rows + for (let i = 0; i < rowCount; i++) { + const row: unknown = {}; + // Add metrics + query.metrics.forEach((metric) => { + row[metric] = Math.floor(Math.random() * 10000); + }); + // Add dimensions + query.dimensions.forEach((dimension) => { + row[dimension] = `Value_${dimension}_${i}`; + }); + results.push(row); + } + return results; } - - return results; - } } diff --git a/src/data-warehouse/quality/data-quality.service.ts b/src/data-warehouse/quality/data-quality.service.ts index c30ddf2d..662bb1c8 100644 --- a/src/data-warehouse/quality/data-quality.service.ts +++ b/src/data-warehouse/quality/data-quality.service.ts @@ -1,539 +1,462 @@ import { Injectable, Logger } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; - export interface DataQualityProfile { - id: string; - name: string; - description: string; - rules: DataQualityRule[]; - createdAt: Date; - updatedAt: Date; + id: string; + name: string; + description: string; + rules: DataQualityRule[]; + createdAt: Date; + updatedAt: Date; } - export interface DataQualityRule { - id: string; - name: string; - description: string; - type: 'completeness' | 'accuracy' | 'consistency' | 'uniqueness' | 'validity' | 'timeliness'; - field: string; - condition: string; - threshold: number; - severity: 'critical' | 'high' | 'medium' | 'low'; + id: string; + name: string; + description: string; + type: 'completeness' | 'accuracy' | 'consistency' | 'uniqueness' | 'validity' | 'timeliness'; + field: string; + condition: string; + threshold: number; + severity: 'critical' | 'high' | 'medium' | 'low'; } - export interface DataQualityCheck { - id: string; - profileId: string; - startTime: Date; - endTime?: Date; - status: 'pending' | 'running' | 'completed' | 'failed'; - results: DataQualityResult[]; - summary: DataQualitySummary; + id: string; + profileId: string; + startTime: Date; + endTime?: Date; + status: 'pending' | 'running' | 'completed' | 'failed'; + results: DataQualityResult[]; + summary: DataQualitySummary; } - export interface DataQualityResult { - ruleId: string; - ruleName: string; - passed: boolean; - actualValue: number; - expectedValue: number; - message: string; - sampleData?: any[]; + ruleId: string; + ruleName: string; + passed: boolean; + actualValue: number; + expectedValue: number; + message: string; + sampleData?: unknown[]; } - export interface DataQualitySummary { - totalRules: number; - passedRules: number; - failedRules: number; - overallScore: number; - criticalFailures: number; - issuesBySeverity: { - critical: number; - high: number; - medium: number; - low: number; - }; + totalRules: number; + passedRules: number; + failedRules: number; + overallScore: number; + criticalFailures: number; + issuesBySeverity: { + critical: number; + high: number; + medium: number; + low: number; + }; } - export interface DataQualityIssue { - id: string; - profileId: string; - ruleId: string; - severity: string; - description: string; - affectedRecords: number; - sampleRecords: any[]; - createdAt: Date; - resolved: boolean; - resolvedAt?: Date; - resolvedBy?: string; + id: string; + profileId: string; + ruleId: string; + severity: string; + description: string; + affectedRecords: number; + sampleRecords: unknown[]; + createdAt: Date; + resolved: boolean; + resolvedAt?: Date; + resolvedBy?: string; } - @Injectable() export class DataQualityService { - private readonly logger = new Logger(DataQualityService.name); - private profiles: Map = new Map(); - private checks: Map = new Map(); - private issues: Map = new Map(); - - /** - * Create a data quality profile - */ - async createProfile( - profileConfig: Omit, - ): Promise { - const profileId = uuidv4(); - const profile: DataQualityProfile = { - id: profileId, - ...profileConfig, - createdAt: new Date(), - updatedAt: new Date(), - }; - - this.profiles.set(profileId, profile); - this.logger.log(`Created data quality profile ${profileId}: ${profile.name}`); - - return profile; - } - - /** - * Get a data quality profile - */ - async getProfile(profileId: string): Promise { - return this.profiles.get(profileId) || null; - } - - /** - * Get all data quality profiles - */ - async getAllProfiles(): Promise { - return Array.from(this.profiles.values()); - } - - /** - * Update a data quality profile - */ - async updateProfile( - profileId: string, - updates: Partial, - ): Promise { - const profile = this.profiles.get(profileId); - if (!profile) { - return null; + private readonly logger = new Logger(DataQualityService.name); + private profiles: Map = new Map(); + private checks: Map = new Map(); + private issues: Map = new Map(); + /** + * Create a data quality profile + */ + async createProfile(profileConfig: Omit): Promise { + const profileId = uuidv4(); + const profile: DataQualityProfile = { + id: profileId, + ...profileConfig, + createdAt: new Date(), + updatedAt: new Date(), + }; + this.profiles.set(profileId, profile); + this.logger.log(`Created data quality profile ${profileId}: ${profile.name}`); + return profile; } - - const updatedProfile = { - ...profile, - ...updates, - updatedAt: new Date(), - }; - - this.profiles.set(profileId, updatedProfile); - this.logger.log(`Updated data quality profile ${profileId}`); - - return updatedProfile; - } - - /** - * Delete a data quality profile - */ - async deleteProfile(profileId: string): Promise { - const exists = this.profiles.has(profileId); - if (exists) { - this.profiles.delete(profileId); - this.logger.log(`Deleted data quality profile ${profileId}`); + /** + * Get a data quality profile + */ + async getProfile(profileId: string): Promise { + return this.profiles.get(profileId) || null; } - return exists; - } - - /** - * Run data quality checks - */ - async runQualityChecks(profileId: string, data: any[]): Promise { - const profile = this.profiles.get(profileId); - if (!profile) { - throw new Error(`Profile ${profileId} not found`); + /** + * Get all data quality profiles + */ + async getAllProfiles(): Promise { + return Array.from(this.profiles.values()); } - - const checkId = uuidv4(); - const check: DataQualityCheck = { - id: checkId, - profileId, - startTime: new Date(), - status: 'running', - results: [], - summary: { - totalRules: 0, - passedRules: 0, - failedRules: 0, - overallScore: 0, - criticalFailures: 0, - issuesBySeverity: { - critical: 0, - high: 0, - medium: 0, - low: 0, - }, - }, - }; - - this.checks.set(checkId, check); - - try { - // Execute each rule - const results: DataQualityResult[] = []; - let passedRules = 0; - let failedRules = 0; - let criticalFailures = 0; - const issuesBySeverity = { critical: 0, high: 0, medium: 0, low: 0 }; - - for (const rule of profile.rules) { - const result = await this.executeRule(rule, data); - results.push(result); - - if (result.passed) { - passedRules++; - } else { - failedRules++; - - // Count severity issues - issuesBySeverity[rule.severity]++; - - if (rule.severity === 'critical') { - criticalFailures++; - } - - // Create quality issue record - await this.createQualityIssue(profileId, rule, result); + /** + * Update a data quality profile + */ + async updateProfile(profileId: string, updates: Partial): Promise { + const profile = this.profiles.get(profileId); + if (!profile) { + return null; } - } - - // Calculate overall score - const overallScore = - profile.rules.length > 0 ? (passedRules / profile.rules.length) * 100 : 0; - - // Update check with results - check.results = results; - check.status = 'completed'; - check.endTime = new Date(); - check.summary = { - totalRules: profile.rules.length, - passedRules, - failedRules, - overallScore, - criticalFailures, - issuesBySeverity, - }; - - this.logger.log(`Data quality check ${checkId} completed with score: ${overallScore}%`); - } catch (error) { - this.logger.error(`Data quality check ${checkId} failed: ${error.message}`); - check.status = 'failed'; - check.endTime = new Date(); + const updatedProfile = { + ...profile, + ...updates, + updatedAt: new Date(), + }; + this.profiles.set(profileId, updatedProfile); + this.logger.log(`Updated data quality profile ${profileId}`); + return updatedProfile; } - - return check; - } - - /** - * Get quality check results - */ - async getCheckResults(checkId: string): Promise { - return this.checks.get(checkId) || null; - } - - /** - * Get all quality checks for a profile - */ - async getChecksForProfile(profileId: string): Promise { - const checks = Array.from(this.checks.values()); - return checks.filter((check) => check.profileId === profileId); - } - - /** - * Create a quality issue - */ - private async createQualityIssue( - profileId: string, - rule: DataQualityRule, - result: DataQualityResult, - ): Promise { - const issueId = uuidv4(); - const issue: DataQualityIssue = { - id: issueId, - profileId, - ruleId: rule.id, - severity: rule.severity, - description: result.message, - affectedRecords: result.sampleData?.length || 0, - sampleRecords: result.sampleData || [], - createdAt: new Date(), - resolved: false, - }; - - this.issues.set(issueId, issue); - this.logger.log(`Created quality issue ${issueId} for rule ${rule.name}`); - - return issue; - } - - /** - * Get quality issues - */ - async getQualityIssues( - profileId?: string, - severity?: string, - resolved?: boolean, - ): Promise { - let issues = Array.from(this.issues.values()); - - if (profileId) { - issues = issues.filter((issue) => issue.profileId === profileId); + /** + * Delete a data quality profile + */ + async deleteProfile(profileId: string): Promise { + const exists = this.profiles.has(profileId); + if (exists) { + this.profiles.delete(profileId); + this.logger.log(`Deleted data quality profile ${profileId}`); + } + return exists; + } + /** + * Run data quality checks + */ + async runQualityChecks(profileId: string, data: unknown[]): Promise { + const profile = this.profiles.get(profileId); + if (!profile) { + throw new Error(`Profile ${profileId} not found`); + } + const checkId = uuidv4(); + const check: DataQualityCheck = { + id: checkId, + profileId, + startTime: new Date(), + status: 'running', + results: [], + summary: { + totalRules: 0, + passedRules: 0, + failedRules: 0, + overallScore: 0, + criticalFailures: 0, + issuesBySeverity: { + critical: 0, + high: 0, + medium: 0, + low: 0, + }, + }, + }; + this.checks.set(checkId, check); + try { + // Execute each rule + const results: DataQualityResult[] = []; + let passedRules = 0; + let failedRules = 0; + let criticalFailures = 0; + const issuesBySeverity = { critical: 0, high: 0, medium: 0, low: 0 }; + for (const rule of profile.rules) { + const result = await this.executeRule(rule, data); + results.push(result); + if (result.passed) { + passedRules++; + } + else { + failedRules++; + // Count severity issues + issuesBySeverity[rule.severity]++; + if (rule.severity === 'critical') { + criticalFailures++; + } + // Create quality issue record + await this.createQualityIssue(profileId, rule, result); + } + } + // Calculate overall score + const overallScore = profile.rules.length > 0 ? (passedRules / profile.rules.length) * 100 : 0; + // Update check with results + check.results = results; + check.status = 'completed'; + check.endTime = new Date(); + check.summary = { + totalRules: profile.rules.length, + passedRules, + failedRules, + overallScore, + criticalFailures, + issuesBySeverity, + }; + this.logger.log(`Data quality check ${checkId} completed with score: ${overallScore}%`); + } + catch (error) { + this.logger.error(`Data quality check ${checkId} failed: ${error.message}`); + check.status = 'failed'; + check.endTime = new Date(); + } + return check; } - - if (severity) { - issues = issues.filter((issue) => issue.severity === severity); + /** + * Get quality check results + */ + async getCheckResults(checkId: string): Promise { + return this.checks.get(checkId) || null; } - - if (resolved !== undefined) { - issues = issues.filter((issue) => issue.resolved === resolved); + /** + * Get all quality checks for a profile + */ + async getChecksForProfile(profileId: string): Promise { + const checks = Array.from(this.checks.values()); + return checks.filter((check) => check.profileId === profileId); } - - return issues; - } - - /** - * Resolve a quality issue - */ - async resolveIssue(issueId: string, resolvedBy: string): Promise { - const issue = this.issues.get(issueId); - if (!issue || issue.resolved) { - return false; + /** + * Create a quality issue + */ + private async createQualityIssue(profileId: string, rule: DataQualityRule, result: DataQualityResult): Promise { + const issueId = uuidv4(); + const issue: DataQualityIssue = { + id: issueId, + profileId, + ruleId: rule.id, + severity: rule.severity, + description: result.message, + affectedRecords: result.sampleData?.length || 0, + sampleRecords: result.sampleData || [], + createdAt: new Date(), + resolved: false, + }; + this.issues.set(issueId, issue); + this.logger.log(`Created quality issue ${issueId} for rule ${rule.name}`); + return issue; } - - issue.resolved = true; - issue.resolvedAt = new Date(); - issue.resolvedBy = resolvedBy; - - this.logger.log(`Resolved quality issue ${issueId} by ${resolvedBy}`); - return true; - } - - /** - * Create standard quality profiles - */ - async createStandardProfiles(): Promise { - const profiles: DataQualityProfile[] = []; - - // Completeness profile - profiles.push( - await this.createProfile({ - name: 'Completeness Check', - description: 'Check for missing or null values in critical fields', - rules: [ - { - id: uuidv4(), - name: 'User Email Completeness', - description: 'Ensure user email addresses are not null or empty', - type: 'completeness', - field: 'email', - condition: 'not null', - threshold: 99.5, - severity: 'high', - }, - { - id: uuidv4(), - name: 'Post Content Completeness', - description: 'Ensure post content is not empty', - type: 'completeness', - field: 'content', - condition: 'not empty', - threshold: 98, - severity: 'medium', - }, - ], - }), - ); - - // Uniqueness profile - profiles.push( - await this.createProfile({ - name: 'Uniqueness Check', - description: 'Check for duplicate or duplicate-like values', - rules: [ - { - id: uuidv4(), - name: 'User ID Uniqueness', - description: 'Ensure user IDs are unique', - type: 'uniqueness', - field: 'id', - condition: 'unique', - threshold: 100, - severity: 'critical', - }, - { - id: uuidv4(), - name: 'Email Uniqueness', - description: 'Ensure email addresses are unique', - type: 'uniqueness', - field: 'email', - condition: 'unique', - threshold: 99.9, - severity: 'high', - }, - ], - }), - ); - - // Validity profile - profiles.push( - await this.createProfile({ - name: 'Validity Check', - description: 'Check data against business rules and constraints', - rules: [ - { - id: uuidv4(), - name: 'Email Format Validation', - description: 'Validate email format', - type: 'validity', - field: 'email', - condition: 'valid email format', - threshold: 99, - severity: 'high', - }, - { - id: uuidv4(), - name: 'Date Range Validation', - description: 'Validate dates are within reasonable ranges', - type: 'validity', - field: 'created_at', - condition: 'within last 5 years', - threshold: 99.5, - severity: 'medium', - }, - ], - }), - ); - - return profiles; - } - - /** - * Execute a single quality rule - */ - private async executeRule(rule: DataQualityRule, data: any[]): Promise { - let passed = true; - let actualValue = 0; - let sampleData: any[] = []; - let message = ''; - - switch (rule.type) { - case 'completeness': - actualValue = this.calculateCompleteness(data, rule.field); - passed = actualValue >= rule.threshold; - message = `Completeness of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; - if (!passed) { - sampleData = data.filter((item) => !item[rule.field] || item[rule.field] === ''); + /** + * Get quality issues + */ + async getQualityIssues(profileId?: string, severity?: string, resolved?: boolean): Promise { + let issues = Array.from(this.issues.values()); + if (profileId) { + issues = issues.filter((issue) => issue.profileId === profileId); } - break; - - case 'uniqueness': - actualValue = this.calculateUniqueness(data, rule.field); - passed = actualValue >= rule.threshold; - message = `Uniqueness of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; - if (!passed) { - const duplicates = this.findDuplicates(data, rule.field); - sampleData = duplicates.slice(0, 10); // Sample first 10 duplicates + if (severity) { + issues = issues.filter((issue) => issue.severity === severity); } - break; - - case 'validity': - actualValue = this.calculateValidity(data, rule.field, rule.condition); - passed = actualValue >= rule.threshold; - message = `Validity of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; - if (!passed) { - sampleData = data - .filter((item) => !this.isValid(item[rule.field], rule.condition)) - .slice(0, 10); + if (resolved !== undefined) { + issues = issues.filter((issue) => issue.resolved === resolved); } - break; - - default: - actualValue = 100; - message = `Rule type ${rule.type} not implemented`; + return issues; } - - return { - ruleId: rule.id, - ruleName: rule.name, - passed, - actualValue, - expectedValue: rule.threshold, - message, - sampleData, - }; - } - - // Helper methods for quality calculations - private calculateCompleteness(data: any[], field: string): number { - if (data.length === 0) return 100; - const completeCount = data.filter( - (item) => item[field] !== null && item[field] !== undefined && item[field] !== '', - ).length; - return (completeCount / data.length) * 100; - } - - private calculateUniqueness(data: any[], field: string): number { - if (data.length === 0) return 100; - const uniqueValues = new Set(data.map((item) => item[field])); - return (uniqueValues.size / data.length) * 100; - } - - private calculateValidity(data: any[], field: string, condition: string): number { - if (data.length === 0) return 100; - const validCount = data.filter((item) => this.isValid(item[field], condition)).length; - return (validCount / data.length) * 100; - } - - private isValid(value: any, condition: string): boolean { - if (value === null || value === undefined) return false; - - switch (condition) { - case 'not null': - return value !== null && value !== undefined; - case 'not empty': - return value !== '' && value !== null && value !== undefined; - case 'valid email format': - return typeof value === 'string' && /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value); - case 'within last 5 years': { - const date = new Date(value); - const fiveYearsAgo = new Date(); - fiveYearsAgo.setFullYear(fiveYearsAgo.getFullYear() - 5); - return date >= fiveYearsAgo; - } - default: + /** + * Resolve a quality issue + */ + async resolveIssue(issueId: string, resolvedBy: string): Promise { + const issue = this.issues.get(issueId); + if (!issue || issue.resolved) { + return false; + } + issue.resolved = true; + issue.resolvedAt = new Date(); + issue.resolvedBy = resolvedBy; + this.logger.log(`Resolved quality issue ${issueId} by ${resolvedBy}`); return true; } - } - - private findDuplicates(data: any[], field: string): any[] { - const seen = new Map(); - const duplicates: any[] = []; - - data.forEach((item) => { - const value = item[field]; - if (seen.has(value)) { - if (seen.get(value) === 1) { - duplicates.push(seen.get('items')[0]); + /** + * Create standard quality profiles + */ + async createStandardProfiles(): Promise { + const profiles: DataQualityProfile[] = []; + // Completeness profile + profiles.push(await this.createProfile({ + name: 'Completeness Check', + description: 'Check for missing or null values in critical fields', + rules: [ + { + id: uuidv4(), + name: 'User Email Completeness', + description: 'Ensure user email addresses are not null or empty', + type: 'completeness', + field: 'email', + condition: 'not null', + threshold: 99.5, + severity: 'high', + }, + { + id: uuidv4(), + name: 'Post Content Completeness', + description: 'Ensure post content is not empty', + type: 'completeness', + field: 'content', + condition: 'not empty', + threshold: 98, + severity: 'medium', + }, + ], + })); + // Uniqueness profile + profiles.push(await this.createProfile({ + name: 'Uniqueness Check', + description: 'Check for duplicate or duplicate-like values', + rules: [ + { + id: uuidv4(), + name: 'User ID Uniqueness', + description: 'Ensure user IDs are unique', + type: 'uniqueness', + field: 'id', + condition: 'unique', + threshold: 100, + severity: 'critical', + }, + { + id: uuidv4(), + name: 'Email Uniqueness', + description: 'Ensure email addresses are unique', + type: 'uniqueness', + field: 'email', + condition: 'unique', + threshold: 99.9, + severity: 'high', + }, + ], + })); + // Validity profile + profiles.push(await this.createProfile({ + name: 'Validity Check', + description: 'Check data against business rules and constraints', + rules: [ + { + id: uuidv4(), + name: 'Email Format Validation', + description: 'Validate email format', + type: 'validity', + field: 'email', + condition: 'valid email format', + threshold: 99, + severity: 'high', + }, + { + id: uuidv4(), + name: 'Date Range Validation', + description: 'Validate dates are within reasonable ranges', + type: 'validity', + field: 'created_at', + condition: 'within last 5 years', + threshold: 99.5, + severity: 'medium', + }, + ], + })); + return profiles; + } + /** + * Execute a single quality rule + */ + private async executeRule(rule: DataQualityRule, data: unknown[]): Promise { + let passed = true; + let actualValue = 0; + let sampleData: unknown[] = []; + let message = ''; + switch (rule.type) { + case 'completeness': + actualValue = this.calculateCompleteness(data, rule.field); + passed = actualValue >= rule.threshold; + message = `Completeness of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; + if (!passed) { + sampleData = data.filter((item) => !item[rule.field] || item[rule.field] === ''); + } + break; + case 'uniqueness': + actualValue = this.calculateUniqueness(data, rule.field); + passed = actualValue >= rule.threshold; + message = `Uniqueness of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; + if (!passed) { + const duplicates = this.findDuplicates(data, rule.field); + sampleData = duplicates.slice(0, 10); // Sample first 10 duplicates + } + break; + case 'validity': + actualValue = this.calculateValidity(data, rule.field, rule.condition); + passed = actualValue >= rule.threshold; + message = `Validity of ${rule.field}: ${actualValue.toFixed(2)}% (threshold: ${rule.threshold}%)`; + if (!passed) { + sampleData = data + .filter((item) => !this.isValid(item[rule.field], rule.condition)) + .slice(0, 10); + } + break; + default: + actualValue = 100; + message = `Rule type ${rule.type} not implemented`; + } + return { + ruleId: rule.id, + ruleName: rule.name, + passed, + actualValue, + expectedValue: rule.threshold, + message, + sampleData, + }; + } + // Helper methods for quality calculations + private calculateCompleteness(data: unknown[], field: string): number { + if (data.length === 0) + return 100; + const completeCount = data.filter((item) => item[field] !== null && item[field] !== undefined && item[field] !== '').length; + return (completeCount / data.length) * 100; + } + private calculateUniqueness(data: unknown[], field: string): number { + if (data.length === 0) + return 100; + const uniqueValues = new Set(data.map((item) => item[field])); + return (uniqueValues.size / data.length) * 100; + } + private calculateValidity(data: unknown[], field: string, condition: string): number { + if (data.length === 0) + return 100; + const validCount = data.filter((item) => this.isValid(item[field], condition)).length; + return (validCount / data.length) * 100; + } + private isValid(value: unknown, condition: string): boolean { + if (value === null || value === undefined) + return false; + switch (condition) { + case 'not null': + return value !== null && value !== undefined; + case 'not empty': + return value !== '' && value !== null && value !== undefined; + case 'valid email format': + return typeof value === 'string' && /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value); + case 'within last 5 years': { + const date = new Date(value); + const fiveYearsAgo = new Date(); + fiveYearsAgo.setFullYear(fiveYearsAgo.getFullYear() - 5); + return date >= fiveYearsAgo; + } + default: + return true; } - duplicates.push(item); - seen.set(value, (seen.get(value) || 0) + 1); - } else { - seen.set(value, 1); - seen.set('items', [item]); - } - }); - - return duplicates; - } + } + private findDuplicates(data: unknown[], field: string): unknown[] { + const seen = new Map(); + const duplicates: unknown[] = []; + data.forEach((item) => { + const value = item[field]; + if (seen.has(value)) { + if (seen.get(value) === 1) { + duplicates.push(seen.get('items')[0]); + } + duplicates.push(item); + seen.set(value, (seen.get(value) || 0) + 1); + } + else { + seen.set(value, 1); + seen.set('items', [item]); + } + }); + return duplicates; + } } diff --git a/src/email-marketing/ab-testing/ab-testing.controller.ts b/src/email-marketing/ab-testing/ab-testing.controller.ts index 13704c7e..98dd6608 100644 --- a/src/email-marketing/ab-testing/ab-testing.controller.ts +++ b/src/email-marketing/ab-testing/ab-testing.controller.ts @@ -1,56 +1,61 @@ import { Controller, Get, Post, Body, Param, Query, ParseUUIDPipe } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { ABTestingService } from './ab-testing.service'; import { CreateABTestDto } from '../dto/create-ab-test.dto'; import { ABTest } from '../entities/ab-test.entity'; - @ApiTags('Email Marketing - A/B Testing') @ApiBearerAuth() @Controller('email-marketing/ab-tests') export class ABTestingController { - constructor(private readonly abTestingService: ABTestingService) {} - - @Post() - @ApiOperation({ summary: 'Create a new A/B test' }) - @ApiResponse({ status: 201, description: 'A/B test created successfully' }) - async create(@Body() createABTestDto: CreateABTestDto): Promise { - return this.abTestingService.create(createABTestDto); - } - - @Get() - @ApiOperation({ summary: 'Get all A/B tests' }) - @ApiQuery({ name: 'page', required: false, type: Number }) - @ApiQuery({ name: 'limit', required: false, type: Number }) - async findAll(@Query('page') page = 1, @Query('limit') limit = 10) { - return this.abTestingService.findAll(page, limit); - } - - @Get(':id') - @ApiOperation({ summary: 'Get an A/B test by ID' }) - @ApiResponse({ status: 404, description: 'A/B test not found' }) - async findOne(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.abTestingService.findOne(id); - } - - @Post(':id/start') - @ApiOperation({ summary: 'Start an A/B test' }) - async startTest(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.abTestingService.startTest(id); - } - - @Get(':id/results') - @ApiOperation({ summary: 'Get A/B test results with statistical analysis' }) - async getResults(@Param('id', ParseUUIDPipe) id: string) { - return this.abTestingService.getTestResults(id); - } - - @Post(':id/winner/:variantId') - @ApiOperation({ summary: 'Declare a winner for the A/B test' }) - async declareWinner( - @Param('id', ParseUUIDPipe) id: string, - @Param('variantId', ParseUUIDPipe) variantId: string, - ): Promise { - return this.abTestingService.declareWinner(id, variantId); - } + constructor(private readonly abTestingService: ABTestingService) { } + @Post() + @ApiOperation({ summary: 'Create a new A/B test' }) + @ApiResponse({ status: 201, description: 'A/B test created successfully' }) + async create( + @Body() + createABTestDto: CreateABTestDto): Promise { + return this.abTestingService.create(createABTestDto); + } + @Get() + @ApiOperation({ summary: 'Get all A/B tests' }) + @ApiQuery({ name: 'page', required: false, type: Number }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + async findAll( + @Query('page') + page = 1, + @Query('limit') + limit = 10) { + return this.abTestingService.findAll(page, limit); + } + @Get(':id') + @ApiOperation({ summary: 'Get an A/B test by ID' }) + @ApiResponse({ status: 404, description: 'A/B test not found' }) + async findOne( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.abTestingService.findOne(id); + } + @Post(':id/start') + @ApiOperation({ summary: 'Start an A/B test' }) + async startTest( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.abTestingService.startTest(id); + } + @Get(':id/results') + @ApiOperation({ summary: 'Get A/B test results with statistical analysis' }) + async getResults( + @Param('id', ParseUUIDPipe) + id: string) { + return this.abTestingService.getTestResults(id); + } + @Post(':id/winner/:variantId') + @ApiOperation({ summary: 'Declare a winner for the A/B test' }) + async declareWinner( + @Param('id', ParseUUIDPipe) + id: string, + @Param('variantId', ParseUUIDPipe) + variantId: string): Promise { + return this.abTestingService.declareWinner(id, variantId); + } } diff --git a/src/email-marketing/ab-testing/ab-testing.service.ts b/src/email-marketing/ab-testing/ab-testing.service.ts index 10bc7944..f669db1b 100644 --- a/src/email-marketing/ab-testing/ab-testing.service.ts +++ b/src/email-marketing/ab-testing/ab-testing.service.ts @@ -1,262 +1,219 @@ import { Injectable, NotFoundException, BadRequestException } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; - import { ABTest } from '../entities/ab-test.entity'; import { ABTestVariant } from '../entities/ab-test-variant.entity'; import { EmailEvent } from '../entities/email-event.entity'; import { CreateABTestDto } from '../dto/create-ab-test.dto'; import { ABTestStatus } from '../enums/ab-test-status.enum'; import { EmailEventType } from '../enums/email-event-type.enum'; - export interface VariantStats { - variantId: string; - name: string; - sent: number; - opened: number; - clicked: number; - openRate: number; - clickRate: number; - isWinner: boolean; - confidenceLevel: number; + variantId: string; + name: string; + sent: number; + opened: number; + clicked: number; + openRate: number; + clickRate: number; + isWinner: boolean; + confidenceLevel: number; } - @Injectable() export class ABTestingService { - constructor( + constructor( @InjectRepository(ABTest) - private readonly abTestRepository: Repository, + private readonly abTestRepository: Repository, @InjectRepository(ABTestVariant) - private readonly variantRepository: Repository, + private readonly variantRepository: Repository, @InjectRepository(EmailEvent) - private readonly eventRepository: Repository, - ) {} - - /** - * Create a new A/B test - */ - async create(createABTestDto: CreateABTestDto): Promise { - if (createABTestDto.variants.length < 2) { - throw new BadRequestException('A/B test requires at least 2 variants'); + private readonly eventRepository: Repository) { } + /** + * Create a new A/B test + */ + async create(createABTestDto: CreateABTestDto): Promise { + if (createABTestDto.variants.length < 2) { + throw new BadRequestException('A/B test requires at least 2 variants'); + } + const totalWeight = createABTestDto.variants.reduce((sum, v) => sum + (v.weight || 50), 0); + if (totalWeight !== 100) { + throw new BadRequestException('Variant weights must sum to 100'); + } + const abTest = this.abTestRepository.create({ + name: createABTestDto.name, + campaignId: createABTestDto.campaignId, + testField: createABTestDto.testField, + winnerCriteria: createABTestDto.winnerCriteria, + sampleSize: createABTestDto.sampleSize, + status: ABTestStatus.DRAFT, + }); + const savedTest = await this.abTestRepository.save(abTest); + const variants = createABTestDto.variants.map((v, index) => this.variantRepository.create({ + ...v, + abTestId: savedTest.id, + name: v.name || `Variant ${String.fromCharCode(65 + index)}`, + })); + await this.variantRepository.save(variants); + return this.findOne(savedTest.id); } - - const totalWeight = createABTestDto.variants.reduce((sum, v) => sum + (v.weight || 50), 0); - if (totalWeight !== 100) { - throw new BadRequestException('Variant weights must sum to 100'); + /** + * Get all A/B tests + */ + async findAll(page = 1, limit = 10) { + const [tests, total] = await this.abTestRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + relations: ['variants'], + }); + return { tests, total, page, totalPages: Math.ceil(total / limit) }; } - - const abTest = this.abTestRepository.create({ - name: createABTestDto.name, - campaignId: createABTestDto.campaignId, - testField: createABTestDto.testField, - winnerCriteria: createABTestDto.winnerCriteria, - sampleSize: createABTestDto.sampleSize, - status: ABTestStatus.DRAFT, - }); - - const savedTest = await this.abTestRepository.save(abTest); - - const variants = createABTestDto.variants.map((v, index) => - this.variantRepository.create({ - ...v, - abTestId: savedTest.id, - name: v.name || `Variant ${String.fromCharCode(65 + index)}`, - }), - ); - - await this.variantRepository.save(variants); - return this.findOne(savedTest.id); - } - - /** - * Get all A/B tests - */ - async findAll(page = 1, limit = 10) { - const [tests, total] = await this.abTestRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - relations: ['variants'], - }); - - return { tests, total, page, totalPages: Math.ceil(total / limit) }; - } - - /** - * Get a single A/B test by ID - */ - async findOne(id: string): Promise { - const test = await this.abTestRepository.findOne({ - where: { id }, - relations: ['variants'], - }); - - if (!test) { - throw new NotFoundException(`A/B test with ID ${id} not found`); + /** + * Get a single A/B test by ID + */ + async findOne(id: string): Promise { + const test = await this.abTestRepository.findOne({ + where: { id }, + relations: ['variants'], + }); + if (!test) { + throw new NotFoundException(`A/B test with ID ${id} not found`); + } + return test; } - return test; - } - - /** - * Start an A/B test - */ - async startTest(id: string): Promise { - const test = await this.findOne(id); - - if (test.status !== ABTestStatus.DRAFT) { - throw new BadRequestException('Only draft tests can be started'); + /** + * Start an A/B test + */ + async startTest(id: string): Promise { + const test = await this.findOne(id); + if (test.status !== ABTestStatus.DRAFT) { + throw new BadRequestException('Only draft tests can be started'); + } + test.status = ABTestStatus.RUNNING; + test.startedAt = new Date(); + return this.abTestRepository.save(test); } - - test.status = ABTestStatus.RUNNING; - test.startedAt = new Date(); - return this.abTestRepository.save(test); - } - - /** - * Get test results with statistical analysis - */ - async getTestResults(id: string): Promise<{ - test: ABTest; - variants: VariantStats[]; - isSignificant: boolean; - recommendedWinner: string | null; - }> { - const test = await this.findOne(id); - const variantStats: VariantStats[] = []; - - for (const variant of test.variants) { - const sent = variant.recipientCount || 0; - const opened = await this.countVariantEvents(variant.id, EmailEventType.OPENED); - const clicked = await this.countVariantEvents(variant.id, EmailEventType.CLICKED); - - variantStats.push({ - variantId: variant.id, - name: variant.name, - sent, - opened, - clicked, - openRate: sent > 0 ? (opened / sent) * 100 : 0, - clickRate: opened > 0 ? (clicked / opened) * 100 : 0, - isWinner: false, - confidenceLevel: 0, - }); + /** + * Get test results with statistical analysis + */ + async getTestResults(id: string): Promise<{ + test: ABTest; + variants: VariantStats[]; + isSignificant: boolean; + recommendedWinner: string | null; + }> { + const test = await this.findOne(id); + const variantStats: VariantStats[] = []; + for (const variant of test.variants) { + const sent = variant.recipientCount || 0; + const opened = await this.countVariantEvents(variant.id, EmailEventType.OPENED); + const clicked = await this.countVariantEvents(variant.id, EmailEventType.CLICKED); + variantStats.push({ + variantId: variant.id, + name: variant.name, + sent, + opened, + clicked, + openRate: sent > 0 ? (opened / sent) * 100 : 0, + clickRate: opened > 0 ? (clicked / opened) * 100 : 0, + isWinner: false, + confidenceLevel: 0, + }); + } + // Calculate statistical significance + const { isSignificant, winner, confidenceLevel } = this.calculateSignificance(variantStats, test.winnerCriteria); + if (winner) { + const winnerStats = variantStats.find((v) => v.variantId === winner); + if (winnerStats) { + winnerStats.isWinner = true; + winnerStats.confidenceLevel = confidenceLevel; + } + } + return { + test, + variants: variantStats, + isSignificant, + recommendedWinner: winner, + }; } - - // Calculate statistical significance - const { isSignificant, winner, confidenceLevel } = this.calculateSignificance( - variantStats, - test.winnerCriteria, - ); - - if (winner) { - const winnerStats = variantStats.find((v) => v.variantId === winner); - if (winnerStats) { - winnerStats.isWinner = true; - winnerStats.confidenceLevel = confidenceLevel; - } + /** + * Declare a winner and end the test + */ + async declareWinner(testId: string, variantId: string): Promise { + const test = await this.findOne(testId); + if (test.status !== ABTestStatus.RUNNING) { + throw new BadRequestException('Only running tests can have a winner declared'); + } + const variant = test.variants.find((v) => v.id === variantId); + if (!variant) { + throw new BadRequestException('Variant not found in this test'); + } + test.status = ABTestStatus.COMPLETED; + test.winnerId = variantId; + test.endedAt = new Date(); + return this.abTestRepository.save(test); } - - return { - test, - variants: variantStats, - isSignificant, - recommendedWinner: winner, - }; - } - - /** - * Declare a winner and end the test - */ - async declareWinner(testId: string, variantId: string): Promise { - const test = await this.findOne(testId); - - if (test.status !== ABTestStatus.RUNNING) { - throw new BadRequestException('Only running tests can have a winner declared'); + /** + * Select variant for a recipient (weighted random) + */ + selectVariantForRecipient(test: ABTest): ABTestVariant { + const random = Math.random() * 100; + let cumulative = 0; + for (const variant of test.variants) { + cumulative += variant.weight; + if (random <= cumulative) { + return variant; + } + } + return test.variants[0]; } - - const variant = test.variants.find((v) => v.id === variantId); - if (!variant) { - throw new BadRequestException('Variant not found in this test'); + // Private helper methods + private async countVariantEvents(variantId: string, eventType: EmailEventType): Promise { + return this.eventRepository.count({ + where: { metadata: { variantId }, eventType }, + }); } - - test.status = ABTestStatus.COMPLETED; - test.winnerId = variantId; - test.endedAt = new Date(); - - return this.abTestRepository.save(test); - } - - /** - * Select variant for a recipient (weighted random) - */ - selectVariantForRecipient(test: ABTest): ABTestVariant { - const random = Math.random() * 100; - let cumulative = 0; - - for (const variant of test.variants) { - cumulative += variant.weight; - if (random <= cumulative) { - return variant; - } + private calculateSignificance(variants: VariantStats[], criteria: string): { + isSignificant: boolean; + winner: string | null; + confidenceLevel: number; + } { + if (variants.length < 2) { + return { isSignificant: false, winner: null, confidenceLevel: 0 }; + } + // Sort by the winning criteria + const sorted = [...variants].sort((a, b) => { + const metricA = criteria === 'click_rate' ? a.clickRate : a.openRate; + const metricB = criteria === 'click_rate' ? b.clickRate : b.openRate; + return metricB - metricA; + }); + const best = sorted[0]; + const second = sorted[1]; + // Simple z-test approximation + const n1 = best.sent; + const n2 = second.sent; + const p1 = criteria === 'click_rate' ? best.clickRate / 100 : best.openRate / 100; + const p2 = criteria === 'click_rate' ? second.clickRate / 100 : second.openRate / 100; + if (n1 < 30 || n2 < 30) { + return { isSignificant: false, winner: null, confidenceLevel: 0 }; + } + const pooledP = (p1 * n1 + p2 * n2) / (n1 + n2); + const se = Math.sqrt(pooledP * (1 - pooledP) * (1 / n1 + 1 / n2)); + if (se === 0) { + return { isSignificant: false, winner: null, confidenceLevel: 0 }; + } + const zScore = Math.abs(p1 - p2) / se; + // Z-score to confidence level + let confidenceLevel = 0; + if (zScore >= 2.576) + confidenceLevel = 99; + else if (zScore >= 1.96) + confidenceLevel = 95; + else if (zScore >= 1.645) + confidenceLevel = 90; + return { + isSignificant: confidenceLevel >= 95, + winner: confidenceLevel >= 95 ? best.variantId : null, + confidenceLevel, + }; } - - return test.variants[0]; - } - - // Private helper methods - private async countVariantEvents(variantId: string, eventType: EmailEventType): Promise { - return this.eventRepository.count({ - where: { metadata: { variantId }, eventType }, - }); - } - - private calculateSignificance( - variants: VariantStats[], - criteria: string, - ): { isSignificant: boolean; winner: string | null; confidenceLevel: number } { - if (variants.length < 2) { - return { isSignificant: false, winner: null, confidenceLevel: 0 }; - } - - // Sort by the winning criteria - const sorted = [...variants].sort((a, b) => { - const metricA = criteria === 'click_rate' ? a.clickRate : a.openRate; - const metricB = criteria === 'click_rate' ? b.clickRate : b.openRate; - return metricB - metricA; - }); - - const best = sorted[0]; - const second = sorted[1]; - - // Simple z-test approximation - const n1 = best.sent; - const n2 = second.sent; - const p1 = criteria === 'click_rate' ? best.clickRate / 100 : best.openRate / 100; - const p2 = criteria === 'click_rate' ? second.clickRate / 100 : second.openRate / 100; - - if (n1 < 30 || n2 < 30) { - return { isSignificant: false, winner: null, confidenceLevel: 0 }; - } - - const pooledP = (p1 * n1 + p2 * n2) / (n1 + n2); - const se = Math.sqrt(pooledP * (1 - pooledP) * (1 / n1 + 1 / n2)); - - if (se === 0) { - return { isSignificant: false, winner: null, confidenceLevel: 0 }; - } - - const zScore = Math.abs(p1 - p2) / se; - - // Z-score to confidence level - let confidenceLevel = 0; - if (zScore >= 2.576) confidenceLevel = 99; - else if (zScore >= 1.96) confidenceLevel = 95; - else if (zScore >= 1.645) confidenceLevel = 90; - - return { - isSignificant: confidenceLevel >= 95, - winner: confidenceLevel >= 95 ? best.variantId : null, - confidenceLevel, - }; - } } diff --git a/src/email-marketing/analytics/email-analytics.controller.ts b/src/email-marketing/analytics/email-analytics.controller.ts index 97fcf224..8d6aeb2b 100644 --- a/src/email-marketing/analytics/email-analytics.controller.ts +++ b/src/email-marketing/analytics/email-analytics.controller.ts @@ -1,51 +1,49 @@ import { Controller, Get, Param, Query, ParseUUIDPipe } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { EmailAnalyticsService } from './email-analytics.service'; - @ApiTags('Email Marketing - Analytics') @ApiBearerAuth() @Controller('email-marketing/analytics') export class EmailAnalyticsController { - constructor(private readonly analyticsService: EmailAnalyticsService) {} - - @Get('campaigns/:id') - @ApiOperation({ summary: 'Get campaign performance metrics' }) - @ApiResponse({ status: 200, description: 'Campaign metrics' }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async getCampaignMetrics(@Param('id', ParseUUIDPipe) id: string) { - return this.analyticsService.getCampaignMetrics(id); - } - - @Get('campaigns/:id/timeline') - @ApiOperation({ summary: 'Get campaign time series data' }) - @ApiQuery({ name: 'startDate', required: true, type: String }) - @ApiQuery({ name: 'endDate', required: true, type: String }) - async getCampaignTimeline( - @Param('id', ParseUUIDPipe) id: string, - @Query('startDate') startDate: string, - @Query('endDate') endDate: string, - ) { - return this.analyticsService.getCampaignTimeSeries(id, new Date(startDate), new Date(endDate)); - } - - @Get('campaigns/:id/links') - @ApiOperation({ summary: 'Get link click analytics for a campaign' }) - async getLinkAnalytics(@Param('id', ParseUUIDPipe) id: string) { - return this.analyticsService.getLinkAnalytics(id); - } - - @Get('overview') - @ApiOperation({ summary: 'Get overall email marketing statistics' }) - @ApiQuery({ name: 'startDate', required: false, type: String }) - @ApiQuery({ name: 'endDate', required: false, type: String }) - async getOverallStats( - @Query('startDate') startDate?: string, - @Query('endDate') endDate?: string, - ) { - return this.analyticsService.getOverallStats( - startDate ? new Date(startDate) : undefined, - endDate ? new Date(endDate) : undefined, - ); - } + constructor(private readonly analyticsService: EmailAnalyticsService) { } + @Get('campaigns/:id') + @ApiOperation({ summary: 'Get campaign performance metrics' }) + @ApiResponse({ status: 200, description: 'Campaign metrics' }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async getCampaignMetrics( + @Param('id', ParseUUIDPipe) + id: string) { + return this.analyticsService.getCampaignMetrics(id); + } + @Get('campaigns/:id/timeline') + @ApiOperation({ summary: 'Get campaign time series data' }) + @ApiQuery({ name: 'startDate', required: true, type: String }) + @ApiQuery({ name: 'endDate', required: true, type: String }) + async getCampaignTimeline( + @Param('id', ParseUUIDPipe) + id: string, + @Query('startDate') + startDate: string, + @Query('endDate') + endDate: string) { + return this.analyticsService.getCampaignTimeSeries(id, new Date(startDate), new Date(endDate)); + } + @Get('campaigns/:id/links') + @ApiOperation({ summary: 'Get link click analytics for a campaign' }) + async getLinkAnalytics( + @Param('id', ParseUUIDPipe) + id: string) { + return this.analyticsService.getLinkAnalytics(id); + } + @Get('overview') + @ApiOperation({ summary: 'Get overall email marketing statistics' }) + @ApiQuery({ name: 'startDate', required: false, type: String }) + @ApiQuery({ name: 'endDate', required: false, type: String }) + async getOverallStats( + @Query('startDate') + startDate?: string, + @Query('endDate') + endDate?: string) { + return this.analyticsService.getOverallStats(startDate ? new Date(startDate) : undefined, endDate ? new Date(endDate) : undefined); + } } diff --git a/src/email-marketing/analytics/email-analytics.service.ts b/src/email-marketing/analytics/email-analytics.service.ts index e7861092..dd27d4cb 100644 --- a/src/email-marketing/analytics/email-analytics.service.ts +++ b/src/email-marketing/analytics/email-analytics.service.ts @@ -1,324 +1,276 @@ import { Injectable, NotFoundException } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository, Between } from 'typeorm'; - import { EmailEvent } from '../entities/email-event.entity'; import { Campaign } from '../entities/campaign.entity'; import { EmailEventType } from '../enums/email-event-type.enum'; - export interface CampaignMetrics { - sent: number; - delivered: number; - opened: number; - clicked: number; - bounced: number; - unsubscribed: number; - openRate: number; - clickRate: number; - bounceRate: number; + sent: number; + delivered: number; + opened: number; + clicked: number; + bounced: number; + unsubscribed: number; + openRate: number; + clickRate: number; + bounceRate: number; } - export interface TimeSeriesData { - date: string; - opens: number; - clicks: number; - bounces: number; + date: string; + opens: number; + clicks: number; + bounces: number; } - @Injectable() export class EmailAnalyticsService { - constructor( + constructor( @InjectRepository(EmailEvent) - private readonly eventRepository: Repository, + private readonly eventRepository: Repository, @InjectRepository(Campaign) - private readonly campaignRepository: Repository, - ) {} - - /** - * Record an email event - */ - async recordEvent( - campaignId: string, - recipientId: string, - eventType: EmailEventType, - metadata?: Record, - ): Promise { - const event = this.eventRepository.create({ - campaignId, - recipientId, - eventType, - metadata, - occurredAt: new Date(), - }); - - return this.eventRepository.save(event); - } - - /** - * Get campaign metrics - */ - async getCampaignMetrics(campaignId: string): Promise { - const campaign = await this.campaignRepository.findOne({ - where: { id: campaignId }, - }); - - if (!campaign) { - throw new NotFoundException(`Campaign ${campaignId} not found`); + private readonly campaignRepository: Repository) { } + /** + * Record an email event + */ + async recordEvent(campaignId: string, recipientId: string, eventType: EmailEventType, metadata?: Record): Promise { + const event = this.eventRepository.create({ + campaignId, + recipientId, + eventType, + metadata, + occurredAt: new Date(), + }); + return this.eventRepository.save(event); } - - const sent = campaign.totalRecipients || 0; - - const [delivered, opened, clicked, bounced, unsubscribed] = await Promise.all([ - this.countEvents(campaignId, EmailEventType.DELIVERED), - this.countUniqueEvents(campaignId, EmailEventType.OPENED), - this.countUniqueEvents(campaignId, EmailEventType.CLICKED), - this.countEvents(campaignId, EmailEventType.BOUNCED), - this.countEvents(campaignId, EmailEventType.UNSUBSCRIBED), - ]); - - return { - sent, - delivered, - opened, - clicked, - bounced, - unsubscribed, - openRate: sent > 0 ? (opened / sent) * 100 : 0, - clickRate: opened > 0 ? (clicked / opened) * 100 : 0, - bounceRate: sent > 0 ? (bounced / sent) * 100 : 0, - }; - } - - /** - * Get time series data for a campaign - */ - async getCampaignTimeSeries( - campaignId: string, - startDate: Date, - endDate: Date, - ): Promise { - const events = await this.eventRepository.find({ - where: { - campaignId, - occurredAt: Between(startDate, endDate), - }, - order: { occurredAt: 'ASC' }, - }); - - const dataMap = new Map(); - - for (const event of events) { - const dateKey = event.occurredAt.toISOString().split('T')[0]; - - if (!dataMap.has(dateKey)) { - dataMap.set(dateKey, { date: dateKey, opens: 0, clicks: 0, bounces: 0 }); - } - - const data = dataMap.get(dateKey) ?? { opens: 0, clicks: 0, bounces: 0, unsubscribes: 0 }; - - if (event.eventType === EmailEventType.OPENED) data.opens++; - if (event.eventType === EmailEventType.CLICKED) data.clicks++; - if (event.eventType === EmailEventType.BOUNCED) data.bounces++; + /** + * Get campaign metrics + */ + async getCampaignMetrics(campaignId: string): Promise { + const campaign = await this.campaignRepository.findOne({ + where: { id: campaignId }, + }); + if (!campaign) { + throw new NotFoundException(`Campaign ${campaignId} not found`); + } + const sent = campaign.totalRecipients || 0; + const [delivered, opened, clicked, bounced, unsubscribed] = await Promise.all([ + this.countEvents(campaignId, EmailEventType.DELIVERED), + this.countUniqueEvents(campaignId, EmailEventType.OPENED), + this.countUniqueEvents(campaignId, EmailEventType.CLICKED), + this.countEvents(campaignId, EmailEventType.BOUNCED), + this.countEvents(campaignId, EmailEventType.UNSUBSCRIBED), + ]); + return { + sent, + delivered, + opened, + clicked, + bounced, + unsubscribed, + openRate: sent > 0 ? (opened / sent) * 100 : 0, + clickRate: opened > 0 ? (clicked / opened) * 100 : 0, + bounceRate: sent > 0 ? (bounced / sent) * 100 : 0, + }; } - - return Array.from(dataMap.values()); - } - - /** - * Get link click analytics - */ - async getLinkAnalytics(campaignId: string): Promise< - Array<{ - url: string; - clicks: number; - uniqueClicks: number; - }> - > { - const clickEvents = await this.eventRepository.find({ - where: { campaignId, eventType: EmailEventType.CLICKED }, - }); - - const linkMap = new Map }>(); - - for (const event of clickEvents) { - const url = event.metadata?.url || 'unknown'; - - if (!linkMap.has(url)) { - linkMap.set(url, { clicks: 0, recipients: new Set() }); - } - - const data = linkMap.get(url) ?? { url, clicks: 0, recipients: new Set() }; - data.clicks++; - data.recipients.add(event.recipientId); + /** + * Get time series data for a campaign + */ + async getCampaignTimeSeries(campaignId: string, startDate: Date, endDate: Date): Promise { + const events = await this.eventRepository.find({ + where: { + campaignId, + occurredAt: Between(startDate, endDate), + }, + order: { occurredAt: 'ASC' }, + }); + const dataMap = new Map(); + for (const event of events) { + const dateKey = event.occurredAt.toISOString().split('T')[0]; + if (!dataMap.has(dateKey)) { + dataMap.set(dateKey, { date: dateKey, opens: 0, clicks: 0, bounces: 0 }); + } + const data = dataMap.get(dateKey) ?? { opens: 0, clicks: 0, bounces: 0, unsubscribes: 0 }; + if (event.eventType === EmailEventType.OPENED) + data.opens++; + if (event.eventType === EmailEventType.CLICKED) + data.clicks++; + if (event.eventType === EmailEventType.BOUNCED) + data.bounces++; + } + return Array.from(dataMap.values()); } - - return Array.from(linkMap.entries()).map(([url, data]) => ({ - url, - clicks: data.clicks, - uniqueClicks: data.recipients.size, - })); - } - - /** - * Get overall email marketing stats - */ - async getOverallStats( - startDate?: Date, - endDate?: Date, - ): Promise<{ - totalCampaigns: number; - totalEmailsSent: number; - averageOpenRate: number; - averageClickRate: number; - }> { - const query = this.campaignRepository - .createQueryBuilder('campaign') - .where('campaign.sentAt IS NOT NULL'); - - if (startDate && endDate) { - query.andWhere('campaign.sentAt BETWEEN :start AND :end', { - start: startDate, - end: endDate, - }); + /** + * Get link click analytics + */ + async getLinkAnalytics(campaignId: string): Promise> { + const clickEvents = await this.eventRepository.find({ + where: { campaignId, eventType: EmailEventType.CLICKED }, + }); + const linkMap = new Map; + }>(); + for (const event of clickEvents) { + const url = event.metadata?.url || 'unknown'; + if (!linkMap.has(url)) { + linkMap.set(url, { clicks: 0, recipients: new Set() }); + } + const data = linkMap.get(url) ?? { url, clicks: 0, recipients: new Set() }; + data.clicks++; + data.recipients.add(event.recipientId); + } + return Array.from(linkMap.entries()).map(([url, data]) => ({ + url, + clicks: data.clicks, + uniqueClicks: data.recipients.size, + })); } - - const campaigns = await query.getMany(); - - const totalCampaigns = campaigns.length; - const totalEmailsSent = campaigns.reduce((sum, c) => sum + (c.totalRecipients || 0), 0); - - const campaignIds = campaigns.map((c) => c.id); - const metricsMap = await this.getBatchMetrics(campaignIds); - - let totalOpenRate = 0; - let totalClickRate = 0; - - for (const campaign of campaigns) { - const metrics = metricsMap.get(campaign.id) || { - sent: campaign.totalRecipients || 0, - delivered: 0, - opened: 0, - clicked: 0, - bounced: 0, - unsubscribed: 0, - openRate: 0, - clickRate: 0, - bounceRate: 0, - }; - totalOpenRate += metrics.openRate; - totalClickRate += metrics.clickRate; + /** + * Get overall email marketing stats + */ + async getOverallStats(startDate?: Date, endDate?: Date): Promise<{ + totalCampaigns: number; + totalEmailsSent: number; + averageOpenRate: number; + averageClickRate: number; + }> { + const query = this.campaignRepository + .createQueryBuilder('campaign') + .where('campaign.sentAt IS NOT NULL'); + if (startDate && endDate) { + query.andWhere('campaign.sentAt BETWEEN :start AND :end', { + start: startDate, + end: endDate, + }); + } + const campaigns = await query.getMany(); + const totalCampaigns = campaigns.length; + const totalEmailsSent = campaigns.reduce((sum, c) => sum + (c.totalRecipients || 0), 0); + const campaignIds = campaigns.map((c) => c.id); + const metricsMap = await this.getBatchMetrics(campaignIds); + let totalOpenRate = 0; + let totalClickRate = 0; + for (const campaign of campaigns) { + const metrics = metricsMap.get(campaign.id) || { + sent: campaign.totalRecipients || 0, + delivered: 0, + opened: 0, + clicked: 0, + bounced: 0, + unsubscribed: 0, + openRate: 0, + clickRate: 0, + bounceRate: 0, + }; + totalOpenRate += metrics.openRate; + totalClickRate += metrics.clickRate; + } + return { + totalCampaigns, + totalEmailsSent, + averageOpenRate: totalCampaigns > 0 ? totalOpenRate / totalCampaigns : 0, + averageClickRate: totalCampaigns > 0 ? totalClickRate / totalCampaigns : 0, + }; } - - return { - totalCampaigns, - totalEmailsSent, - averageOpenRate: totalCampaigns > 0 ? totalOpenRate / totalCampaigns : 0, - averageClickRate: totalCampaigns > 0 ? totalClickRate / totalCampaigns : 0, - }; - } - - /** - * Batch fetch metrics for multiple campaigns to prevent N+1 queries - */ - private async getBatchMetrics(campaignIds: string[]): Promise> { - if (!campaignIds.length) return new Map(); - - // Query 1: Regular counts (delivered, bounced, unsubscribed) - const regularCounts = await this.eventRepository - .createQueryBuilder('event') - .select('event.campaignId', 'campaignId') - .addSelect('event.eventType', 'eventType') - .addSelect('COUNT(*)', 'count') - .where('event.campaignId IN (:...campaignIds)', { campaignIds }) - .andWhere('event.eventType IN (:...eventTypes)', { - eventTypes: [EmailEventType.DELIVERED, EmailEventType.BOUNCED, EmailEventType.UNSUBSCRIBED], - }) - .groupBy('event.campaignId') - .addGroupBy('event.eventType') - .getRawMany(); - - // Query 2: Unique recipient counts (opened, clicked) - const uniqueCounts = await this.eventRepository - .createQueryBuilder('event') - .select('event.campaignId', 'campaignId') - .addSelect('event.eventType', 'eventType') - .addSelect('COUNT(DISTINCT event.recipientId)', 'count') - .where('event.campaignId IN (:...campaignIds)', { campaignIds }) - .andWhere('event.eventType IN (:...eventTypes)', { - eventTypes: [EmailEventType.OPENED, EmailEventType.CLICKED], - }) - .groupBy('event.campaignId') - .addGroupBy('event.eventType') - .getRawMany(); - - // Fetch campaigns to get totalRecipients - const campaigns = await this.campaignRepository.findByIds(campaignIds); - const campaignMap = new Map(campaigns.map((c) => [c.id, c])); - - const metricsMap = new Map(); - - const getMetrics = (id: string): CampaignMetrics => { - let metrics = metricsMap.get(id); - if (!metrics) { - const campaign = campaignMap.get(id); - const sent = campaign?.totalRecipients || 0; - metrics = { - sent, - delivered: 0, - opened: 0, - clicked: 0, - bounced: 0, - unsubscribed: 0, - openRate: 0, - clickRate: 0, - bounceRate: 0, + /** + * Batch fetch metrics for multiple campaigns to prevent N+1 queries + */ + private async getBatchMetrics(campaignIds: string[]): Promise> { + if (!campaignIds.length) + return new Map(); + // Query 1: Regular counts (delivered, bounced, unsubscribed) + const regularCounts = await this.eventRepository + .createQueryBuilder('event') + .select('event.campaignId', 'campaignId') + .addSelect('event.eventType', 'eventType') + .addSelect('COUNT(*)', 'count') + .where('event.campaignId IN (:...campaignIds)', { campaignIds }) + .andWhere('event.eventType IN (:...eventTypes)', { + eventTypes: [EmailEventType.DELIVERED, EmailEventType.BOUNCED, EmailEventType.UNSUBSCRIBED], + }) + .groupBy('event.campaignId') + .addGroupBy('event.eventType') + .getRawMany(); + // Query 2: Unique recipient counts (opened, clicked) + const uniqueCounts = await this.eventRepository + .createQueryBuilder('event') + .select('event.campaignId', 'campaignId') + .addSelect('event.eventType', 'eventType') + .addSelect('COUNT(DISTINCT event.recipientId)', 'count') + .where('event.campaignId IN (:...campaignIds)', { campaignIds }) + .andWhere('event.eventType IN (:...eventTypes)', { + eventTypes: [EmailEventType.OPENED, EmailEventType.CLICKED], + }) + .groupBy('event.campaignId') + .addGroupBy('event.eventType') + .getRawMany(); + // Fetch campaigns to get totalRecipients + const campaigns = await this.campaignRepository.findByIds(campaignIds); + const campaignMap = new Map(campaigns.map((c) => [c.id, c])); + const metricsMap = new Map(); + const getMetrics = (id: string): CampaignMetrics => { + let metrics = metricsMap.get(id); + if (!metrics) { + const campaign = campaignMap.get(id); + const sent = campaign?.totalRecipients || 0; + metrics = { + sent, + delivered: 0, + opened: 0, + clicked: 0, + bounced: 0, + unsubscribed: 0, + openRate: 0, + clickRate: 0, + bounceRate: 0, + }; + metricsMap.set(id, metrics); + } + return metrics; }; - metricsMap.set(id, metrics); - } - return metrics; - }; - - // Process regular counts - for (const row of regularCounts) { - const metrics = getMetrics(row.campaignId); - const count = parseInt(row.count, 10); - if (row.eventType === EmailEventType.DELIVERED) metrics.delivered = count; - if (row.eventType === EmailEventType.BOUNCED) metrics.bounced = count; - if (row.eventType === EmailEventType.UNSUBSCRIBED) metrics.unsubscribed = count; + // Process regular counts + for (const row of regularCounts) { + const metrics = getMetrics(row.campaignId); + const count = parseInt(row.count, 10); + if (row.eventType === EmailEventType.DELIVERED) + metrics.delivered = count; + if (row.eventType === EmailEventType.BOUNCED) + metrics.bounced = count; + if (row.eventType === EmailEventType.UNSUBSCRIBED) + metrics.unsubscribed = count; + } + // Process unique counts + for (const row of uniqueCounts) { + const metrics = getMetrics(row.campaignId); + const count = parseInt(row.count, 10); + if (row.eventType === EmailEventType.OPENED) + metrics.opened = count; + if (row.eventType === EmailEventType.CLICKED) + metrics.clicked = count; + } + // Calculate rates + for (const metrics of metricsMap.values()) { + const sent = metrics.sent; + metrics.openRate = sent > 0 ? (metrics.opened / sent) * 100 : 0; + metrics.clickRate = metrics.opened > 0 ? (metrics.clicked / metrics.opened) * 100 : 0; + metrics.bounceRate = sent > 0 ? (metrics.bounced / sent) * 100 : 0; + } + return metricsMap; } - - // Process unique counts - for (const row of uniqueCounts) { - const metrics = getMetrics(row.campaignId); - const count = parseInt(row.count, 10); - if (row.eventType === EmailEventType.OPENED) metrics.opened = count; - if (row.eventType === EmailEventType.CLICKED) metrics.clicked = count; + // Helper methods + private async countEvents(campaignId: string, eventType: EmailEventType): Promise { + return this.eventRepository.count({ where: { campaignId, eventType } }); } - - // Calculate rates - for (const metrics of metricsMap.values()) { - const sent = metrics.sent; - metrics.openRate = sent > 0 ? (metrics.opened / sent) * 100 : 0; - metrics.clickRate = metrics.opened > 0 ? (metrics.clicked / metrics.opened) * 100 : 0; - metrics.bounceRate = sent > 0 ? (metrics.bounced / sent) * 100 : 0; + private async countUniqueEvents(campaignId: string, eventType: EmailEventType): Promise { + const result = await this.eventRepository + .createQueryBuilder('event') + .select('COUNT(DISTINCT event.recipientId)', 'count') + .where('event.campaignId = :campaignId', { campaignId }) + .andWhere('event.eventType = :eventType', { eventType }) + .getRawOne(); + return parseInt(result?.count || '0', 10); } - - return metricsMap; - } - - // Helper methods - private async countEvents(campaignId: string, eventType: EmailEventType): Promise { - return this.eventRepository.count({ where: { campaignId, eventType } }); - } - - private async countUniqueEvents(campaignId: string, eventType: EmailEventType): Promise { - const result = await this.eventRepository - .createQueryBuilder('event') - .select('COUNT(DISTINCT event.recipientId)', 'count') - .where('event.campaignId = :campaignId', { campaignId }) - .andWhere('event.eventType = :eventType', { eventType }) - .getRawOne(); - - return parseInt(result?.count || '0', 10); - } } diff --git a/src/email-marketing/automation/automation.controller.ts b/src/email-marketing/automation/automation.controller.ts index 2519dd7d..81c338f6 100644 --- a/src/email-marketing/automation/automation.controller.ts +++ b/src/email-marketing/automation/automation.controller.ts @@ -1,96 +1,91 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Body, - Param, - Query, - ParseUUIDPipe, - HttpCode, - HttpStatus, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Body, Param, Query, ParseUUIDPipe, HttpCode, HttpStatus, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { AutomationService } from './automation.service'; import { CreateAutomationDto } from '../dto/create-automation.dto'; import { UpdateAutomationDto } from '../dto/update-automation.dto'; import { AutomationWorkflow } from '../entities/automation-workflow.entity'; - @ApiTags('Email Marketing - Automation') @ApiBearerAuth() @Controller('email-marketing/automation') export class AutomationController { - constructor(private readonly automationService: AutomationService) {} - - @Post() - @ApiOperation({ summary: 'Create a new automation workflow' }) - @ApiResponse({ status: 201, description: 'Workflow created successfully' }) - @ApiResponse({ status: 400, description: 'Invalid input data' }) - async create(@Body() createAutomationDto: CreateAutomationDto): Promise { - return this.automationService.create(createAutomationDto); - } - - @Get() - @ApiOperation({ summary: 'Get all automation workflows' }) - @ApiQuery({ name: 'page', required: false, type: Number }) - @ApiQuery({ name: 'limit', required: false, type: Number }) - @ApiResponse({ status: 200, description: 'List of automation workflows' }) - async findAll(@Query('page') page: number = 1, @Query('limit') limit: number = 10) { - return this.automationService.findAll(page, limit); - } - - @Get(':id') - @ApiOperation({ summary: 'Get an automation workflow by ID' }) - @ApiResponse({ status: 200, description: 'Workflow details' }) - @ApiResponse({ status: 404, description: 'Workflow not found' }) - async findOne(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.automationService.findOne(id); - } - - @Put(':id') - @ApiOperation({ summary: 'Update an automation workflow' }) - @ApiResponse({ status: 200, description: 'Workflow updated successfully' }) - @ApiResponse({ status: 400, description: 'Cannot update active workflow' }) - @ApiResponse({ status: 404, description: 'Workflow not found' }) - async update( - @Param('id', ParseUUIDPipe) id: string, - @Body() updateAutomationDto: UpdateAutomationDto, - ): Promise { - return this.automationService.update(id, updateAutomationDto); - } - - @Delete(':id') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete an automation workflow' }) - @ApiResponse({ status: 204, description: 'Workflow deleted successfully' }) - @ApiResponse({ status: 400, description: 'Cannot delete active workflow' }) - @ApiResponse({ status: 404, description: 'Workflow not found' }) - async remove(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.automationService.remove(id); - } - - @Post(':id/activate') - @ApiOperation({ summary: 'Activate an automation workflow' }) - @ApiResponse({ status: 200, description: 'Workflow activated' }) - @ApiResponse({ status: 400, description: 'Workflow has no triggers or actions' }) - async activate(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.automationService.activate(id); - } - - @Post(':id/deactivate') - @ApiOperation({ summary: 'Deactivate an automation workflow' }) - @ApiResponse({ status: 200, description: 'Workflow deactivated' }) - async deactivate(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.automationService.deactivate(id); - } - - @Get(':id/stats') - @ApiOperation({ summary: 'Get workflow execution statistics' }) - @ApiResponse({ status: 200, description: 'Workflow statistics' }) - @ApiResponse({ status: 404, description: 'Workflow not found' }) - async getStats(@Param('id', ParseUUIDPipe) id: string) { - return this.automationService.getWorkflowStats(id); - } + constructor(private readonly automationService: AutomationService) { } + @Post() + @ApiOperation({ summary: 'Create a new automation workflow' }) + @ApiResponse({ status: 201, description: 'Workflow created successfully' }) + @ApiResponse({ status: 400, description: 'Invalid input data' }) + async create( + @Body() + createAutomationDto: CreateAutomationDto): Promise { + return this.automationService.create(createAutomationDto); + } + @Get() + @ApiOperation({ summary: 'Get all automation workflows' }) + @ApiQuery({ name: 'page', required: false, type: Number }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiResponse({ status: 200, description: 'List of automation workflows' }) + async findAll( + @Query('page') + page: number = 1, + @Query('limit') + limit: number = 10) { + return this.automationService.findAll(page, limit); + } + @Get(':id') + @ApiOperation({ summary: 'Get an automation workflow by ID' }) + @ApiResponse({ status: 200, description: 'Workflow details' }) + @ApiResponse({ status: 404, description: 'Workflow not found' }) + async findOne( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.automationService.findOne(id); + } + @Put(':id') + @ApiOperation({ summary: 'Update an automation workflow' }) + @ApiResponse({ status: 200, description: 'Workflow updated successfully' }) + @ApiResponse({ status: 400, description: 'Cannot update active workflow' }) + @ApiResponse({ status: 404, description: 'Workflow not found' }) + async update( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + updateAutomationDto: UpdateAutomationDto): Promise { + return this.automationService.update(id, updateAutomationDto); + } + @Delete(':id') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete an automation workflow' }) + @ApiResponse({ status: 204, description: 'Workflow deleted successfully' }) + @ApiResponse({ status: 400, description: 'Cannot delete active workflow' }) + @ApiResponse({ status: 404, description: 'Workflow not found' }) + async remove( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.automationService.remove(id); + } + @Post(':id/activate') + @ApiOperation({ summary: 'Activate an automation workflow' }) + @ApiResponse({ status: 200, description: 'Workflow activated' }) + @ApiResponse({ status: 400, description: 'Workflow has no triggers or actions' }) + async activate( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.automationService.activate(id); + } + @Post(':id/deactivate') + @ApiOperation({ summary: 'Deactivate an automation workflow' }) + @ApiResponse({ status: 200, description: 'Workflow deactivated' }) + async deactivate( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.automationService.deactivate(id); + } + @Get(':id/stats') + @ApiOperation({ summary: 'Get workflow execution statistics' }) + @ApiResponse({ status: 200, description: 'Workflow statistics' }) + @ApiResponse({ status: 404, description: 'Workflow not found' }) + async getStats( + @Param('id', ParseUUIDPipe) + id: string) { + return this.automationService.getWorkflowStats(id); + } } diff --git a/src/email-marketing/automation/automation.service.ts b/src/email-marketing/automation/automation.service.ts index b4b5f76a..21ac505a 100644 --- a/src/email-marketing/automation/automation.service.ts +++ b/src/email-marketing/automation/automation.service.ts @@ -6,391 +6,325 @@ import { Queue } from 'bull'; import { EventEmitter2, OnEvent } from '@nestjs/event-emitter'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; import { APP_EVENTS } from '../../common/constants/event.constants'; - import { AutomationWorkflow } from '../entities/automation-workflow.entity'; import { AutomationTrigger } from '../entities/automation-trigger.entity'; import { AutomationAction } from '../entities/automation-action.entity'; - import { CreateAutomationDto } from '../dto/create-automation.dto'; import { UpdateAutomationDto } from '../dto/update-automation.dto'; import { TriggerType } from '../enums/trigger-type.enum'; import { ActionType } from '../enums/action-type.enum'; import { WorkflowStatus } from '../enums/workflow-status.enum'; - @Injectable() export class AutomationService { - constructor( + constructor( @InjectRepository(AutomationWorkflow) - private readonly workflowRepository: Repository, + private readonly workflowRepository: Repository, @InjectRepository(AutomationTrigger) - private readonly triggerRepository: Repository, + private readonly triggerRepository: Repository, @InjectRepository(AutomationAction) - private readonly actionRepository: Repository, + private readonly actionRepository: Repository, @InjectQueue(QUEUE_NAMES.EMAIL_MARKETING) - private readonly emailQueue: Queue, - private readonly eventEmitter: EventEmitter2, - ) {} - - /** - * Create a new automation workflow - */ - async create(createAutomationDto: CreateAutomationDto): Promise { - const workflow = this.workflowRepository.create({ - name: createAutomationDto.name, - description: createAutomationDto.description, - status: WorkflowStatus.DRAFT, - }); - - const savedWorkflow = await this.workflowRepository.save(workflow); - - // Create triggers - if (createAutomationDto.triggers?.length) { - const triggers = createAutomationDto.triggers.map((trigger) => - this.triggerRepository.create({ - ...trigger, - workflowId: savedWorkflow.id, - }), - ); - await this.triggerRepository.save(triggers); - } - - // Create actions - if (createAutomationDto.actions?.length) { - const actions = createAutomationDto.actions.map((action, index) => - this.actionRepository.create({ - ...action, - workflowId: savedWorkflow.id, - order: index, - }), - ); - await this.actionRepository.save(actions); + private readonly emailQueue: Queue, private readonly eventEmitter: EventEmitter2) { } + /** + * Create a new automation workflow + */ + async create(createAutomationDto: CreateAutomationDto): Promise { + const workflow = this.workflowRepository.create({ + name: createAutomationDto.name, + description: createAutomationDto.description, + status: WorkflowStatus.DRAFT, + }); + const savedWorkflow = await this.workflowRepository.save(workflow); + // Create triggers + if (createAutomationDto.triggers?.length) { + const triggers = createAutomationDto.triggers.map((trigger) => this.triggerRepository.create({ + ...trigger, + workflowId: savedWorkflow.id, + })); + await this.triggerRepository.save(triggers); + } + // Create actions + if (createAutomationDto.actions?.length) { + const actions = createAutomationDto.actions.map((action, index) => this.actionRepository.create({ + ...action, + workflowId: savedWorkflow.id, + order: index, + })); + await this.actionRepository.save(actions); + } + return this.findOne(savedWorkflow.id); } - - return this.findOne(savedWorkflow.id); - } - - /** - * Get all automation workflows - */ - async findAll( - page: number = 1, - limit: number = 10, - ): Promise<{ - workflows: AutomationWorkflow[]; - total: number; - page: number; - totalPages: number; - }> { - const [workflows, total] = await this.workflowRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - relations: ['triggers', 'actions'], - }); - - return { - workflows, - total, - page, - totalPages: Math.ceil(total / limit), - }; - } - - /** - * Get a single workflow by ID - */ - async findOne(id: string): Promise { - const workflow = await this.workflowRepository.findOne({ - where: { id }, - relations: ['triggers', 'actions'], - }); - - if (!workflow) { - throw new NotFoundException(`Automation workflow with ID ${id} not found`); + /** + * Get all automation workflows + */ + async findAll(page: number = 1, limit: number = 10): Promise<{ + workflows: AutomationWorkflow[]; + total: number; + page: number; + totalPages: number; + }> { + const [workflows, total] = await this.workflowRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + relations: ['triggers', 'actions'], + }); + return { + workflows, + total, + page, + totalPages: Math.ceil(total / limit), + }; } - - return workflow; - } - - /** - * Update a workflow - */ - async update(id: string, updateAutomationDto: UpdateAutomationDto): Promise { - const workflow = await this.findOne(id); - - if (workflow.status === WorkflowStatus.ACTIVE) { - throw new BadRequestException('Deactivate workflow before making changes'); + /** + * Get a single workflow by ID + */ + async findOne(id: string): Promise { + const workflow = await this.workflowRepository.findOne({ + where: { id }, + relations: ['triggers', 'actions'], + }); + if (!workflow) { + throw new NotFoundException(`Automation workflow with ID ${id} not found`); + } + return workflow; } - - Object.assign(workflow, { - name: updateAutomationDto.name ?? workflow.name, - description: updateAutomationDto.description ?? workflow.description, - }); - - await this.workflowRepository.save(workflow); - - // Update triggers if provided - if (updateAutomationDto.triggers) { - await this.triggerRepository.softDelete({ workflowId: id }); - const triggers = updateAutomationDto.triggers.map((trigger) => - this.triggerRepository.create({ - ...trigger, - workflowId: id, - }), - ); - await this.triggerRepository.save(triggers); + /** + * Update a workflow + */ + async update(id: string, updateAutomationDto: UpdateAutomationDto): Promise { + const workflow = await this.findOne(id); + if (workflow.status === WorkflowStatus.ACTIVE) { + throw new BadRequestException('Deactivate workflow before making changes'); + } + Object.assign(workflow, { + name: updateAutomationDto.name ?? workflow.name, + description: updateAutomationDto.description ?? workflow.description, + }); + await this.workflowRepository.save(workflow); + // Update triggers if provided + if (updateAutomationDto.triggers) { + await this.triggerRepository.softDelete({ workflowId: id }); + const triggers = updateAutomationDto.triggers.map((trigger) => this.triggerRepository.create({ + ...trigger, + workflowId: id, + })); + await this.triggerRepository.save(triggers); + } + // Update actions if provided + if (updateAutomationDto.actions) { + await this.actionRepository.softDelete({ workflowId: id }); + const actions = updateAutomationDto.actions.map((action, index) => this.actionRepository.create({ + ...action, + workflowId: id, + order: index, + })); + await this.actionRepository.save(actions); + } + return this.findOne(id); } - - // Update actions if provided - if (updateAutomationDto.actions) { - await this.actionRepository.softDelete({ workflowId: id }); - const actions = updateAutomationDto.actions.map((action, index) => - this.actionRepository.create({ - ...action, - workflowId: id, - order: index, - }), - ); - await this.actionRepository.save(actions); + /** + * Delete a workflow + */ + async remove(id: string): Promise { + const workflow = await this.findOne(id); + if (workflow.status === WorkflowStatus.ACTIVE) { + throw new BadRequestException('Deactivate workflow before deleting'); + } + await this.workflowRepository.manager.transaction(async (manager) => { + await manager.getRepository(AutomationTrigger).softDelete({ workflowId: id }); + await manager.getRepository(AutomationAction).softDelete({ workflowId: id }); + await manager.getRepository(AutomationWorkflow).softDelete(id); + }); } - - return this.findOne(id); - } - - /** - * Delete a workflow - */ - async remove(id: string): Promise { - const workflow = await this.findOne(id); - - if (workflow.status === WorkflowStatus.ACTIVE) { - throw new BadRequestException('Deactivate workflow before deleting'); + /** + * Activate a workflow + */ + async activate(id: string): Promise { + const workflow = await this.findOne(id); + if (!workflow.triggers?.length) { + throw new BadRequestException('Workflow must have at least one trigger'); + } + if (!workflow.actions?.length) { + throw new BadRequestException('Workflow must have at least one action'); + } + workflow.status = WorkflowStatus.ACTIVE; + workflow.activatedAt = new Date(); + return this.workflowRepository.save(workflow); } - - await this.workflowRepository.manager.transaction(async (manager) => { - await manager.getRepository(AutomationTrigger).softDelete({ workflowId: id }); - await manager.getRepository(AutomationAction).softDelete({ workflowId: id }); - await manager.getRepository(AutomationWorkflow).softDelete(id); - }); - } - - /** - * Activate a workflow - */ - async activate(id: string): Promise { - const workflow = await this.findOne(id); - - if (!workflow.triggers?.length) { - throw new BadRequestException('Workflow must have at least one trigger'); + /** + * Deactivate a workflow + */ + async deactivate(id: string): Promise { + const workflow = await this.findOne(id); + workflow.status = WorkflowStatus.INACTIVE; + workflow.deactivatedAt = new Date(); + return this.workflowRepository.save(workflow); } - - if (!workflow.actions?.length) { - throw new BadRequestException('Workflow must have at least one action'); + /** + * Handle user signup event + */ + @OnEvent(APP_EVENTS.USER_SIGNUP) + async handleUserSignup(payload: { + userId: string; + email: string; + }) { + await this.executeTriggeredWorkflows(TriggerType.USER_SIGNUP, payload); } - - workflow.status = WorkflowStatus.ACTIVE; - workflow.activatedAt = new Date(); - - return this.workflowRepository.save(workflow); - } - - /** - * Deactivate a workflow - */ - async deactivate(id: string): Promise { - const workflow = await this.findOne(id); - workflow.status = WorkflowStatus.INACTIVE; - workflow.deactivatedAt = new Date(); - - return this.workflowRepository.save(workflow); - } - - /** - * Handle user signup event - */ - @OnEvent(APP_EVENTS.USER_SIGNUP) - async handleUserSignup(payload: { userId: string; email: string }) { - await this.executeTriggeredWorkflows(TriggerType.USER_SIGNUP, payload); - } - - /** - * Handle course enrollment event - */ - @OnEvent(APP_EVENTS.COURSE_ENROLLED) - async handleCourseEnrollment(payload: { userId: string; courseId: string }) { - await this.executeTriggeredWorkflows(TriggerType.COURSE_ENROLLED, payload); - } - - /** - * Handle course completion event - */ - @OnEvent(APP_EVENTS.COURSE_COMPLETED) - async handleCourseCompletion(payload: { userId: string; courseId: string }) { - await this.executeTriggeredWorkflows(TriggerType.COURSE_COMPLETED, payload); - } - - /** - * Handle purchase event - */ - @OnEvent(APP_EVENTS.PAYMENT_COMPLETED) - async handlePurchase(payload: { userId: string; amount: number; productId: string }) { - await this.executeTriggeredWorkflows(TriggerType.PURCHASE_MADE, payload); - } - - /** - * Handle user inactivity (called by scheduled job) - */ - async handleUserInactivity(payload: { userId: string; daysSinceLastActivity: number }) { - await this.executeTriggeredWorkflows(TriggerType.USER_INACTIVE, payload); - } - - /** - * Execute workflows that match the trigger type - */ - private async executeTriggeredWorkflows( - triggerType: TriggerType, - payload: Record, - ): Promise { - // Find all active workflows with matching trigger - const triggers = await this.triggerRepository.find({ - where: { type: triggerType }, - relations: ['workflow', 'workflow.actions'], - }); - - for (const trigger of triggers) { - if (trigger.workflow.status !== WorkflowStatus.ACTIVE) { - continue; - } - - // Check trigger conditions - if (this.evaluateTriggerConditions(trigger, payload)) { - await this.executeWorkflowActions(trigger.workflow, payload); - } + /** + * Handle course enrollment event + */ + @OnEvent(APP_EVENTS.COURSE_ENROLLED) + async handleCourseEnrollment(payload: { + userId: string; + courseId: string; + }) { + await this.executeTriggeredWorkflows(TriggerType.COURSE_ENROLLED, payload); } - } - - /** - * Evaluate trigger conditions - */ - private evaluateTriggerConditions( - trigger: AutomationTrigger, - payload: Record, - ): boolean { - if (!trigger.conditions || Object.keys(trigger.conditions).length === 0) { - return true; + /** + * Handle course completion event + */ + @OnEvent(APP_EVENTS.COURSE_COMPLETED) + async handleCourseCompletion(payload: { + userId: string; + courseId: string; + }) { + await this.executeTriggeredWorkflows(TriggerType.COURSE_COMPLETED, payload); } - - // Simple condition matching - for (const [key, value] of Object.entries(trigger.conditions)) { - if (payload[key] !== value) { - return false; - } + /** + * Handle purchase event + */ + @OnEvent(APP_EVENTS.PAYMENT_COMPLETED) + async handlePurchase(payload: { + userId: string; + amount: number; + productId: string; + }) { + await this.executeTriggeredWorkflows(TriggerType.PURCHASE_MADE, payload); } - - return true; - } - - /** - * Execute workflow actions in order - */ - private async executeWorkflowActions( - workflow: AutomationWorkflow, - payload: Record, - ): Promise { - const sortedActions = workflow.actions.sort((a, b) => a.order - b.order); - - for (const action of sortedActions) { - await this.executeAction(action, payload); + /** + * Handle user inactivity (called by scheduled job) + */ + async handleUserInactivity(payload: { + userId: string; + daysSinceLastActivity: number; + }) { + await this.executeTriggeredWorkflows(TriggerType.USER_INACTIVE, payload); } - - // Update workflow stats - workflow.executionCount = (workflow.executionCount || 0) + 1; - workflow.lastExecutedAt = new Date(); - await this.workflowRepository.save(workflow); - } - - /** - * Execute a single action - */ - private async executeAction( - action: AutomationAction, - payload: Record, - ): Promise { - switch (action.type) { - case ActionType.SEND_EMAIL: - await this.emailQueue.add(JOB_NAMES.SEND_AUTOMATION_EMAIL, { - actionId: action.id, - templateId: action.config.templateId, - userId: payload.userId, - variables: { ...payload, ...action.config.variables }, - }); - break; - - case ActionType.WAIT: - await this.emailQueue.add( - JOB_NAMES.CONTINUE_AUTOMATION, - { - workflowId: action.workflowId, - nextActionOrder: action.order + 1, - payload, - }, - { delay: action.config.delayMs || 0 }, - ); - break; - - case ActionType.ADD_TAG: - this.eventEmitter.emit(APP_EVENTS.USER_ADD_TAG, { - userId: payload.userId, - tag: action.config.tag, + /** + * Execute workflows that match the trigger type + */ + private async executeTriggeredWorkflows(triggerType: TriggerType, payload: Record): Promise { + // Find all active workflows with matching trigger + const triggers = await this.triggerRepository.find({ + where: { type: triggerType }, + relations: ['workflow', 'workflow.actions'], }); - break; - - case ActionType.REMOVE_TAG: - this.eventEmitter.emit(APP_EVENTS.USER_REMOVE_TAG, { - userId: payload.userId, - tag: action.config.tag, - }); - break; - - case ActionType.ADD_TO_SEGMENT: - this.eventEmitter.emit(APP_EVENTS.SEGMENT_ADD_USER, { - userId: payload.userId, - segmentId: action.config.segmentId, - }); - break; - - case ActionType.WEBHOOK: - await this.emailQueue.add(JOB_NAMES.CALL_WEBHOOK, { - url: action.config.webhookUrl, - method: action.config.method || 'POST', - payload: { ...payload, ...action.config.webhookPayload }, - }); - break; - - default: - console.warn(`Unknown action type: ${action.type}`); + for (const trigger of triggers) { + if (trigger.workflow.status !== WorkflowStatus.ACTIVE) { + continue; + } + // Check trigger conditions + if (this.evaluateTriggerConditions(trigger, payload)) { + await this.executeWorkflowActions(trigger.workflow, payload); + } + } + } + /** + * Evaluate trigger conditions + */ + private evaluateTriggerConditions(trigger: AutomationTrigger, payload: Record): boolean { + if (!trigger.conditions || Object.keys(trigger.conditions).length === 0) { + return true; + } + // Simple condition matching + for (const [key, value] of Object.entries(trigger.conditions)) { + if (payload[key] !== value) { + return false; + } + } + return true; + } + /** + * Execute workflow actions in order + */ + private async executeWorkflowActions(workflow: AutomationWorkflow, payload: Record): Promise { + const sortedActions = workflow.actions.sort((a, b) => a.order - b.order); + for (const action of sortedActions) { + await this.executeAction(action, payload); + } + // Update workflow stats + workflow.executionCount = (workflow.executionCount || 0) + 1; + workflow.lastExecutedAt = new Date(); + await this.workflowRepository.save(workflow); + } + /** + * Execute a single action + */ + private async executeAction(action: AutomationAction, payload: Record): Promise { + switch (action.type) { + case ActionType.SEND_EMAIL: + await this.emailQueue.add(JOB_NAMES.SEND_AUTOMATION_EMAIL, { + actionId: action.id, + templateId: action.config.templateId, + userId: payload.userId, + variables: { ...payload, ...action.config.variables }, + }); + break; + case ActionType.WAIT: + await this.emailQueue.add(JOB_NAMES.CONTINUE_AUTOMATION, { + workflowId: action.workflowId, + nextActionOrder: action.order + 1, + payload, + }, { delay: action.config.delayMs || 0 }); + break; + case ActionType.ADD_TAG: + this.eventEmitter.emit(APP_EVENTS.USER_ADD_TAG, { + userId: payload.userId, + tag: action.config.tag, + }); + break; + case ActionType.REMOVE_TAG: + this.eventEmitter.emit(APP_EVENTS.USER_REMOVE_TAG, { + userId: payload.userId, + tag: action.config.tag, + }); + break; + case ActionType.ADD_TO_SEGMENT: + this.eventEmitter.emit(APP_EVENTS.SEGMENT_ADD_USER, { + userId: payload.userId, + segmentId: action.config.segmentId, + }); + break; + case ActionType.WEBHOOK: + await this.emailQueue.add(JOB_NAMES.CALL_WEBHOOK, { + url: action.config.webhookUrl, + method: action.config.method || 'POST', + payload: { ...payload, ...action.config.webhookPayload }, + }); + break; + default: + console.warn(`Unknown action type: ${action.type}`); + } + } + /** + * Get workflow execution statistics + */ + async getWorkflowStats(id: string): Promise<{ + executionCount: number; + lastExecutedAt: Date | null; + emailsSent: number; + openRate: number; + clickRate: number; + }> { + const workflow = await this.findOne(id); + // TODO: Calculate email stats from analytics + return { + executionCount: workflow.executionCount || 0, + lastExecutedAt: workflow.lastExecutedAt, + emailsSent: 0, + openRate: 0, + clickRate: 0, + }; } - } - - /** - * Get workflow execution statistics - */ - async getWorkflowStats(id: string): Promise<{ - executionCount: number; - lastExecutedAt: Date | null; - emailsSent: number; - openRate: number; - clickRate: number; - }> { - const workflow = await this.findOne(id); - - // TODO: Calculate email stats from analytics - return { - executionCount: workflow.executionCount || 0, - lastExecutedAt: workflow.lastExecutedAt, - emailsSent: 0, - openRate: 0, - clickRate: 0, - }; - } } diff --git a/src/email-marketing/dto/add-segment-members.dto.ts b/src/email-marketing/dto/add-segment-members.dto.ts index 72ec280e..39c28630 100644 --- a/src/email-marketing/dto/add-segment-members.dto.ts +++ b/src/email-marketing/dto/add-segment-members.dto.ts @@ -1,15 +1,14 @@ import { IsArray, IsString, ArrayNotEmpty, IsNotEmpty } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class AddSegmentMembersDto { - @ApiProperty({ - description: 'Array of user IDs to add to the segment', - example: ['user1', 'user2', 'user3'], - type: [String], - }) - @IsArray() - @ArrayNotEmpty() - @IsString({ each: true }) - @IsNotEmpty() - userIds: string[]; + @ApiProperty({ + description: 'Array of user IDs to add to the segment', + example: ['user1', 'user2', 'user3'], + type: [String], + }) + @IsArray() + @ArrayNotEmpty() + @IsString({ each: true }) + @IsNotEmpty() + userIds: string[]; } diff --git a/src/email-marketing/dto/create-ab-test.dto.ts b/src/email-marketing/dto/create-ab-test.dto.ts index d88c909f..d17e501a 100644 --- a/src/email-marketing/dto/create-ab-test.dto.ts +++ b/src/email-marketing/dto/create-ab-test.dto.ts @@ -1,84 +1,63 @@ -import { - IsString, - IsArray, - IsUUID, - IsNotEmpty, - IsNumber, - IsOptional, - ValidateNested, - Min, - Max, -} from 'class-validator'; +import { IsString, IsArray, IsUUID, IsNotEmpty, IsNumber, IsOptional, ValidateNested, Min, Max, } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { Type } from 'class-transformer'; - export class CreateABTestVariantDto { - @ApiPropertyOptional({ description: 'Variant name' }) - @IsString() - @IsOptional() - name?: string; - - @ApiPropertyOptional({ description: 'Subject line for this variant' }) - @IsString() - @IsOptional() - subject?: string; - - @ApiPropertyOptional({ description: 'Template ID for this variant' }) - @IsUUID() - @IsOptional() - @IsString() - templateId?: string; - - @ApiPropertyOptional({ description: 'Sender name for this variant' }) - @IsString() - @IsOptional() - senderName?: string; - - @ApiProperty({ description: 'Traffic weight (percentage)', example: 50 }) - @IsNumber() - @Min(1) - @Max(99) - @IsNotEmpty() - weight: number; + @ApiPropertyOptional({ description: 'Variant name' }) + @IsString() + @IsOptional() + name?: string; + @ApiPropertyOptional({ description: 'Subject line for this variant' }) + @IsString() + @IsOptional() + subject?: string; + @ApiPropertyOptional({ description: 'Template ID for this variant' }) + @IsUUID() + @IsOptional() + @IsString() + templateId?: string; + @ApiPropertyOptional({ description: 'Sender name for this variant' }) + @IsString() + @IsOptional() + senderName?: string; + @ApiProperty({ description: 'Traffic weight (percentage)', example: 50 }) + @IsNumber() + @Min(1) + @Max(99) + @IsNotEmpty() + weight: number; } - export class CreateABTestDto { - @ApiProperty({ description: 'Test name', example: 'Subject Line Test' }) - @IsString() - @IsNotEmpty() - name: string; - - @ApiProperty({ description: 'Campaign ID to run test on' }) - @IsUUID() - @IsNotEmpty() - @IsString() - campaignId: string; - - @ApiProperty({ description: 'Field to test', example: 'subject' }) - @IsString() - @IsNotEmpty() - testField: string; - - @ApiPropertyOptional({ - description: 'Winner criteria', - enum: ['open_rate', 'click_rate'], - default: 'open_rate', - }) - @IsString() - @IsOptional() - winnerCriteria?: string; - - @ApiPropertyOptional({ description: 'Sample size percentage', default: 20 }) - @IsNumber() - @Min(5) - @Max(50) - @IsOptional() - sampleSize?: number; - - @ApiProperty({ type: [CreateABTestVariantDto], description: 'Test variants (min 2)' }) - @IsArray() - @ValidateNested({ each: true }) - @Type(() => CreateABTestVariantDto) - @IsNotEmpty() - variants: CreateABTestVariantDto[]; + @ApiProperty({ description: 'Test name', example: 'Subject Line Test' }) + @IsString() + @IsNotEmpty() + name: string; + @ApiProperty({ description: 'Campaign ID to run test on' }) + @IsUUID() + @IsNotEmpty() + @IsString() + campaignId: string; + @ApiProperty({ description: 'Field to test', example: 'subject' }) + @IsString() + @IsNotEmpty() + testField: string; + @ApiPropertyOptional({ + description: 'Winner criteria', + enum: ['open_rate', 'click_rate'], + default: 'open_rate', + }) + @IsString() + @IsOptional() + winnerCriteria?: string; + @ApiPropertyOptional({ description: 'Sample size percentage', default: 20 }) + @IsNumber() + @Min(5) + @Max(50) + @IsOptional() + sampleSize?: number; + @ApiProperty({ type: [CreateABTestVariantDto], description: 'Test variants (min 2)' }) + @IsArray() + @ValidateNested({ each: true }) + @Type(() => CreateABTestVariantDto) + @IsNotEmpty() + variants: CreateABTestVariantDto[]; } diff --git a/src/email-marketing/dto/create-automation.dto.ts b/src/email-marketing/dto/create-automation.dto.ts index aa76578d..76b92f2d 100644 --- a/src/email-marketing/dto/create-automation.dto.ts +++ b/src/email-marketing/dto/create-automation.dto.ts @@ -3,63 +3,53 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { Type } from 'class-transformer'; import { TriggerType } from '../enums/trigger-type.enum'; import { ActionType } from '../enums/action-type.enum'; - export class CreateTriggerDto { - @ApiProperty({ enum: TriggerType }) - @IsEnum(TriggerType) - @IsNotEmpty() - type: TriggerType; - - @ApiPropertyOptional({ description: 'Trigger conditions' }) - @IsOptional() - @IsString() - conditions?: Record; - - @ApiPropertyOptional() - @IsString() - @IsOptional() - description?: string; + @ApiProperty({ enum: TriggerType }) + @IsEnum(TriggerType) + @IsNotEmpty() + type: TriggerType; + @ApiPropertyOptional({ description: 'Trigger conditions' }) + @IsOptional() + @IsString() + conditions?: Record; + @ApiPropertyOptional() + @IsString() + @IsOptional() + description?: string; } - export class CreateActionDto { - @ApiProperty({ enum: ActionType }) - @IsEnum(ActionType) - @IsNotEmpty() - type: ActionType; - - @ApiProperty({ description: 'Action configuration' }) - @IsNotEmpty() - @IsString() - config: Record; - - @ApiPropertyOptional() - @IsString() - @IsOptional() - description?: string; + @ApiProperty({ enum: ActionType }) + @IsEnum(ActionType) + @IsNotEmpty() + type: ActionType; + @ApiProperty({ description: 'Action configuration' }) + @IsNotEmpty() + @IsString() + config: Record; + @ApiPropertyOptional() + @IsString() + @IsOptional() + description?: string; } - export class CreateAutomationDto { - @ApiProperty({ description: 'Workflow name', example: 'Welcome Series' }) - @IsString() - @IsNotEmpty() - name: string; - - @ApiPropertyOptional({ description: 'Workflow description' }) - @IsString() - @IsOptional() - description?: string; - - @ApiPropertyOptional({ type: [CreateTriggerDto] }) - @IsArray() - @ValidateNested({ each: true }) - @Type(() => CreateTriggerDto) - @IsOptional() - triggers?: CreateTriggerDto[]; - - @ApiPropertyOptional({ type: [CreateActionDto] }) - @IsArray() - @ValidateNested({ each: true }) - @Type(() => CreateActionDto) - @IsOptional() - actions?: CreateActionDto[]; + @ApiProperty({ description: 'Workflow name', example: 'Welcome Series' }) + @IsString() + @IsNotEmpty() + name: string; + @ApiPropertyOptional({ description: 'Workflow description' }) + @IsString() + @IsOptional() + description?: string; + @ApiPropertyOptional({ type: [CreateTriggerDto] }) + @IsArray() + @ValidateNested({ each: true }) + @Type(() => CreateTriggerDto) + @IsOptional() + triggers?: CreateTriggerDto[]; + @ApiPropertyOptional({ type: [CreateActionDto] }) + @IsArray() + @ValidateNested({ each: true }) + @Type(() => CreateActionDto) + @IsOptional() + actions?: CreateActionDto[]; } diff --git a/src/email-marketing/dto/create-campaign.dto.ts b/src/email-marketing/dto/create-campaign.dto.ts index 52a2a4cd..75983c65 100644 --- a/src/email-marketing/dto/create-campaign.dto.ts +++ b/src/email-marketing/dto/create-campaign.dto.ts @@ -1,40 +1,34 @@ import { IsString, IsOptional, IsArray, IsUUID, IsNotEmpty, MaxLength } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; - export class CreateCampaignDto { - @ApiProperty({ description: 'Campaign name', example: 'Welcome Campaign' }) - @IsString() - @IsNotEmpty() - @MaxLength(255) - name: string; - - @ApiProperty({ description: 'Email subject line', example: 'Welcome to TeachLink!' }) - @IsString() - @IsNotEmpty() - @MaxLength(255) - subject: string; - - @ApiPropertyOptional({ description: 'Preview text shown in inbox' }) - @IsString() - @IsOptional() - @MaxLength(255) - previewText?: string; - - @ApiPropertyOptional({ description: 'Raw HTML content (if not using template)' }) - @IsString() - @IsOptional() - content?: string; - - @ApiPropertyOptional({ description: 'Template ID to use' }) - @IsUUID() - @IsOptional() - @IsString() - templateId?: string; - - @ApiPropertyOptional({ description: 'Segment IDs to target', type: [String] }) - @IsArray() - @IsUUID('4', { each: true }) - @IsOptional() - @IsString() - segmentIds?: string[]; + @ApiProperty({ description: 'Campaign name', example: 'Welcome Campaign' }) + @IsString() + @IsNotEmpty() + @MaxLength(255) + name: string; + @ApiProperty({ description: 'Email subject line', example: 'Welcome to TeachLink!' }) + @IsString() + @IsNotEmpty() + @MaxLength(255) + subject: string; + @ApiPropertyOptional({ description: 'Preview text shown in inbox' }) + @IsString() + @IsOptional() + @MaxLength(255) + previewText?: string; + @ApiPropertyOptional({ description: 'Raw HTML content (if not using template)' }) + @IsString() + @IsOptional() + content?: string; + @ApiPropertyOptional({ description: 'Template ID to use' }) + @IsUUID() + @IsOptional() + @IsString() + templateId?: string; + @ApiPropertyOptional({ description: 'Segment IDs to target', type: [String] }) + @IsArray() + @IsUUID('4', { each: true }) + @IsOptional() + @IsString() + segmentIds?: string[]; } diff --git a/src/email-marketing/dto/create-segment.dto.ts b/src/email-marketing/dto/create-segment.dto.ts index 74df6e9e..160c94c0 100644 --- a/src/email-marketing/dto/create-segment.dto.ts +++ b/src/email-marketing/dto/create-segment.dto.ts @@ -1,58 +1,42 @@ -import { - IsString, - IsOptional, - IsArray, - IsBoolean, - IsNotEmpty, - ValidateNested, - IsEnum, -} from 'class-validator'; +import { IsString, IsOptional, IsArray, IsBoolean, IsNotEmpty, ValidateNested, IsEnum, } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { Type } from 'class-transformer'; import { SegmentRuleField } from '../enums/segment-rule-field.enum'; import { SegmentRuleOperator } from '../enums/segment-rule-operator.enum'; - export class CreateSegmentRuleDto { - @ApiProperty({ enum: SegmentRuleField, example: 'email' }) - @IsEnum(SegmentRuleField) - @IsNotEmpty() - field: SegmentRuleField; - - @ApiProperty({ enum: SegmentRuleOperator, example: 'contains' }) - @IsEnum(SegmentRuleOperator) - @IsNotEmpty() - operator: SegmentRuleOperator; - - @ApiProperty({ description: 'Rule value', example: 'gmail.com' }) - @IsNotEmpty() - value: any; - - @ApiPropertyOptional({ enum: ['AND', 'OR'], default: 'AND' }) - @IsOptional() - @IsString() - logicalOperator?: 'AND' | 'OR'; + @ApiProperty({ enum: SegmentRuleField, example: 'email' }) + @IsEnum(SegmentRuleField) + @IsNotEmpty() + field: SegmentRuleField; + @ApiProperty({ enum: SegmentRuleOperator, example: 'contains' }) + @IsEnum(SegmentRuleOperator) + @IsNotEmpty() + operator: SegmentRuleOperator; + @ApiProperty({ description: 'Rule value', example: 'gmail.com' }) + @IsNotEmpty() + value: unknown; + @ApiPropertyOptional({ enum: ['AND', 'OR'], default: 'AND' }) + @IsOptional() + @IsString() + logicalOperator?: 'AND' | 'OR'; } - export class CreateSegmentDto { - @ApiProperty({ description: 'Segment name', example: 'Active Users' }) - @IsString() - @IsNotEmpty() - name: string; - - @ApiPropertyOptional({ description: 'Segment description' }) - @IsString() - @IsOptional() - description?: string; - - @ApiPropertyOptional({ description: 'Dynamic or static segment', default: true }) - @IsBoolean() - @IsOptional() - isDynamic?: boolean; - - @ApiPropertyOptional({ type: [CreateSegmentRuleDto] }) - @IsArray() - @ValidateNested({ each: true }) - @Type(() => CreateSegmentRuleDto) - @IsOptional() - rules?: CreateSegmentRuleDto[]; + @ApiProperty({ description: 'Segment name', example: 'Active Users' }) + @IsString() + @IsNotEmpty() + name: string; + @ApiPropertyOptional({ description: 'Segment description' }) + @IsString() + @IsOptional() + description?: string; + @ApiPropertyOptional({ description: 'Dynamic or static segment', default: true }) + @IsBoolean() + @IsOptional() + isDynamic?: boolean; + @ApiPropertyOptional({ type: [CreateSegmentRuleDto] }) + @IsArray() + @ValidateNested({ each: true }) + @Type(() => CreateSegmentRuleDto) + @IsOptional() + rules?: CreateSegmentRuleDto[]; } diff --git a/src/email-marketing/dto/create-template.dto.ts b/src/email-marketing/dto/create-template.dto.ts index 3ebb26ae..64df8a51 100644 --- a/src/email-marketing/dto/create-template.dto.ts +++ b/src/email-marketing/dto/create-template.dto.ts @@ -1,32 +1,27 @@ import { IsString, IsNotEmpty, IsOptional, MaxLength } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; - export class CreateTemplateDto { - @ApiProperty({ description: 'Template name', example: 'Welcome Email' }) - @IsString() - @IsNotEmpty() - @MaxLength(255) - name: string; - - @ApiProperty({ description: 'Email subject with variables', example: 'Welcome, {{firstName}}!' }) - @IsString() - @IsNotEmpty() - @MaxLength(255) - subject: string; - - @ApiProperty({ description: 'HTML content with Handlebars variables' }) - @IsString() - @IsNotEmpty() - htmlContent: string; - - @ApiPropertyOptional({ description: 'Plain text version' }) - @IsString() - @IsOptional() - textContent?: string; - - @ApiPropertyOptional({ description: 'Category for organization' }) - @IsString() - @IsOptional() - @MaxLength(100) - category?: string; + @ApiProperty({ description: 'Template name', example: 'Welcome Email' }) + @IsString() + @IsNotEmpty() + @MaxLength(255) + name: string; + @ApiProperty({ description: 'Email subject with variables', example: 'Welcome, {{firstName}}!' }) + @IsString() + @IsNotEmpty() + @MaxLength(255) + subject: string; + @ApiProperty({ description: 'HTML content with Handlebars variables' }) + @IsString() + @IsNotEmpty() + htmlContent: string; + @ApiPropertyOptional({ description: 'Plain text version' }) + @IsString() + @IsOptional() + textContent?: string; + @ApiPropertyOptional({ description: 'Category for organization' }) + @IsString() + @IsOptional() + @MaxLength(100) + category?: string; } diff --git a/src/email-marketing/dto/schedule-campaign.dto.ts b/src/email-marketing/dto/schedule-campaign.dto.ts index 00946a87..46c21426 100644 --- a/src/email-marketing/dto/schedule-campaign.dto.ts +++ b/src/email-marketing/dto/schedule-campaign.dto.ts @@ -1,10 +1,9 @@ import { IsDateString, IsNotEmpty, IsString } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; - export class ScheduleCampaignDto { - @ApiProperty({ description: 'Scheduled send time (ISO 8601)', example: '2026-02-01T10:00:00Z' }) - @IsDateString() - @IsNotEmpty() - @IsString() - scheduledAt: string; + @ApiProperty({ description: 'Scheduled send time (ISO 8601)', example: '2026-02-01T10:00:00Z' }) + @IsDateString() + @IsNotEmpty() + @IsString() + scheduledAt: string; } diff --git a/src/email-marketing/dto/update-automation.dto.ts b/src/email-marketing/dto/update-automation.dto.ts index 5f00c82d..0ac4af48 100644 --- a/src/email-marketing/dto/update-automation.dto.ts +++ b/src/email-marketing/dto/update-automation.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateAutomationDto } from './create-automation.dto'; - -export class UpdateAutomationDto extends PartialType(CreateAutomationDto) {} +export class UpdateAutomationDto extends PartialType(CreateAutomationDto) { +} diff --git a/src/email-marketing/dto/update-campaign.dto.ts b/src/email-marketing/dto/update-campaign.dto.ts index e5a8ecdd..aaaaee76 100644 --- a/src/email-marketing/dto/update-campaign.dto.ts +++ b/src/email-marketing/dto/update-campaign.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateCampaignDto } from './create-campaign.dto'; - -export class UpdateCampaignDto extends PartialType(CreateCampaignDto) {} +export class UpdateCampaignDto extends PartialType(CreateCampaignDto) { +} diff --git a/src/email-marketing/dto/update-segment.dto.ts b/src/email-marketing/dto/update-segment.dto.ts index 8cfac3d6..91392736 100644 --- a/src/email-marketing/dto/update-segment.dto.ts +++ b/src/email-marketing/dto/update-segment.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateSegmentDto } from './create-segment.dto'; - -export class UpdateSegmentDto extends PartialType(CreateSegmentDto) {} +export class UpdateSegmentDto extends PartialType(CreateSegmentDto) { +} diff --git a/src/email-marketing/dto/update-template.dto.ts b/src/email-marketing/dto/update-template.dto.ts index 176dc19e..888ce9a0 100644 --- a/src/email-marketing/dto/update-template.dto.ts +++ b/src/email-marketing/dto/update-template.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateTemplateDto } from './create-template.dto'; - -export class UpdateTemplateDto extends PartialType(CreateTemplateDto) {} +export class UpdateTemplateDto extends PartialType(CreateTemplateDto) { +} diff --git a/src/email-marketing/email-marketing.controller.ts b/src/email-marketing/email-marketing.controller.ts index 4c322977..4d914a8a 100644 --- a/src/email-marketing/email-marketing.controller.ts +++ b/src/email-marketing/email-marketing.controller.ts @@ -1,125 +1,122 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Body, - Param, - Query, - ParseUUIDPipe, - HttpCode, - HttpStatus, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Body, Param, Query, ParseUUIDPipe, HttpCode, HttpStatus, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { EmailMarketingService } from './email-marketing.service'; import { CreateCampaignDto } from './dto/create-campaign.dto'; import { UpdateCampaignDto } from './dto/update-campaign.dto'; import { ScheduleCampaignDto } from './dto/schedule-campaign.dto'; import { Campaign } from './entities/campaign.entity'; - @ApiTags('Email Marketing - Campaigns') @ApiBearerAuth() @Controller('email-marketing/campaigns') export class EmailMarketingController { - constructor(private readonly emailMarketingService: EmailMarketingService) {} - - @Post() - @ApiOperation({ summary: 'Create a new email campaign' }) - @ApiResponse({ status: 201, description: 'Campaign created successfully', type: Campaign }) - @ApiResponse({ status: 400, description: 'Invalid input data' }) - async create(@Body() createCampaignDto: CreateCampaignDto): Promise { - return this.emailMarketingService.createCampaign(createCampaignDto); - } - - @Get() - @ApiOperation({ summary: 'Get all campaigns with pagination' }) - @ApiQuery({ name: 'page', required: false, type: Number, example: 1 }) - @ApiQuery({ name: 'limit', required: false, type: Number, example: 10 }) - @ApiResponse({ status: 200, description: 'List of campaigns' }) - async findAll(@Query('page') page: number = 1, @Query('limit') limit: number = 10) { - return this.emailMarketingService.findAll(page, limit); - } - - @Get(':id') - @ApiOperation({ summary: 'Get a campaign by ID' }) - @ApiResponse({ status: 200, description: 'Campaign details', type: Campaign }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async findOne(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.findOne(id); - } - - @Put(':id') - @ApiOperation({ summary: 'Update a campaign' }) - @ApiResponse({ status: 200, description: 'Campaign updated successfully', type: Campaign }) - @ApiResponse({ status: 400, description: 'Cannot update sent campaign' }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async update( - @Param('id', ParseUUIDPipe) id: string, - @Body() updateCampaignDto: UpdateCampaignDto, - ): Promise { - return this.emailMarketingService.update(id, updateCampaignDto); - } - - @Delete(':id') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete a campaign' }) - @ApiResponse({ status: 204, description: 'Campaign deleted successfully' }) - @ApiResponse({ status: 400, description: 'Cannot delete sending campaign' }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async remove(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.remove(id); - } - - @Post(':id/schedule') - @ApiOperation({ summary: 'Schedule a campaign for future sending' }) - @ApiResponse({ status: 200, description: 'Campaign scheduled successfully', type: Campaign }) - @ApiResponse({ status: 400, description: 'Invalid schedule or campaign status' }) - async schedule( - @Param('id', ParseUUIDPipe) id: string, - @Body() scheduleDto: ScheduleCampaignDto, - ): Promise { - return this.emailMarketingService.scheduleCampaign(id, scheduleDto); - } - - @Post(':id/send') - @ApiOperation({ summary: 'Send a campaign immediately' }) - @ApiResponse({ status: 200, description: 'Campaign sending initiated', type: Campaign }) - @ApiResponse({ status: 400, description: 'Campaign cannot be sent' }) - async send(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.sendCampaign(id); - } - - @Post(':id/pause') - @ApiOperation({ summary: 'Pause a scheduled or sending campaign' }) - @ApiResponse({ status: 200, description: 'Campaign paused successfully', type: Campaign }) - @ApiResponse({ status: 400, description: 'Campaign cannot be paused' }) - async pause(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.pauseCampaign(id); - } - - @Post(':id/resume') - @ApiOperation({ summary: 'Resume a paused campaign' }) - @ApiResponse({ status: 200, description: 'Campaign resumed successfully', type: Campaign }) - @ApiResponse({ status: 400, description: 'Campaign cannot be resumed' }) - async resume(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.resumeCampaign(id); - } - - @Post(':id/duplicate') - @ApiOperation({ summary: 'Duplicate a campaign' }) - @ApiResponse({ status: 201, description: 'Campaign duplicated successfully', type: Campaign }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async duplicate(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.emailMarketingService.duplicateCampaign(id); - } - - @Get(':id/stats') - @ApiOperation({ summary: 'Get campaign statistics' }) - @ApiResponse({ status: 200, description: 'Campaign statistics' }) - @ApiResponse({ status: 404, description: 'Campaign not found' }) - async getStats(@Param('id', ParseUUIDPipe) id: string) { - return this.emailMarketingService.getCampaignStats(id); - } + constructor(private readonly emailMarketingService: EmailMarketingService) { } + @Post() + @ApiOperation({ summary: 'Create a new email campaign' }) + @ApiResponse({ status: 201, description: 'Campaign created successfully', type: Campaign }) + @ApiResponse({ status: 400, description: 'Invalid input data' }) + async create( + @Body() + createCampaignDto: CreateCampaignDto): Promise { + return this.emailMarketingService.createCampaign(createCampaignDto); + } + @Get() + @ApiOperation({ summary: 'Get all campaigns with pagination' }) + @ApiQuery({ name: 'page', required: false, type: Number, example: 1 }) + @ApiQuery({ name: 'limit', required: false, type: Number, example: 10 }) + @ApiResponse({ status: 200, description: 'List of campaigns' }) + async findAll( + @Query('page') + page: number = 1, + @Query('limit') + limit: number = 10) { + return this.emailMarketingService.findAll(page, limit); + } + @Get(':id') + @ApiOperation({ summary: 'Get a campaign by ID' }) + @ApiResponse({ status: 200, description: 'Campaign details', type: Campaign }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async findOne( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.findOne(id); + } + @Put(':id') + @ApiOperation({ summary: 'Update a campaign' }) + @ApiResponse({ status: 200, description: 'Campaign updated successfully', type: Campaign }) + @ApiResponse({ status: 400, description: 'Cannot update sent campaign' }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async update( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + updateCampaignDto: UpdateCampaignDto): Promise { + return this.emailMarketingService.update(id, updateCampaignDto); + } + @Delete(':id') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete a campaign' }) + @ApiResponse({ status: 204, description: 'Campaign deleted successfully' }) + @ApiResponse({ status: 400, description: 'Cannot delete sending campaign' }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async remove( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.remove(id); + } + @Post(':id/schedule') + @ApiOperation({ summary: 'Schedule a campaign for future sending' }) + @ApiResponse({ status: 200, description: 'Campaign scheduled successfully', type: Campaign }) + @ApiResponse({ status: 400, description: 'Invalid schedule or campaign status' }) + async schedule( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + scheduleDto: ScheduleCampaignDto): Promise { + return this.emailMarketingService.scheduleCampaign(id, scheduleDto); + } + @Post(':id/send') + @ApiOperation({ summary: 'Send a campaign immediately' }) + @ApiResponse({ status: 200, description: 'Campaign sending initiated', type: Campaign }) + @ApiResponse({ status: 400, description: 'Campaign cannot be sent' }) + async send( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.sendCampaign(id); + } + @Post(':id/pause') + @ApiOperation({ summary: 'Pause a scheduled or sending campaign' }) + @ApiResponse({ status: 200, description: 'Campaign paused successfully', type: Campaign }) + @ApiResponse({ status: 400, description: 'Campaign cannot be paused' }) + async pause( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.pauseCampaign(id); + } + @Post(':id/resume') + @ApiOperation({ summary: 'Resume a paused campaign' }) + @ApiResponse({ status: 200, description: 'Campaign resumed successfully', type: Campaign }) + @ApiResponse({ status: 400, description: 'Campaign cannot be resumed' }) + async resume( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.resumeCampaign(id); + } + @Post(':id/duplicate') + @ApiOperation({ summary: 'Duplicate a campaign' }) + @ApiResponse({ status: 201, description: 'Campaign duplicated successfully', type: Campaign }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async duplicate( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.emailMarketingService.duplicateCampaign(id); + } + @Get(':id/stats') + @ApiOperation({ summary: 'Get campaign statistics' }) + @ApiResponse({ status: 200, description: 'Campaign statistics' }) + @ApiResponse({ status: 404, description: 'Campaign not found' }) + async getStats( + @Param('id', ParseUUIDPipe) + id: string) { + return this.emailMarketingService.getCampaignStats(id); + } } diff --git a/src/email-marketing/email-marketing.module.ts b/src/email-marketing/email-marketing.module.ts index 8f7c11d6..27b2792c 100644 --- a/src/email-marketing/email-marketing.module.ts +++ b/src/email-marketing/email-marketing.module.ts @@ -3,7 +3,6 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { BullModule } from '@nestjs/bull'; import { ConfigModule } from '@nestjs/config'; import { QUEUE_NAMES } from '../common/constants/queue.constants'; - // Services import { EmailMarketingService } from './email-marketing.service'; import { AutomationService } from './automation/automation.service'; @@ -12,7 +11,6 @@ import { EmailAnalyticsService } from './analytics/email-analytics.service'; import { TemplateManagementService } from './templates/template-management.service'; import { ABTestingService } from './ab-testing/ab-testing.service'; import { EmailSenderService } from './sender/email-sender.service'; - // Controllers import { EmailMarketingController } from './email-marketing.controller'; import { TemplateController } from './templates/template.controller'; @@ -21,7 +19,6 @@ import { SegmentController } from './segmentation/segment.controller'; import { EmailAnalyticsController } from './analytics/email-analytics.controller'; import { ABTestingController } from './ab-testing/ab-testing.controller'; import { TrackingController } from './tracking/tracking.controller'; - // Entities import { Campaign } from './entities/campaign.entity'; import { EmailTemplate } from './entities/email-template.entity'; @@ -35,57 +32,56 @@ import { ABTest } from './entities/ab-test.entity'; import { ABTestVariant } from './entities/ab-test-variant.entity'; import { CampaignRecipient } from './entities/campaign-recipient.entity'; import { EmailSubscription } from './entities/email-subscription.entity'; - // Processors (Bull Queue) import { EmailQueueProcessor } from './processors/email-queue.processor'; - @Module({ - imports: [ - ConfigModule, - TypeOrmModule.forFeature([ - Campaign, - EmailTemplate, - AutomationWorkflow, - AutomationTrigger, - AutomationAction, - Segment, - SegmentRule, - EmailEvent, - ABTest, - ABTestVariant, - CampaignRecipient, - EmailSubscription, - ]), - BullModule.registerQueue({ - name: QUEUE_NAMES.EMAIL_MARKETING, - }), - ], - controllers: [ - EmailMarketingController, - TemplateController, - AutomationController, - SegmentController, - EmailAnalyticsController, - ABTestingController, - TrackingController, - ], - providers: [ - EmailMarketingService, - AutomationService, - SegmentationService, - EmailAnalyticsService, - TemplateManagementService, - ABTestingService, - EmailSenderService, - EmailQueueProcessor, - ], - exports: [ - EmailMarketingService, - AutomationService, - SegmentationService, - EmailAnalyticsService, - TemplateManagementService, - ABTestingService, - ], + imports: [ + ConfigModule, + TypeOrmModule.forFeature([ + Campaign, + EmailTemplate, + AutomationWorkflow, + AutomationTrigger, + AutomationAction, + Segment, + SegmentRule, + EmailEvent, + ABTest, + ABTestVariant, + CampaignRecipient, + EmailSubscription, + ]), + BullModule.registerQueue({ + name: QUEUE_NAMES.EMAIL_MARKETING, + }), + ], + controllers: [ + EmailMarketingController, + TemplateController, + AutomationController, + SegmentController, + EmailAnalyticsController, + ABTestingController, + TrackingController, + ], + providers: [ + EmailMarketingService, + AutomationService, + SegmentationService, + EmailAnalyticsService, + TemplateManagementService, + ABTestingService, + EmailSenderService, + EmailQueueProcessor, + ], + exports: [ + EmailMarketingService, + AutomationService, + SegmentationService, + EmailAnalyticsService, + TemplateManagementService, + ABTestingService, + ], }) -export class EmailMarketingModule {} +export class EmailMarketingModule { +} diff --git a/src/email-marketing/email-marketing.service.ts b/src/email-marketing/email-marketing.service.ts index f5cc3e0e..56781bc5 100644 --- a/src/email-marketing/email-marketing.service.ts +++ b/src/email-marketing/email-marketing.service.ts @@ -4,273 +4,213 @@ import { Repository } from 'typeorm'; import { InjectQueue } from '@nestjs/bull'; import { Queue } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../common/constants/queue.constants'; - import { Campaign } from './entities/campaign.entity'; import { CampaignRecipient } from './entities/campaign-recipient.entity'; import { SegmentationService } from './segmentation/segmentation.service'; import { TemplateManagementService } from './templates/template-management.service'; import { ABTestingService } from './ab-testing/ab-testing.service'; import { EmailAnalyticsService } from './analytics/email-analytics.service'; - import { CreateCampaignDto } from './dto/create-campaign.dto'; import { UpdateCampaignDto } from './dto/update-campaign.dto'; import { ScheduleCampaignDto } from './dto/schedule-campaign.dto'; import { CampaignStatus } from './enums/campaign-status.enum'; - @Injectable() export class EmailMarketingService { - constructor( + constructor( @InjectRepository(Campaign) - private readonly campaignRepository: Repository, + private readonly campaignRepository: Repository, @InjectRepository(CampaignRecipient) - private readonly recipientRepository: Repository, + private readonly recipientRepository: Repository, @InjectQueue(QUEUE_NAMES.EMAIL_MARKETING) - private readonly emailQueue: Queue, - private readonly segmentationService: SegmentationService, - private readonly templateService: TemplateManagementService, - private readonly abTestingService: ABTestingService, - private readonly analyticsService: EmailAnalyticsService, - ) {} - - /** - * Create a new email campaign - */ - async createCampaign(createCampaignDto: CreateCampaignDto): Promise { - // Validate template exists - if (createCampaignDto.templateId) { - await this.templateService.findOne(createCampaignDto.templateId); + private readonly emailQueue: Queue, private readonly segmentationService: SegmentationService, private readonly templateService: TemplateManagementService, private readonly abTestingService: ABTestingService, private readonly analyticsService: EmailAnalyticsService) { } + /** + * Create a new email campaign + */ + async createCampaign(createCampaignDto: CreateCampaignDto): Promise { + // Validate template exists + if (createCampaignDto.templateId) { + await this.templateService.findOne(createCampaignDto.templateId); + } + // Validate segments exist + if (createCampaignDto.segmentIds?.length) { + const segments = await this.segmentationService.findByIds(createCampaignDto.segmentIds); + if (segments.length !== createCampaignDto.segmentIds.length) { + throw new NotFoundException('One or more segments not found'); + } + } + const campaign = this.campaignRepository.create({ + ...createCampaignDto, + status: CampaignStatus.DRAFT, + }); + return this.campaignRepository.save(campaign); } - - // Validate segments exist - if (createCampaignDto.segmentIds?.length) { - const segments = await this.segmentationService.findByIds(createCampaignDto.segmentIds); - if (segments.length !== createCampaignDto.segmentIds.length) { - throw new NotFoundException('One or more segments not found'); - } + /** + * Get all campaigns with pagination + */ + async findAll(page: number = 1, limit: number = 10): Promise<{ + campaigns: Campaign[]; + total: number; + page: number; + totalPages: number; + }> { + const [campaigns, total] = await this.campaignRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + relations: ['template'], + }); + return { + campaigns, + total, + page, + totalPages: Math.ceil(total / limit), + }; } - - const campaign = this.campaignRepository.create({ - ...createCampaignDto, - status: CampaignStatus.DRAFT, - }); - - return this.campaignRepository.save(campaign); - } - - /** - * Get all campaigns with pagination - */ - async findAll( - page: number = 1, - limit: number = 10, - ): Promise<{ - campaigns: Campaign[]; - total: number; - page: number; - totalPages: number; - }> { - const [campaigns, total] = await this.campaignRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - relations: ['template'], - }); - - return { - campaigns, - total, - page, - totalPages: Math.ceil(total / limit), - }; - } - - /** - * Get a single campaign by ID - */ - async findOne(id: string): Promise { - const campaign = await this.campaignRepository.findOne({ - where: { id }, - relations: ['template', 'abTest', 'recipients'], - }); - - if (!campaign) { - throw new NotFoundException(`Campaign with ID ${id} not found`); + /** + * Get a single campaign by ID + */ + async findOne(id: string): Promise { + const campaign = await this.campaignRepository.findOne({ + where: { id }, + relations: ['template', 'abTest', 'recipients'], + }); + if (!campaign) { + throw new NotFoundException(`Campaign with ID ${id} not found`); + } + return campaign; } - - return campaign; - } - - /** - * Update a campaign - */ - async update(id: string, updateCampaignDto: UpdateCampaignDto): Promise { - const campaign = await this.findOne(id); - - if (campaign.status === CampaignStatus.SENT) { - throw new BadRequestException('Cannot update a sent campaign'); + /** + * Update a campaign + */ + async update(id: string, updateCampaignDto: UpdateCampaignDto): Promise { + const campaign = await this.findOne(id); + if (campaign.status === CampaignStatus.SENT) { + throw new BadRequestException('Cannot update a sent campaign'); + } + Object.assign(campaign, updateCampaignDto); + return this.campaignRepository.save(campaign); } - - Object.assign(campaign, updateCampaignDto); - return this.campaignRepository.save(campaign); - } - - /** - * Delete a campaign - */ - async remove(id: string): Promise { - const campaign = await this.findOne(id); - - if (campaign.status === CampaignStatus.SENDING) { - throw new BadRequestException('Cannot delete a campaign that is currently sending'); + /** + * Delete a campaign + */ + async remove(id: string): Promise { + const campaign = await this.findOne(id); + if (campaign.status === CampaignStatus.SENDING) { + throw new BadRequestException('Cannot delete a campaign that is currently sending'); + } + await this.campaignRepository.manager.transaction(async (manager) => { + await manager.getRepository(CampaignRecipient).softDelete({ campaignId: id }); + await manager.getRepository(Campaign).softDelete(id); + }); } - - await this.campaignRepository.manager.transaction(async (manager) => { - await manager.getRepository(CampaignRecipient).softDelete({ campaignId: id }); - await manager.getRepository(Campaign).softDelete(id); - }); - } - - /** - * Schedule a campaign for future sending - */ - async scheduleCampaign(id: string, scheduleDto: ScheduleCampaignDto): Promise { - const campaign = await this.findOne(id); - - if (campaign.status !== CampaignStatus.DRAFT) { - throw new BadRequestException('Only draft campaigns can be scheduled'); + /** + * Schedule a campaign for future sending + */ + async scheduleCampaign(id: string, scheduleDto: ScheduleCampaignDto): Promise { + const campaign = await this.findOne(id); + if (campaign.status !== CampaignStatus.DRAFT) { + throw new BadRequestException('Only draft campaigns can be scheduled'); + } + const scheduledDate = new Date(scheduleDto.scheduledAt); + if (scheduledDate <= new Date()) { + throw new BadRequestException('Scheduled date must be in the future'); + } + campaign.scheduledAt = scheduledDate; + campaign.status = CampaignStatus.SCHEDULED; + // Add to queue with delay + const delay = scheduledDate.getTime() - Date.now(); + await this.emailQueue.add(JOB_NAMES.SEND_CAMPAIGN, { campaignId: id }, { delay, jobId: `campaign-${id}` }); + return this.campaignRepository.save(campaign); } - - const scheduledDate = new Date(scheduleDto.scheduledAt); - if (scheduledDate <= new Date()) { - throw new BadRequestException('Scheduled date must be in the future'); + /** + * Send a campaign immediately + */ + async sendCampaign(id: string): Promise { + const campaign = await this.findOne(id); + if (campaign.status === CampaignStatus.SENT || campaign.status === CampaignStatus.SENDING) { + throw new BadRequestException('Campaign has already been sent or is sending'); + } + // Get recipients from segments + const recipients = await this.segmentationService.getUsersFromSegments(campaign.segmentIds || []); + if (recipients.length === 0) { + throw new BadRequestException('No recipients found for this campaign'); + } + // Update campaign status + campaign.status = CampaignStatus.SENDING; + campaign.sentAt = new Date(); + campaign.totalRecipients = recipients.length; + await this.campaignRepository.save(campaign); + // Queue emails for sending + await this.emailQueue.add(JOB_NAMES.PROCESS_CAMPAIGN, { + campaignId: id, + recipients: recipients.map((r) => r.id), + }); + return campaign; } - - campaign.scheduledAt = scheduledDate; - campaign.status = CampaignStatus.SCHEDULED; - - // Add to queue with delay - const delay = scheduledDate.getTime() - Date.now(); - await this.emailQueue.add( - JOB_NAMES.SEND_CAMPAIGN, - { campaignId: id }, - { delay, jobId: `campaign-${id}` }, - ); - - return this.campaignRepository.save(campaign); - } - - /** - * Send a campaign immediately - */ - async sendCampaign(id: string): Promise { - const campaign = await this.findOne(id); - - if (campaign.status === CampaignStatus.SENT || campaign.status === CampaignStatus.SENDING) { - throw new BadRequestException('Campaign has already been sent or is sending'); + /** + * Pause a scheduled or sending campaign + */ + async pauseCampaign(id: string): Promise { + const campaign = await this.findOne(id); + if (campaign.status !== CampaignStatus.SCHEDULED && + campaign.status !== CampaignStatus.SENDING) { + throw new BadRequestException('Only scheduled or sending campaigns can be paused'); + } + // Remove from queue if scheduled + if (campaign.status === CampaignStatus.SCHEDULED) { + await this.emailQueue.removeJobs(`campaign-${id}`); + } + campaign.status = CampaignStatus.PAUSED; + return this.campaignRepository.save(campaign); } - - // Get recipients from segments - const recipients = await this.segmentationService.getUsersFromSegments( - campaign.segmentIds || [], - ); - - if (recipients.length === 0) { - throw new BadRequestException('No recipients found for this campaign'); + /** + * Resume a paused campaign + */ + async resumeCampaign(id: string): Promise { + const campaign = await this.findOne(id); + if (campaign.status !== CampaignStatus.PAUSED) { + throw new BadRequestException('Only paused campaigns can be resumed'); + } + // If it was scheduled, re-schedule + if (campaign.scheduledAt && campaign.scheduledAt > new Date()) { + return this.scheduleCampaign(id, { scheduledAt: campaign.scheduledAt.toISOString() }); + } + // Otherwise, resume sending + campaign.status = CampaignStatus.SENDING; + await this.emailQueue.add(JOB_NAMES.RESUME_CAMPAIGN, { campaignId: id }); + return this.campaignRepository.save(campaign); } - - // Update campaign status - campaign.status = CampaignStatus.SENDING; - campaign.sentAt = new Date(); - campaign.totalRecipients = recipients.length; - await this.campaignRepository.save(campaign); - - // Queue emails for sending - await this.emailQueue.add(JOB_NAMES.PROCESS_CAMPAIGN, { - campaignId: id, - recipients: recipients.map((r) => r.id), - }); - - return campaign; - } - - /** - * Pause a scheduled or sending campaign - */ - async pauseCampaign(id: string): Promise { - const campaign = await this.findOne(id); - - if ( - campaign.status !== CampaignStatus.SCHEDULED && - campaign.status !== CampaignStatus.SENDING - ) { - throw new BadRequestException('Only scheduled or sending campaigns can be paused'); + /** + * Duplicate a campaign + */ + async duplicateCampaign(id: string): Promise { + const original = await this.findOne(id); + const duplicate = this.campaignRepository.create({ + name: `${original.name} (Copy)`, + subject: original.subject, + previewText: original.previewText, + templateId: original.templateId, + segmentIds: original.segmentIds, + content: original.content, + status: CampaignStatus.DRAFT, + }); + return this.campaignRepository.save(duplicate); } - - // Remove from queue if scheduled - if (campaign.status === CampaignStatus.SCHEDULED) { - await this.emailQueue.removeJobs(`campaign-${id}`); + /** + * Get campaign statistics + */ + async getCampaignStats(id: string): Promise<{ + sent: number; + delivered: number; + opened: number; + clicked: number; + bounced: number; + unsubscribed: number; + openRate: number; + clickRate: number; + bounceRate: number; + }> { + await this.findOne(id); + return this.analyticsService.getCampaignMetrics(id); } - - campaign.status = CampaignStatus.PAUSED; - return this.campaignRepository.save(campaign); - } - - /** - * Resume a paused campaign - */ - async resumeCampaign(id: string): Promise { - const campaign = await this.findOne(id); - - if (campaign.status !== CampaignStatus.PAUSED) { - throw new BadRequestException('Only paused campaigns can be resumed'); - } - - // If it was scheduled, re-schedule - if (campaign.scheduledAt && campaign.scheduledAt > new Date()) { - return this.scheduleCampaign(id, { scheduledAt: campaign.scheduledAt.toISOString() }); - } - - // Otherwise, resume sending - campaign.status = CampaignStatus.SENDING; - await this.emailQueue.add(JOB_NAMES.RESUME_CAMPAIGN, { campaignId: id }); - - return this.campaignRepository.save(campaign); - } - - /** - * Duplicate a campaign - */ - async duplicateCampaign(id: string): Promise { - const original = await this.findOne(id); - - const duplicate = this.campaignRepository.create({ - name: `${original.name} (Copy)`, - subject: original.subject, - previewText: original.previewText, - templateId: original.templateId, - segmentIds: original.segmentIds, - content: original.content, - status: CampaignStatus.DRAFT, - }); - - return this.campaignRepository.save(duplicate); - } - - /** - * Get campaign statistics - */ - async getCampaignStats(id: string): Promise<{ - sent: number; - delivered: number; - opened: number; - clicked: number; - bounced: number; - unsubscribed: number; - openRate: number; - clickRate: number; - bounceRate: number; - }> { - await this.findOne(id); - return this.analyticsService.getCampaignMetrics(id); - } } diff --git a/src/email-marketing/entities/ab-test-variant.entity.ts b/src/email-marketing/entities/ab-test-variant.entity.ts index 7d4f3d5b..98d4317e 100644 --- a/src/email-marketing/entities/ab-test-variant.entity.ts +++ b/src/email-marketing/entities/ab-test-variant.entity.ts @@ -1,43 +1,33 @@ import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { ABTest } from './ab-test.entity'; - @Entity('ab_test_variants') export class ABTestVariant { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - abTestId: string; - - @ManyToOne(() => ABTest, (abTest) => abTest.variants, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'abTestId' }) - abTest: ABTest; - - @ApiProperty() - @Column() - name: string; // e.g., 'Variant A', 'Variant B' - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - subject?: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - templateId?: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - senderName?: string; - - @ApiProperty() - @Column({ default: 50 }) - weight: number; // Percentage of traffic - - @ApiProperty() - @Column({ default: 0 }) - recipientCount: number; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + abTestId: string; + @ManyToOne(() => ABTest, (abTest) => abTest.variants, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'abTestId' }) + abTest: ABTest; + @ApiProperty() + @Column() + name: string; // e.g., 'Variant A', 'Variant B' + @ApiProperty({ required: false }) + @Column({ nullable: true }) + subject?: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + templateId?: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + senderName?: string; + @ApiProperty() + @Column({ default: 50 }) + weight: number; // Percentage of traffic + @ApiProperty() + @Column({ default: 0 }) + recipientCount: number; } diff --git a/src/email-marketing/entities/ab-test.entity.ts b/src/email-marketing/entities/ab-test.entity.ts index 089d8b1f..99d5afd6 100644 --- a/src/email-marketing/entities/ab-test.entity.ts +++ b/src/email-marketing/entities/ab-test.entity.ts @@ -1,68 +1,46 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - OneToOne, - OneToMany, - JoinColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, OneToOne, OneToMany, JoinColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { Campaign } from './campaign.entity'; import { ABTestVariant } from './ab-test-variant.entity'; import { ABTestStatus } from '../enums/ab-test-status.enum'; - @Entity('ab_tests') export class ABTest { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - name: string; - - @ApiProperty() - @Column() - campaignId: string; - - @OneToOne(() => Campaign) - @JoinColumn({ name: 'campaignId' }) - campaign: Campaign; - - @ApiProperty() - @Column() - testField: string; // 'subject', 'template', 'sender', 'sendTime' - - @ApiProperty() - @Column({ default: 'open_rate' }) - winnerCriteria: string; // 'open_rate', 'click_rate' - - @ApiProperty() - @Column({ default: 20 }) - sampleSize: number; // Percentage of total recipients for test - - @ApiProperty({ enum: ABTestStatus }) - @Column({ type: 'enum', enum: ABTestStatus, default: ABTestStatus.DRAFT }) - status: ABTestStatus; - - @OneToMany(() => ABTestVariant, (variant) => variant.abTest, { cascade: true }) - variants: ABTestVariant[]; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - winnerId?: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - startedAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - endedAt?: Date; - - @ApiProperty() - @CreateDateColumn() - createdAt: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + name: string; + @ApiProperty() + @Column() + campaignId: string; + @OneToOne(() => Campaign) + @JoinColumn({ name: 'campaignId' }) + campaign: Campaign; + @ApiProperty() + @Column() + testField: string; // 'subject', 'template', 'sender', 'sendTime' + @ApiProperty() + @Column({ default: 'open_rate' }) + winnerCriteria: string; // 'open_rate', 'click_rate' + @ApiProperty() + @Column({ default: 20 }) + sampleSize: number; // Percentage of total recipients for test + @ApiProperty({ enum: ABTestStatus }) + @Column({ type: 'enum', enum: ABTestStatus, default: ABTestStatus.DRAFT }) + status: ABTestStatus; + @OneToMany(() => ABTestVariant, (variant) => variant.abTest, { cascade: true }) + variants: ABTestVariant[]; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + winnerId?: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + startedAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + endedAt?: Date; + @ApiProperty() + @CreateDateColumn() + createdAt: Date; } diff --git a/src/email-marketing/entities/automation-action.entity.ts b/src/email-marketing/entities/automation-action.entity.ts index f67c399b..dafa7803 100644 --- a/src/email-marketing/entities/automation-action.entity.ts +++ b/src/email-marketing/entities/automation-action.entity.ts @@ -1,46 +1,30 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - JoinColumn, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn, DeleteDateColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { AutomationWorkflow } from './automation-workflow.entity'; import { ActionType } from '../enums/action-type.enum'; - @Entity('automation_actions') export class AutomationAction { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - workflowId: string; - - @ManyToOne(() => AutomationWorkflow, (workflow) => workflow.actions, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'workflowId' }) - workflow: AutomationWorkflow; - - @ApiProperty({ enum: ActionType }) - @Column({ type: 'enum', enum: ActionType }) - type: ActionType; - - @ApiProperty() - @Column({ type: 'jsonb' }) - config: Record; - - @ApiProperty() - @Column({ default: 0 }) - order: number; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - description?: string; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + workflowId: string; + @ManyToOne(() => AutomationWorkflow, (workflow) => workflow.actions, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'workflowId' }) + workflow: AutomationWorkflow; + @ApiProperty({ enum: ActionType }) + @Column({ type: 'enum', enum: ActionType }) + type: ActionType; + @ApiProperty() + @Column({ type: 'jsonb' }) + config: Record; + @ApiProperty() + @Column({ default: 0 }) + order: number; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + description?: string; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/automation-trigger.entity.ts b/src/email-marketing/entities/automation-trigger.entity.ts index be89e55d..de28baa9 100644 --- a/src/email-marketing/entities/automation-trigger.entity.ts +++ b/src/email-marketing/entities/automation-trigger.entity.ts @@ -1,42 +1,27 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - JoinColumn, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn, DeleteDateColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { AutomationWorkflow } from './automation-workflow.entity'; import { TriggerType } from '../enums/trigger-type.enum'; - @Entity('automation_triggers') export class AutomationTrigger { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - workflowId: string; - - @ManyToOne(() => AutomationWorkflow, (workflow) => workflow.triggers, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'workflowId' }) - workflow: AutomationWorkflow; - - @ApiProperty({ enum: TriggerType }) - @Column({ type: 'enum', enum: TriggerType }) - type: TriggerType; - - @ApiProperty({ required: false }) - @Column({ type: 'jsonb', nullable: true }) - conditions?: Record; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - description?: string; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + workflowId: string; + @ManyToOne(() => AutomationWorkflow, (workflow) => workflow.triggers, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'workflowId' }) + workflow: AutomationWorkflow; + @ApiProperty({ enum: TriggerType }) + @Column({ type: 'enum', enum: TriggerType }) + type: TriggerType; + @ApiProperty({ required: false }) + @Column({ type: 'jsonb', nullable: true }) + conditions?: Record; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + description?: string; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/automation-workflow.entity.ts b/src/email-marketing/entities/automation-workflow.entity.ts index 5de35511..63c68eb7 100644 --- a/src/email-marketing/entities/automation-workflow.entity.ts +++ b/src/email-marketing/entities/automation-workflow.entity.ts @@ -1,66 +1,44 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - OneToMany, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, OneToMany, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { AutomationTrigger } from './automation-trigger.entity'; import { AutomationAction } from './automation-action.entity'; import { WorkflowStatus } from '../enums/workflow-status.enum'; - @Entity('automation_workflows') export class AutomationWorkflow { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - name: string; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - description?: string; - - @ApiProperty({ enum: WorkflowStatus }) - @Column({ type: 'enum', enum: WorkflowStatus, default: WorkflowStatus.DRAFT }) - status: WorkflowStatus; - - @OneToMany(() => AutomationTrigger, (trigger) => trigger.workflow, { cascade: true }) - triggers: AutomationTrigger[]; - - @OneToMany(() => AutomationAction, (action) => action.workflow, { cascade: true }) - actions: AutomationAction[]; - - @ApiProperty() - @Column({ default: 0 }) - executionCount: number; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - lastExecutedAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - activatedAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - deactivatedAt?: Date; - - @ApiProperty() - @CreateDateColumn() - createdAt: Date; - - @ApiProperty() - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + name: string; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + description?: string; + @ApiProperty({ enum: WorkflowStatus }) + @Column({ type: 'enum', enum: WorkflowStatus, default: WorkflowStatus.DRAFT }) + status: WorkflowStatus; + @OneToMany(() => AutomationTrigger, (trigger) => trigger.workflow, { cascade: true }) + triggers: AutomationTrigger[]; + @OneToMany(() => AutomationAction, (action) => action.workflow, { cascade: true }) + actions: AutomationAction[]; + @ApiProperty() + @Column({ default: 0 }) + executionCount: number; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + lastExecutedAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + activatedAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + deactivatedAt?: Date; + @ApiProperty() + @CreateDateColumn() + createdAt: Date; + @ApiProperty() + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/campaign-recipient.entity.ts b/src/email-marketing/entities/campaign-recipient.entity.ts index 4f637470..97bb6eb9 100644 --- a/src/email-marketing/entities/campaign-recipient.entity.ts +++ b/src/email-marketing/entities/campaign-recipient.entity.ts @@ -1,52 +1,34 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - JoinColumn, - Index, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn, Index, DeleteDateColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { Campaign } from './campaign.entity'; import { RecipientStatus } from '../enums/recipient-status.enum'; - @Entity('campaign_recipients') @Index(['campaignId', 'status']) export class CampaignRecipient { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - campaignId: string; - - @ManyToOne(() => Campaign, (campaign) => campaign.recipients, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'campaignId' }) - campaign: Campaign; - - @ApiProperty() - @Column() - userId: string; - - @ApiProperty() - @Column() - email: string; - - @ApiProperty({ enum: RecipientStatus }) - @Column({ type: 'enum', enum: RecipientStatus, default: RecipientStatus.PENDING }) - status: RecipientStatus; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - sentAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - variantId?: string; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + campaignId: string; + @ManyToOne(() => Campaign, (campaign) => campaign.recipients, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'campaignId' }) + campaign: Campaign; + @ApiProperty() + @Column() + userId: string; + @ApiProperty() + @Column() + email: string; + @ApiProperty({ enum: RecipientStatus }) + @Column({ type: 'enum', enum: RecipientStatus, default: RecipientStatus.PENDING }) + status: RecipientStatus; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + sentAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + variantId?: string; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/campaign.entity.ts b/src/email-marketing/entities/campaign.entity.ts index 87772367..5aeba691 100644 --- a/src/email-marketing/entities/campaign.entity.ts +++ b/src/email-marketing/entities/campaign.entity.ts @@ -1,93 +1,63 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - OneToMany, - OneToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, OneToMany, OneToOne, JoinColumn, Index, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { EmailTemplate } from './email-template.entity'; import { ABTest } from './ab-test.entity'; import { CampaignRecipient } from './campaign-recipient.entity'; import { CampaignStatus } from '../enums/campaign-status.enum'; - @Entity('email_campaigns') export class Campaign { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - @Index() - name: string; - - @ApiProperty() - @Column() - @Index() - subject: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - previewText?: string; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - content?: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - @Index() - templateId?: string; - - @ManyToOne(() => EmailTemplate, { nullable: true }) - @JoinColumn({ name: 'templateId' }) - template?: EmailTemplate; - - @ApiProperty({ type: [String] }) - @Column('simple-array', { nullable: true }) - segmentIds?: string[]; - - @ApiProperty({ enum: CampaignStatus }) - @Column({ type: 'enum', enum: CampaignStatus, default: CampaignStatus.DRAFT }) - @Index() - status: CampaignStatus; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - @Index() - scheduledAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - @Index() - sentAt?: Date; - - @ApiProperty() - @Column({ default: 0 }) - totalRecipients: number; - - @OneToOne(() => ABTest, (abTest) => abTest.campaign, { nullable: true }) - abTest?: ABTest; - - @OneToMany(() => CampaignRecipient, (recipient) => recipient.campaign) - recipients: CampaignRecipient[]; - - @ApiProperty() - @CreateDateColumn() - createdAt: Date; - - @ApiProperty() - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + @Index() + name: string; + @ApiProperty() + @Column() + @Index() + subject: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + previewText?: string; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + content?: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + @Index() + templateId?: string; + @ManyToOne(() => EmailTemplate, { nullable: true }) + @JoinColumn({ name: 'templateId' }) + template?: EmailTemplate; + @ApiProperty({ type: [String] }) + @Column('simple-array', { nullable: true }) + segmentIds?: string[]; + @ApiProperty({ enum: CampaignStatus }) + @Column({ type: 'enum', enum: CampaignStatus, default: CampaignStatus.DRAFT }) + @Index() + status: CampaignStatus; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + @Index() + scheduledAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + @Index() + sentAt?: Date; + @ApiProperty() + @Column({ default: 0 }) + totalRecipients: number; + @OneToOne(() => ABTest, (abTest) => abTest.campaign, { nullable: true }) + abTest?: ABTest; + @OneToMany(() => CampaignRecipient, (recipient) => recipient.campaign) + recipients: CampaignRecipient[]; + @ApiProperty() + @CreateDateColumn() + createdAt: Date; + @ApiProperty() + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/email-event.entity.ts b/src/email-marketing/entities/email-event.entity.ts index 632c2f6d..993e9a6a 100644 --- a/src/email-marketing/entities/email-event.entity.ts +++ b/src/email-marketing/entities/email-event.entity.ts @@ -1,33 +1,26 @@ import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, Index } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { EmailEventType } from '../enums/email-event-type.enum'; - @Entity('email_events') @Index(['campaignId', 'eventType']) @Index(['recipientId', 'eventType']) export class EmailEvent { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - campaignId: string; - - @ApiProperty() - @Column() - recipientId: string; - - @ApiProperty({ enum: EmailEventType }) - @Column({ type: 'enum', enum: EmailEventType }) - eventType: EmailEventType; - - @ApiProperty({ required: false }) - @Column({ type: 'jsonb', nullable: true }) - metadata?: Record; - - @ApiProperty() - @CreateDateColumn() - occurredAt: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + campaignId: string; + @ApiProperty() + @Column() + recipientId: string; + @ApiProperty({ enum: EmailEventType }) + @Column({ type: 'enum', enum: EmailEventType }) + eventType: EmailEventType; + @ApiProperty({ required: false }) + @Column({ type: 'jsonb', nullable: true }) + metadata?: Record; + @ApiProperty() + @CreateDateColumn() + occurredAt: Date; } diff --git a/src/email-marketing/entities/email-subscription.entity.ts b/src/email-marketing/entities/email-subscription.entity.ts index 60bcaaf6..ba86ee00 100644 --- a/src/email-marketing/entities/email-subscription.entity.ts +++ b/src/email-marketing/entities/email-subscription.entity.ts @@ -1,38 +1,30 @@ import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, Index } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - @Entity('email_subscriptions') @Index(['email'], { unique: true }) export class EmailSubscription { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column({ unique: true }) - email: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - userId?: string; - - @ApiProperty() - @Column({ default: true }) - isSubscribed: boolean; - - @ApiProperty({ type: [String] }) - @Column('simple-array', { nullable: true }) - preferences?: string[]; // e.g., ['marketing', 'product_updates', 'newsletters'] - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - unsubscribedAt?: Date; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - unsubscribeReason?: string; - - @ApiProperty() - @CreateDateColumn() - subscribedAt: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column({ unique: true }) + email: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + userId?: string; + @ApiProperty() + @Column({ default: true }) + isSubscribed: boolean; + @ApiProperty({ type: [String] }) + @Column('simple-array', { nullable: true }) + preferences?: string[]; // e.g., ['marketing', 'product_updates', 'newsletters'] + @ApiProperty({ required: false }) + @Column({ nullable: true }) + unsubscribedAt?: Date; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + unsubscribeReason?: string; + @ApiProperty() + @CreateDateColumn() + subscribedAt: Date; } diff --git a/src/email-marketing/entities/email-template.entity.ts b/src/email-marketing/entities/email-template.entity.ts index 3e3fbd94..36958a18 100644 --- a/src/email-marketing/entities/email-template.entity.ts +++ b/src/email-marketing/entities/email-template.entity.ts @@ -1,59 +1,40 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - @Entity('email_templates') export class EmailTemplate { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - name: string; - - @ApiProperty() - @Column() - subject: string; - - @ApiProperty() - @Column({ type: 'text' }) - htmlContent: string; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - textContent?: string; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - category?: string; - - @ApiProperty({ type: [String] }) - @Column('simple-array', { nullable: true }) - variables?: string[]; - - @ApiProperty({ required: false }) - @Column({ nullable: true }) - thumbnailUrl?: string; - - @ApiProperty() - @Column({ default: true }) - isActive: boolean; - - @ApiProperty() - @CreateDateColumn() - createdAt: Date; - - @ApiProperty() - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + name: string; + @ApiProperty() + @Column() + subject: string; + @ApiProperty() + @Column({ type: 'text' }) + htmlContent: string; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + textContent?: string; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + category?: string; + @ApiProperty({ type: [String] }) + @Column('simple-array', { nullable: true }) + variables?: string[]; + @ApiProperty({ required: false }) + @Column({ nullable: true }) + thumbnailUrl?: string; + @ApiProperty() + @Column({ default: true }) + isActive: boolean; + @ApiProperty() + @CreateDateColumn() + createdAt: Date; + @ApiProperty() + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/segment-rule.entity.ts b/src/email-marketing/entities/segment-rule.entity.ts index 51b86cc1..2c50787c 100644 --- a/src/email-marketing/entities/segment-rule.entity.ts +++ b/src/email-marketing/entities/segment-rule.entity.ts @@ -1,51 +1,34 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - JoinColumn, - DeleteDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn, DeleteDateColumn, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { Segment } from './segment.entity'; import { SegmentRuleOperator } from '../enums/segment-rule-operator.enum'; import { SegmentRuleField } from '../enums/segment-rule-field.enum'; - @Entity('segment_rules') export class SegmentRule { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - segmentId: string; - - @ManyToOne(() => Segment, (segment) => segment.rules, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'segmentId' }) - segment: Segment; - - @ApiProperty({ enum: SegmentRuleField }) - @Column({ type: 'enum', enum: SegmentRuleField }) - field: SegmentRuleField; - - @ApiProperty({ enum: SegmentRuleOperator }) - @Column({ type: 'enum', enum: SegmentRuleOperator }) - operator: SegmentRuleOperator; - - @ApiProperty() - @Column({ type: 'jsonb' }) - value: any; - - @ApiProperty() - @Column({ default: 0 }) - order: number; - - @ApiProperty({ default: 'AND' }) - @Column({ default: 'AND' }) - logicalOperator: 'AND' | 'OR'; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + segmentId: string; + @ManyToOne(() => Segment, (segment) => segment.rules, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'segmentId' }) + segment: Segment; + @ApiProperty({ enum: SegmentRuleField }) + @Column({ type: 'enum', enum: SegmentRuleField }) + field: SegmentRuleField; + @ApiProperty({ enum: SegmentRuleOperator }) + @Column({ type: 'enum', enum: SegmentRuleOperator }) + operator: SegmentRuleOperator; + @ApiProperty() + @Column({ type: 'jsonb' }) + value: unknown; + @ApiProperty() + @Column({ default: 0 }) + order: number; + @ApiProperty({ default: 'AND' }) + @Column({ default: 'AND' }) + logicalOperator: 'AND' | 'OR'; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/entities/segment.entity.ts b/src/email-marketing/entities/segment.entity.ts index 338e4acd..685cd6f2 100644 --- a/src/email-marketing/entities/segment.entity.ts +++ b/src/email-marketing/entities/segment.entity.ts @@ -1,52 +1,33 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - OneToMany, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, OneToMany, } from 'typeorm'; import { ApiProperty } from '@nestjs/swagger'; - import { SegmentRule } from './segment-rule.entity'; - @Entity('segments') export class Segment { - @ApiProperty() - @PrimaryGeneratedColumn('uuid') - id: string; - - @ApiProperty() - @Column() - name: string; - - @ApiProperty({ required: false }) - @Column({ type: 'text', nullable: true }) - description?: string; - - @ApiProperty() - @Column({ default: true }) - isDynamic: boolean; - - @OneToMany(() => SegmentRule, (rule) => rule.segment, { cascade: true }) - rules: SegmentRule[]; - - @ApiProperty({ type: [String] }) - @Column('simple-array', { nullable: true }) - staticMemberIds?: string[]; - - @ApiProperty() - memberCount?: number; // Calculated field - - @ApiProperty() - @CreateDateColumn() - createdAt: Date; - - @ApiProperty() - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @ApiProperty() + @PrimaryGeneratedColumn('uuid') + id: string; + @ApiProperty() + @Column() + name: string; + @ApiProperty({ required: false }) + @Column({ type: 'text', nullable: true }) + description?: string; + @ApiProperty() + @Column({ default: true }) + isDynamic: boolean; + @OneToMany(() => SegmentRule, (rule) => rule.segment, { cascade: true }) + rules: SegmentRule[]; + @ApiProperty({ type: [String] }) + @Column('simple-array', { nullable: true }) + staticMemberIds?: string[]; + @ApiProperty() + memberCount?: number; // Calculated field + @ApiProperty() + @CreateDateColumn() + createdAt: Date; + @ApiProperty() + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/email-marketing/enums/ab-test-status.enum.ts b/src/email-marketing/enums/ab-test-status.enum.ts index e3881a91..f0579455 100644 --- a/src/email-marketing/enums/ab-test-status.enum.ts +++ b/src/email-marketing/enums/ab-test-status.enum.ts @@ -1,6 +1,6 @@ export enum ABTestStatus { - DRAFT = 'draft', - RUNNING = 'running', - COMPLETED = 'completed', - CANCELLED = 'cancelled', + DRAFT = 'draft', + RUNNING = 'running', + COMPLETED = 'completed', + CANCELLED = 'cancelled' } diff --git a/src/email-marketing/enums/action-type.enum.ts b/src/email-marketing/enums/action-type.enum.ts index 00780238..0b70fd60 100644 --- a/src/email-marketing/enums/action-type.enum.ts +++ b/src/email-marketing/enums/action-type.enum.ts @@ -1,12 +1,12 @@ export enum ActionType { - SEND_EMAIL = 'send_email', - WAIT = 'wait', - ADD_TAG = 'add_tag', - REMOVE_TAG = 'remove_tag', - ADD_TO_SEGMENT = 'add_to_segment', - REMOVE_FROM_SEGMENT = 'remove_from_segment', - UPDATE_PROPERTY = 'update_property', - WEBHOOK = 'webhook', - SEND_SMS = 'send_sms', - SEND_PUSH_NOTIFICATION = 'send_push_notification', + SEND_EMAIL = 'send_email', + WAIT = 'wait', + ADD_TAG = 'add_tag', + REMOVE_TAG = 'remove_tag', + ADD_TO_SEGMENT = 'add_to_segment', + REMOVE_FROM_SEGMENT = 'remove_from_segment', + UPDATE_PROPERTY = 'update_property', + WEBHOOK = 'webhook', + SEND_SMS = 'send_sms', + SEND_PUSH_NOTIFICATION = 'send_push_notification' } diff --git a/src/email-marketing/enums/campaign-status.enum.ts b/src/email-marketing/enums/campaign-status.enum.ts index 7dd75505..a356f0b9 100644 --- a/src/email-marketing/enums/campaign-status.enum.ts +++ b/src/email-marketing/enums/campaign-status.enum.ts @@ -1,8 +1,8 @@ export enum CampaignStatus { - DRAFT = 'draft', - SCHEDULED = 'scheduled', - SENDING = 'sending', - SENT = 'sent', - PAUSED = 'paused', - CANCELLED = 'cancelled', + DRAFT = 'draft', + SCHEDULED = 'scheduled', + SENDING = 'sending', + SENT = 'sent', + PAUSED = 'paused', + CANCELLED = 'cancelled' } diff --git a/src/email-marketing/enums/email-event-type.enum.ts b/src/email-marketing/enums/email-event-type.enum.ts index 4139f533..22b9b892 100644 --- a/src/email-marketing/enums/email-event-type.enum.ts +++ b/src/email-marketing/enums/email-event-type.enum.ts @@ -1,10 +1,10 @@ export enum EmailEventType { - SENT = 'sent', - DELIVERED = 'delivered', - OPENED = 'opened', - CLICKED = 'clicked', - BOUNCED = 'bounced', - SOFT_BOUNCED = 'soft_bounced', - COMPLAINED = 'complained', - UNSUBSCRIBED = 'unsubscribed', + SENT = 'sent', + DELIVERED = 'delivered', + OPENED = 'opened', + CLICKED = 'clicked', + BOUNCED = 'bounced', + SOFT_BOUNCED = 'soft_bounced', + COMPLAINED = 'complained', + UNSUBSCRIBED = 'unsubscribed' } diff --git a/src/email-marketing/enums/recipient-status.enum.ts b/src/email-marketing/enums/recipient-status.enum.ts index 82ac7834..dfd00527 100644 --- a/src/email-marketing/enums/recipient-status.enum.ts +++ b/src/email-marketing/enums/recipient-status.enum.ts @@ -1,6 +1,6 @@ export enum RecipientStatus { - PENDING = 'pending', - SENT = 'sent', - FAILED = 'failed', - SKIPPED = 'skipped', + PENDING = 'pending', + SENT = 'sent', + FAILED = 'failed', + SKIPPED = 'skipped' } diff --git a/src/email-marketing/enums/segment-rule-field.enum.ts b/src/email-marketing/enums/segment-rule-field.enum.ts index 585cbd68..b10d4a01 100644 --- a/src/email-marketing/enums/segment-rule-field.enum.ts +++ b/src/email-marketing/enums/segment-rule-field.enum.ts @@ -1,12 +1,12 @@ export enum SegmentRuleField { - EMAIL = 'email', - FIRST_NAME = 'first_name', - LAST_NAME = 'last_name', - CREATED_AT = 'created_at', - LAST_LOGIN = 'last_login', - COURSE_COUNT = 'course_count', - TOTAL_SPENT = 'total_spent', - TAG = 'tag', - COUNTRY = 'country', - SUBSCRIPTION_STATUS = 'subscription_status', + EMAIL = 'email', + FIRST_NAME = 'first_name', + LAST_NAME = 'last_name', + CREATED_AT = 'created_at', + LAST_LOGIN = 'last_login', + COURSE_COUNT = 'course_count', + TOTAL_SPENT = 'total_spent', + TAG = 'tag', + COUNTRY = 'country', + SUBSCRIPTION_STATUS = 'subscription_status' } diff --git a/src/email-marketing/enums/segment-rule-operator.enum.ts b/src/email-marketing/enums/segment-rule-operator.enum.ts index 69d75fed..bed35553 100644 --- a/src/email-marketing/enums/segment-rule-operator.enum.ts +++ b/src/email-marketing/enums/segment-rule-operator.enum.ts @@ -1,19 +1,19 @@ export enum SegmentRuleOperator { - EQUALS = 'equals', - NOT_EQUALS = 'not_equals', - CONTAINS = 'contains', - NOT_CONTAINS = 'not_contains', - STARTS_WITH = 'starts_with', - ENDS_WITH = 'ends_with', - GREATER_THAN = 'greater_than', - LESS_THAN = 'less_than', - GREATER_OR_EQUAL = 'greater_or_equal', - LESS_OR_EQUAL = 'less_or_equal', - IS_SET = 'is_set', - IS_NOT_SET = 'is_not_set', - IN_LIST = 'in_list', - NOT_IN_LIST = 'not_in_list', - BEFORE = 'before', - AFTER = 'after', - BETWEEN = 'between', + EQUALS = 'equals', + NOT_EQUALS = 'not_equals', + CONTAINS = 'contains', + NOT_CONTAINS = 'not_contains', + STARTS_WITH = 'starts_with', + ENDS_WITH = 'ends_with', + GREATER_THAN = 'greater_than', + LESS_THAN = 'less_than', + GREATER_OR_EQUAL = 'greater_or_equal', + LESS_OR_EQUAL = 'less_or_equal', + IS_SET = 'is_set', + IS_NOT_SET = 'is_not_set', + IN_LIST = 'in_list', + NOT_IN_LIST = 'not_in_list', + BEFORE = 'before', + AFTER = 'after', + BETWEEN = 'between' } diff --git a/src/email-marketing/enums/trigger-type.enum.ts b/src/email-marketing/enums/trigger-type.enum.ts index fd4012ad..982efcb8 100644 --- a/src/email-marketing/enums/trigger-type.enum.ts +++ b/src/email-marketing/enums/trigger-type.enum.ts @@ -1,14 +1,14 @@ export enum TriggerType { - USER_SIGNUP = 'user_signup', - COURSE_ENROLLED = 'course_enrolled', - COURSE_COMPLETED = 'course_completed', - PURCHASE_MADE = 'purchase_made', - USER_INACTIVE = 'user_inactive', - SUBSCRIPTION_CREATED = 'subscription_created', - SUBSCRIPTION_CANCELLED = 'subscription_cancelled', - BIRTHDAY = 'birthday', - CUSTOM_EVENT = 'custom_event', - DATE_BASED = 'date_based', - SEGMENT_ENTERED = 'segment_entered', - SEGMENT_LEFT = 'segment_left', + USER_SIGNUP = 'user_signup', + COURSE_ENROLLED = 'course_enrolled', + COURSE_COMPLETED = 'course_completed', + PURCHASE_MADE = 'purchase_made', + USER_INACTIVE = 'user_inactive', + SUBSCRIPTION_CREATED = 'subscription_created', + SUBSCRIPTION_CANCELLED = 'subscription_cancelled', + BIRTHDAY = 'birthday', + CUSTOM_EVENT = 'custom_event', + DATE_BASED = 'date_based', + SEGMENT_ENTERED = 'segment_entered', + SEGMENT_LEFT = 'segment_left' } diff --git a/src/email-marketing/enums/workflow-status.enum.ts b/src/email-marketing/enums/workflow-status.enum.ts index d8179562..758b92a4 100644 --- a/src/email-marketing/enums/workflow-status.enum.ts +++ b/src/email-marketing/enums/workflow-status.enum.ts @@ -1,6 +1,6 @@ export enum WorkflowStatus { - DRAFT = 'draft', - ACTIVE = 'active', - INACTIVE = 'inactive', - ARCHIVED = 'archived', + DRAFT = 'draft', + ACTIVE = 'active', + INACTIVE = 'inactive', + ARCHIVED = 'archived' } diff --git a/src/email-marketing/processors/email-queue.processor.ts b/src/email-marketing/processors/email-queue.processor.ts index d6108f9f..09d68aad 100644 --- a/src/email-marketing/processors/email-queue.processor.ts +++ b/src/email-marketing/processors/email-queue.processor.ts @@ -4,7 +4,6 @@ import { Job } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; - import { Campaign } from '../entities/campaign.entity'; import { CampaignRecipient } from '../entities/campaign-recipient.entity'; import { EmailSenderService } from '../sender/email-sender.service'; @@ -12,190 +11,156 @@ import { SegmentationService } from '../segmentation/segmentation.service'; import { ABTestingService } from '../ab-testing/ab-testing.service'; import { CampaignStatus } from '../enums/campaign-status.enum'; import { RecipientStatus } from '../enums/recipient-status.enum'; - @Processor(QUEUE_NAMES.EMAIL_MARKETING) export class EmailQueueProcessor { - private readonly logger = new Logger(EmailQueueProcessor.name); - - constructor( + private readonly logger = new Logger(EmailQueueProcessor.name); + constructor( @InjectRepository(Campaign) - private readonly campaignRepository: Repository, + private readonly campaignRepository: Repository, @InjectRepository(CampaignRecipient) - private readonly recipientRepository: Repository, - private readonly emailSenderService: EmailSenderService, - private readonly segmentationService: SegmentationService, - private readonly abTestingService: ABTestingService, - ) {} - - @Process(JOB_NAMES.SEND_CAMPAIGN) - async handleScheduledCampaign(job: Job<{ campaignId: string }>) { - this.logger.log(`Processing scheduled campaign: ${job.data.campaignId}`); - - const campaign = await this.campaignRepository.findOne({ - where: { id: job.data.campaignId }, - relations: ['abTest', 'abTest.variants'], - }); - - if (!campaign || campaign.status !== CampaignStatus.SCHEDULED) { - this.logger.warn(`Campaign ${job.data.campaignId} not found or not scheduled`); - return; + private readonly recipientRepository: Repository, private readonly emailSenderService: EmailSenderService, private readonly segmentationService: SegmentationService, private readonly abTestingService: ABTestingService) { } + @Process(JOB_NAMES.SEND_CAMPAIGN) + async handleScheduledCampaign(job: Job<{ + campaignId: string; + }>) { + this.logger.log(`Processing scheduled campaign: ${job.data.campaignId}`); + const campaign = await this.campaignRepository.findOne({ + where: { id: job.data.campaignId }, + relations: ['abTest', 'abTest.variants'], + }); + if (!campaign || campaign.status !== CampaignStatus.SCHEDULED) { + this.logger.warn(`Campaign ${job.data.campaignId} not found or not scheduled`); + return; + } + // Get recipients + const users = await this.segmentationService.getUsersFromSegments(campaign.segmentIds || []); + campaign.status = CampaignStatus.SENDING; + campaign.sentAt = new Date(); + campaign.totalRecipients = users.length; + await this.campaignRepository.save(campaign); + await this.processRecipients(campaign, users); } - - // Get recipients - const users = await this.segmentationService.getUsersFromSegments(campaign.segmentIds || []); - - campaign.status = CampaignStatus.SENDING; - campaign.sentAt = new Date(); - campaign.totalRecipients = users.length; - await this.campaignRepository.save(campaign); - - await this.processRecipients(campaign, users); - } - - @Process(JOB_NAMES.PROCESS_CAMPAIGN) - async handleCampaignProcessing(job: Job<{ campaignId: string; recipients: string[] }>) { - this.logger.log(`Processing campaign: ${job.data.campaignId}`); - - const campaign = await this.campaignRepository.findOne({ - where: { id: job.data.campaignId }, - relations: ['abTest', 'abTest.variants'], - }); - - if (!campaign) { - return; + @Process(JOB_NAMES.PROCESS_CAMPAIGN) + async handleCampaignProcessing(job: Job<{ + campaignId: string; + recipients: string[]; + }>) { + this.logger.log(`Processing campaign: ${job.data.campaignId}`); + const campaign = await this.campaignRepository.findOne({ + where: { id: job.data.campaignId }, + relations: ['abTest', 'abTest.variants'], + }); + if (!campaign) { + return; + } + // Fetch user details for recipients + // TODO: Integrate with Users module + const users = job.data.recipients.map((id) => ({ + id, + email: `user-${id}@example.com`, // Placeholder + })); + await this.processRecipients(campaign, users); } - - // Fetch user details for recipients - // TODO: Integrate with Users module - const users = job.data.recipients.map((id) => ({ - id, - email: `user-${id}@example.com`, // Placeholder - })); - - await this.processRecipients(campaign, users); - } - - @Process(JOB_NAMES.SEND_AUTOMATION_EMAIL) - async handleAutomationEmail( - job: Job<{ - actionId: string; - templateId: string; - userId: string; - variables: Record; - }>, - ) { - this.logger.log(`Sending automation email for action: ${job.data.actionId}`); - - // TODO: Get user email from Users module - const userEmail = `user-${job.data.userId}@example.com`; - - await this.emailSenderService.sendEmail({ - to: userEmail, - templateId: job.data.templateId, - variables: job.data.variables, - trackOpens: true, - trackClicks: true, - }); - } - - @Process(JOB_NAMES.RESUME_CAMPAIGN) - async handleResumeCampaign(job: Job<{ campaignId: string }>) { - this.logger.log(`Resuming campaign: ${job.data.campaignId}`); - - const pendingRecipients = await this.recipientRepository.find({ - where: { - campaignId: job.data.campaignId, - status: RecipientStatus.PENDING, - }, - }); - - const campaign = await this.campaignRepository.findOne({ - where: { id: job.data.campaignId }, - relations: ['abTest', 'abTest.variants'], - }); - - if (!campaign) return; - - for (const recipient of pendingRecipients) { - await this.sendToRecipient(campaign, recipient); + @Process(JOB_NAMES.SEND_AUTOMATION_EMAIL) + async handleAutomationEmail(job: Job<{ + actionId: string; + templateId: string; + userId: string; + variables: Record; + }>) { + this.logger.log(`Sending automation email for action: ${job.data.actionId}`); + // TODO: Get user email from Users module + const userEmail = `user-${job.data.userId}@example.com`; + await this.emailSenderService.sendEmail({ + to: userEmail, + templateId: job.data.templateId, + variables: job.data.variables, + trackOpens: true, + trackClicks: true, + }); } - - await this.finalizeCampaign(job.data.campaignId); - } - - // Private helper methods - private async processRecipients( - campaign: Campaign, - users: Array<{ id: string; email: string }>, - ): Promise { - // Create recipient records - const recipients = users.map((user) => - this.recipientRepository.create({ - campaignId: campaign.id, - userId: user.id, - email: user.email, - status: RecipientStatus.PENDING, - }), - ); - - await this.recipientRepository.save(recipients); - - // Process in batches - const batchSize = 100; - for (let i = 0; i < recipients.length; i += batchSize) { - const batch = recipients.slice(i, i + batchSize); - - await Promise.all(batch.map((recipient) => this.sendToRecipient(campaign, recipient))); - - // Update progress - const progress = Math.round(((i + batch.length) / recipients.length) * 100); - this.logger.log(`Campaign ${campaign.id} progress: ${progress}%`); + @Process(JOB_NAMES.RESUME_CAMPAIGN) + async handleResumeCampaign(job: Job<{ + campaignId: string; + }>) { + this.logger.log(`Resuming campaign: ${job.data.campaignId}`); + const pendingRecipients = await this.recipientRepository.find({ + where: { + campaignId: job.data.campaignId, + status: RecipientStatus.PENDING, + }, + }); + const campaign = await this.campaignRepository.findOne({ + where: { id: job.data.campaignId }, + relations: ['abTest', 'abTest.variants'], + }); + if (!campaign) + return; + for (const recipient of pendingRecipients) { + await this.sendToRecipient(campaign, recipient); + } + await this.finalizeCampaign(job.data.campaignId); } - - await this.finalizeCampaign(campaign.id); - } - - private async sendToRecipient(campaign: Campaign, recipient: CampaignRecipient): Promise { - try { - // Select A/B test variant if applicable - let variantId: string | undefined; - let templateId = campaign.templateId; - - if (campaign.abTest) { - const variant = this.abTestingService.selectVariantForRecipient(campaign.abTest); - variantId = variant.id; - templateId = variant.templateId || templateId; - } - - const result = await this.emailSenderService.sendEmail({ - to: recipient.email, - templateId, - variables: { userId: recipient.userId }, - campaignId: campaign.id, - recipientId: recipient.id, - variantId, - trackOpens: true, - trackClicks: true, - }); - - recipient.status = result.success ? RecipientStatus.SENT : RecipientStatus.FAILED; - recipient.sentAt = new Date(); - - await this.recipientRepository.save(recipient); - } catch (error) { - this.logger.error(`Failed to send to ${recipient.email}:`, error); - recipient.status = RecipientStatus.FAILED; - await this.recipientRepository.save(recipient); + // Private helper methods + private async processRecipients(campaign: Campaign, users: Array<{ + id: string; + email: string; + }>): Promise { + // Create recipient records + const recipients = users.map((user) => this.recipientRepository.create({ + campaignId: campaign.id, + userId: user.id, + email: user.email, + status: RecipientStatus.PENDING, + })); + await this.recipientRepository.save(recipients); + // Process in batches + const batchSize = 100; + for (let i = 0; i < recipients.length; i += batchSize) { + const batch = recipients.slice(i, i + batchSize); + await Promise.all(batch.map((recipient) => this.sendToRecipient(campaign, recipient))); + // Update progress + const progress = Math.round(((i + batch.length) / recipients.length) * 100); + this.logger.log(`Campaign ${campaign.id} progress: ${progress}%`); + } + await this.finalizeCampaign(campaign.id); } - } - - private async finalizeCampaign(campaignId: string): Promise { - const campaign = await this.campaignRepository.findOne({ where: { id: campaignId } }); - - if (campaign && campaign.status === CampaignStatus.SENDING) { - campaign.status = CampaignStatus.SENT; - await this.campaignRepository.save(campaign); - this.logger.log(`Campaign ${campaignId} completed`); + private async sendToRecipient(campaign: Campaign, recipient: CampaignRecipient): Promise { + try { + // Select A/B test variant if applicable + let variantId: string | undefined; + let templateId = campaign.templateId; + if (campaign.abTest) { + const variant = this.abTestingService.selectVariantForRecipient(campaign.abTest); + variantId = variant.id; + templateId = variant.templateId || templateId; + } + const result = await this.emailSenderService.sendEmail({ + to: recipient.email, + templateId, + variables: { userId: recipient.userId }, + campaignId: campaign.id, + recipientId: recipient.id, + variantId, + trackOpens: true, + trackClicks: true, + }); + recipient.status = result.success ? RecipientStatus.SENT : RecipientStatus.FAILED; + recipient.sentAt = new Date(); + await this.recipientRepository.save(recipient); + } + catch (error) { + this.logger.error(`Failed to send to ${recipient.email}:`, error); + recipient.status = RecipientStatus.FAILED; + await this.recipientRepository.save(recipient); + } + } + private async finalizeCampaign(campaignId: string): Promise { + const campaign = await this.campaignRepository.findOne({ where: { id: campaignId } }); + if (campaign && campaign.status === CampaignStatus.SENDING) { + campaign.status = CampaignStatus.SENT; + await this.campaignRepository.save(campaign); + this.logger.log(`Campaign ${campaignId} completed`); + } } - } } diff --git a/src/email-marketing/segmentation/segment.controller.ts b/src/email-marketing/segmentation/segment.controller.ts index 12e499e8..1b52dbd3 100644 --- a/src/email-marketing/segmentation/segment.controller.ts +++ b/src/email-marketing/segmentation/segment.controller.ts @@ -1,91 +1,88 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Body, - Param, - Query, - ParseUUIDPipe, - HttpCode, - HttpStatus, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Body, Param, Query, ParseUUIDPipe, HttpCode, HttpStatus, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { SegmentationService } from './segmentation.service'; import { CreateSegmentDto } from '../dto/create-segment.dto'; import { UpdateSegmentDto } from '../dto/update-segment.dto'; import { AddSegmentMembersDto } from '../dto/add-segment-members.dto'; import { Segment } from '../entities/segment.entity'; - @ApiTags('Email Marketing - Segments') @ApiBearerAuth() @Controller('email-marketing/segments') export class SegmentController { - constructor(private readonly segmentationService: SegmentationService) {} - - @Post() - @ApiOperation({ summary: 'Create a new audience segment' }) - @ApiResponse({ status: 201, description: 'Segment created successfully' }) - async create(@Body() createSegmentDto: CreateSegmentDto): Promise { - return this.segmentationService.create(createSegmentDto); - } - - @Get() - @ApiOperation({ summary: 'Get all segments with pagination' }) - @ApiQuery({ name: 'page', required: false, type: Number }) - @ApiQuery({ name: 'limit', required: false, type: Number }) - async findAll(@Query('page') page = 1, @Query('limit') limit = 10) { - return this.segmentationService.findAll(page, limit); - } - - @Get(':id') - @ApiOperation({ summary: 'Get a segment by ID' }) - @ApiResponse({ status: 404, description: 'Segment not found' }) - async findOne(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.segmentationService.findOne(id); - } - - @Put(':id') - @ApiOperation({ summary: 'Update a segment' }) - async update( - @Param('id', ParseUUIDPipe) id: string, - @Body() updateSegmentDto: UpdateSegmentDto, - ): Promise { - return this.segmentationService.update(id, updateSegmentDto); - } - - @Delete(':id') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete a segment' }) - async remove(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.segmentationService.remove(id); - } - - @Get(':id/members') - @ApiOperation({ summary: 'Get members of a segment' }) - async getMembers(@Param('id', ParseUUIDPipe) id: string) { - return this.segmentationService.getSegmentMembers(id); - } - - @Post(':id/members') - @ApiOperation({ summary: 'Add users to a static segment' }) - @ApiResponse({ status: 200, description: 'Users added successfully' }) - async addMembers( - @Param('id', ParseUUIDPipe) id: string, - @Body() addMembersDto: AddSegmentMembersDto, - ): Promise<{ message: string; addedCount: number }> { - await this.segmentationService.addUsersToSegment(id, addMembersDto.userIds); - return { - message: 'Users added successfully', - addedCount: addMembersDto.userIds.length, - }; - } - - @Post('preview') - @ApiOperation({ summary: 'Preview segment members without saving' }) - async preview(@Body() createSegmentDto: CreateSegmentDto) { - return this.segmentationService.previewSegment(createSegmentDto.rules); - } + constructor(private readonly segmentationService: SegmentationService) { } + @Post() + @ApiOperation({ summary: 'Create a new audience segment' }) + @ApiResponse({ status: 201, description: 'Segment created successfully' }) + async create( + @Body() + createSegmentDto: CreateSegmentDto): Promise { + return this.segmentationService.create(createSegmentDto); + } + @Get() + @ApiOperation({ summary: 'Get all segments with pagination' }) + @ApiQuery({ name: 'page', required: false, type: Number }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + async findAll( + @Query('page') + page = 1, + @Query('limit') + limit = 10) { + return this.segmentationService.findAll(page, limit); + } + @Get(':id') + @ApiOperation({ summary: 'Get a segment by ID' }) + @ApiResponse({ status: 404, description: 'Segment not found' }) + async findOne( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.segmentationService.findOne(id); + } + @Put(':id') + @ApiOperation({ summary: 'Update a segment' }) + async update( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + updateSegmentDto: UpdateSegmentDto): Promise { + return this.segmentationService.update(id, updateSegmentDto); + } + @Delete(':id') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete a segment' }) + async remove( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.segmentationService.remove(id); + } + @Get(':id/members') + @ApiOperation({ summary: 'Get members of a segment' }) + async getMembers( + @Param('id', ParseUUIDPipe) + id: string) { + return this.segmentationService.getSegmentMembers(id); + } + @Post(':id/members') + @ApiOperation({ summary: 'Add users to a static segment' }) + @ApiResponse({ status: 200, description: 'Users added successfully' }) + async addMembers( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + addMembersDto: AddSegmentMembersDto): Promise<{ + message: string; + addedCount: number; + }> { + await this.segmentationService.addUsersToSegment(id, addMembersDto.userIds); + return { + message: 'Users added successfully', + addedCount: addMembersDto.userIds.length, + }; + } + @Post('preview') + @ApiOperation({ summary: 'Preview segment members without saving' }) + async preview( + @Body() + createSegmentDto: CreateSegmentDto) { + return this.segmentationService.previewSegment(createSegmentDto.rules); + } } diff --git a/src/email-marketing/segmentation/segmentation.service.ts b/src/email-marketing/segmentation/segmentation.service.ts index 6ecc1769..41e441e8 100644 --- a/src/email-marketing/segmentation/segmentation.service.ts +++ b/src/email-marketing/segmentation/segmentation.service.ts @@ -1,465 +1,384 @@ import { Injectable, NotFoundException, BadRequestException } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository, In } from 'typeorm'; - import { Segment } from '../entities/segment.entity'; import { SegmentRule } from '../entities/segment-rule.entity'; import { CreateSegmentDto } from '../dto/create-segment.dto'; import { UpdateSegmentDto } from '../dto/update-segment.dto'; import { SegmentRuleOperator } from '../enums/segment-rule-operator.enum'; import { SegmentRuleField } from '../enums/segment-rule-field.enum'; - // Note: Import User entity from users module when integrating export interface UserProfile { - id: string; - email: string; - firstName?: string; - lastName?: string; - createdAt: Date; - lastLoginAt?: Date; - tags?: string[]; - preferences?: Record; + id: string; + email: string; + firstName?: string; + lastName?: string; + createdAt: Date; + lastLoginAt?: Date; + tags?: string[]; + preferences?: Record; } - @Injectable() export class SegmentationService { - constructor( + constructor( @InjectRepository(Segment) - private readonly segmentRepository: Repository, + private readonly segmentRepository: Repository, @InjectRepository(SegmentRule) - private readonly ruleRepository: Repository, - ) {} - - /** - * Create a new segment - */ - async create(createSegmentDto: CreateSegmentDto): Promise { - const segment = this.segmentRepository.create({ - name: createSegmentDto.name, - description: createSegmentDto.description, - isDynamic: createSegmentDto.isDynamic ?? true, - }); - - const savedSegment = await this.segmentRepository.save(segment); - - // Create rules - if (createSegmentDto.rules?.length) { - const rules = createSegmentDto.rules.map((rule, index) => - this.ruleRepository.create({ - ...rule, - segmentId: savedSegment.id, - order: index, - }), - ); - await this.ruleRepository.save(rules); + private readonly ruleRepository: Repository) { } + /** + * Create a new segment + */ + async create(createSegmentDto: CreateSegmentDto): Promise { + const segment = this.segmentRepository.create({ + name: createSegmentDto.name, + description: createSegmentDto.description, + isDynamic: createSegmentDto.isDynamic ?? true, + }); + const savedSegment = await this.segmentRepository.save(segment); + // Create rules + if (createSegmentDto.rules?.length) { + const rules = createSegmentDto.rules.map((rule, index) => this.ruleRepository.create({ + ...rule, + segmentId: savedSegment.id, + order: index, + })); + await this.ruleRepository.save(rules); + } + return this.findOne(savedSegment.id); } - - return this.findOne(savedSegment.id); - } - - /** - * Get all segments - */ - async findAll( - page: number = 1, - limit: number = 10, - ): Promise<{ - segments: Segment[]; - total: number; - page: number; - totalPages: number; - }> { - const [segments, total] = await this.segmentRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - relations: ['rules'], - }); - - // Calculate member count for each segment using the segment object already loaded with rules - for (const segment of segments) { - segment.memberCount = await this.calculateMemberCountForSegment(segment); + /** + * Get all segments + */ + async findAll(page: number = 1, limit: number = 10): Promise<{ + segments: Segment[]; + total: number; + page: number; + totalPages: number; + }> { + const [segments, total] = await this.segmentRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + relations: ['rules'], + }); + // Calculate member count for each segment using the segment object already loaded with rules + for (const segment of segments) { + segment.memberCount = await this.calculateMemberCountForSegment(segment); + } + return { + segments, + total, + page, + totalPages: Math.ceil(total / limit), + }; } - - return { - segments, - total, - page, - totalPages: Math.ceil(total / limit), - }; - } - - /** - * Get a single segment by ID - */ - async findOne(id: string): Promise { - const segment = await this.segmentRepository.findOne({ - where: { id }, - relations: ['rules'], - }); - - if (!segment) { - throw new NotFoundException(`Segment with ID ${id} not found`); + /** + * Get a single segment by ID + */ + async findOne(id: string): Promise { + const segment = await this.segmentRepository.findOne({ + where: { id }, + relations: ['rules'], + }); + if (!segment) { + throw new NotFoundException(`Segment with ID ${id} not found`); + } + segment.memberCount = await this.calculateMemberCountForSegment(segment); + return segment; } - - segment.memberCount = await this.calculateMemberCountForSegment(segment); - return segment; - } - - /** - * Find multiple segments by IDs - */ - async findByIds(ids: string[]): Promise { - if (!ids.length) return []; - return this.segmentRepository.find({ - where: { id: In(ids) }, - relations: ['rules'], - }); - } - - /** - * Update a segment - */ - async update(id: string, updateSegmentDto: UpdateSegmentDto): Promise { - const segment = await this.findOne(id); - - Object.assign(segment, { - name: updateSegmentDto.name ?? segment.name, - description: updateSegmentDto.description ?? segment.description, - isDynamic: updateSegmentDto.isDynamic ?? segment.isDynamic, - }); - - await this.segmentRepository.save(segment); - - // Update rules if provided - if (updateSegmentDto.rules) { - await this.ruleRepository.softDelete({ segmentId: id }); - const rules = updateSegmentDto.rules.map((rule, index) => - this.ruleRepository.create({ - ...rule, - segmentId: id, - order: index, - }), - ); - await this.ruleRepository.save(rules); + /** + * Find multiple segments by IDs + */ + async findByIds(ids: string[]): Promise { + if (!ids.length) + return []; + return this.segmentRepository.find({ + where: { id: In(ids) }, + relations: ['rules'], + }); } - - return this.findOne(id); - } - - /** - * Delete a segment - */ - async remove(id: string): Promise { - await this.findOne(id); - await this.segmentRepository.manager.transaction(async (manager) => { - await manager.getRepository(SegmentRule).softDelete({ segmentId: id }); - await manager.getRepository(Segment).softDelete(id); - }); - } - - /** - * Get users from multiple segments - */ - async getUsersFromSegments(segmentIds: string[]): Promise { - if (!segmentIds.length) { - return []; + /** + * Update a segment + */ + async update(id: string, updateSegmentDto: UpdateSegmentDto): Promise { + const segment = await this.findOne(id); + Object.assign(segment, { + name: updateSegmentDto.name ?? segment.name, + description: updateSegmentDto.description ?? segment.description, + isDynamic: updateSegmentDto.isDynamic ?? segment.isDynamic, + }); + await this.segmentRepository.save(segment); + // Update rules if provided + if (updateSegmentDto.rules) { + await this.ruleRepository.softDelete({ segmentId: id }); + const rules = updateSegmentDto.rules.map((rule, index) => this.ruleRepository.create({ + ...rule, + segmentId: id, + order: index, + })); + await this.ruleRepository.save(rules); + } + return this.findOne(id); } - - const segments = await this.segmentRepository.find({ - where: { id: In(segmentIds) }, - relations: ['rules'], - }); - - const userSets = await Promise.all(segments.map((segment) => this.getSegmentMembers(segment))); - - // Combine all users (union) - const userMap = new Map(); - for (const users of userSets) { - for (const user of users) { - userMap.set(user.id, user); - } + /** + * Delete a segment + */ + async remove(id: string): Promise { + await this.findOne(id); + await this.segmentRepository.manager.transaction(async (manager) => { + await manager.getRepository(SegmentRule).softDelete({ segmentId: id }); + await manager.getRepository(Segment).softDelete(id); + }); } - - return Array.from(userMap.values()); - } - - /** - * Get members of a specific segment - */ - async getSegmentMembers(segmentOrId: Segment | string): Promise { - let segment: Segment; - - if (typeof segmentOrId === 'string') { - // Direct query to avoid infinite recursion - const found = await this.segmentRepository.findOne({ - where: { id: segmentOrId }, - relations: ['rules'], - }); - if (!found) { - throw new NotFoundException(`Segment with ID ${segmentOrId} not found`); - } - segment = found; - } else { - segment = segmentOrId; + /** + * Get users from multiple segments + */ + async getUsersFromSegments(segmentIds: string[]): Promise { + if (!segmentIds.length) { + return []; + } + const segments = await this.segmentRepository.find({ + where: { id: In(segmentIds) }, + relations: ['rules'], + }); + const userSets = await Promise.all(segments.map((segment) => this.getSegmentMembers(segment))); + // Combine all users (union) + const userMap = new Map(); + for (const users of userSets) { + for (const user of users) { + userMap.set(user.id, user); + } + } + return Array.from(userMap.values()); } - - if (!segment.isDynamic) { - // Static segment - return manually added members - return this.getStaticSegmentMembers(segment.id); + /** + * Get members of a specific segment + */ + async getSegmentMembers(segmentOrId: Segment | string): Promise { + let segment: Segment; + if (typeof segmentOrId === 'string') { + // Direct query to avoid infinite recursion + const found = await this.segmentRepository.findOne({ + where: { id: segmentOrId }, + relations: ['rules'], + }); + if (!found) { + throw new NotFoundException(`Segment with ID ${segmentOrId} not found`); + } + segment = found; + } + else { + segment = segmentOrId; + } + if (!segment.isDynamic) { + // Static segment - return manually added members + return this.getStaticSegmentMembers(segment.id); + } + // Dynamic segment - evaluate rules + return this.evaluateSegmentRules(segment.rules); } - - // Dynamic segment - evaluate rules - return this.evaluateSegmentRules(segment.rules); - } - - /** - * Preview segment members without saving - */ - async previewSegment(rules: CreateSegmentDto['rules']): Promise<{ - count: number; - sample: UserProfile[]; - }> { - const ruleEntities = rules.map((rule, index) => - this.ruleRepository.create({ ...rule, order: index }), - ); - - const members = await this.evaluateSegmentRules(ruleEntities); - - return { - count: members.length, - sample: members.slice(0, 10), - }; - } - - /** - * Add users manually to a static segment - */ - async addUsersToSegment(segmentId: string, userIds: string[]): Promise { - const segment = await this.findOne(segmentId); - - if (segment.isDynamic) { - throw new BadRequestException('Cannot manually add users to a dynamic segment'); + /** + * Preview segment members without saving + */ + async previewSegment(rules: CreateSegmentDto['rules']): Promise<{ + count: number; + sample: UserProfile[]; + }> { + const ruleEntities = rules.map((rule, index) => this.ruleRepository.create({ ...rule, order: index })); + const members = await this.evaluateSegmentRules(ruleEntities); + return { + count: members.length, + sample: members.slice(0, 10), + }; } - - // Add to static member list - const currentMembers = segment.staticMemberIds || []; - const newMembers = [...new Set([...currentMembers, ...userIds])]; - - await this.segmentRepository.update(segmentId, { - staticMemberIds: newMembers, - }); - } - - /** - * Remove users from a static segment - */ - async removeUsersFromSegment(segmentId: string, userIds: string[]): Promise { - const segment = await this.findOne(segmentId); - - if (segment.isDynamic) { - throw new BadRequestException('Cannot manually remove users from a dynamic segment'); + /** + * Add users manually to a static segment + */ + async addUsersToSegment(segmentId: string, userIds: string[]): Promise { + const segment = await this.findOne(segmentId); + if (segment.isDynamic) { + throw new BadRequestException('Cannot manually add users to a dynamic segment'); + } + // Add to static member list + const currentMembers = segment.staticMemberIds || []; + const newMembers = [...new Set([...currentMembers, ...userIds])]; + await this.segmentRepository.update(segmentId, { + staticMemberIds: newMembers, + }); } - - const currentMembers = segment.staticMemberIds || []; - const newMembers = currentMembers.filter((id) => !userIds.includes(id)); - - await this.segmentRepository.update(segmentId, { - staticMemberIds: newMembers, - }); - } - - /** - * Check if a user belongs to a segment - */ - async isUserInSegment(userId: string, segmentOrId: Segment | string): Promise { - const members = await this.getSegmentMembers(segmentOrId); - return members.some((member) => member.id === userId); - } - - /** - * Get all segments a user belongs to - */ - async getUserSegments(userId: string): Promise { - const allSegments = await this.segmentRepository.find({ - relations: ['rules'], - }); - - const userSegments: Segment[] = []; - - for (const segment of allSegments) { - if (await this.isUserInSegment(userId, segment)) { - userSegments.push(segment); - } + /** + * Remove users from a static segment + */ + async removeUsersFromSegment(segmentId: string, userIds: string[]): Promise { + const segment = await this.findOne(segmentId); + if (segment.isDynamic) { + throw new BadRequestException('Cannot manually remove users from a dynamic segment'); + } + const currentMembers = segment.staticMemberIds || []; + const newMembers = currentMembers.filter((id) => !userIds.includes(id)); + await this.segmentRepository.update(segmentId, { + staticMemberIds: newMembers, + }); } - - return userSegments; - } - - // ==================== Private Helper Methods ==================== - - /** - * Calculate member count for a segment (by ID - may cause recursion, use carefully) - */ - private async calculateMemberCount(segmentId: string): Promise { - const members = await this.getSegmentMembers(segmentId); - return members.length; - } - - /** - * Calculate member count for a segment (accepts segment object to avoid recursion) - */ - private async calculateMemberCountForSegment(segment: Segment): Promise { - if (!segment.isDynamic) { - return segment.staticMemberIds?.length || 0; + /** + * Check if a user belongs to a segment + */ + async isUserInSegment(userId: string, segmentOrId: Segment | string): Promise { + const members = await this.getSegmentMembers(segmentOrId); + return members.some((member) => member.id === userId); } - const members = await this.evaluateSegmentRules(segment.rules); - return members.length; - } - - /** - * Get members of a static segment - */ - private async getStaticSegmentMembers(segmentId: string): Promise { - const segment = await this.segmentRepository.findOne({ - where: { id: segmentId }, - }); - - if (!segment?.staticMemberIds?.length) { - return []; + /** + * Get all segments a user belongs to + */ + async getUserSegments(userId: string): Promise { + const allSegments = await this.segmentRepository.find({ + relations: ['rules'], + }); + const userSegments: Segment[] = []; + for (const segment of allSegments) { + if (await this.isUserInSegment(userId, segment)) { + userSegments.push(segment); + } + } + return userSegments; } - - // TODO: Fetch actual user data from Users module - // For now, return mock data - return segment.staticMemberIds.map((id) => ({ - id, - email: `user-${id}@example.com`, - createdAt: new Date(), - })); - } - - /** - * Evaluate segment rules and return matching users - */ - private async evaluateSegmentRules(rules: SegmentRule[]): Promise { - if (!rules.length) { - return []; + // ==================== Private Helper Methods ==================== + /** + * Calculate member count for a segment (by ID - may cause recursion, use carefully) + */ + private async calculateMemberCount(segmentId: string): Promise { + const members = await this.getSegmentMembers(segmentId); + return members.length; } - - // TODO: Build actual query against User entity - // This is a placeholder implementation showing the query building logic - - // Group rules by AND/OR logic - const sortedRules = [...rules].sort((a, b) => a.order - b.order); - - // Build query conditions - const conditions: string[] = []; - - for (const rule of sortedRules) { - const condition = this.buildRuleCondition(rule); - if (condition) { - conditions.push(condition); - } + /** + * Calculate member count for a segment (accepts segment object to avoid recursion) + */ + private async calculateMemberCountForSegment(segment: Segment): Promise { + if (!segment.isDynamic) { + return segment.staticMemberIds?.length || 0; + } + const members = await this.evaluateSegmentRules(segment.rules); + return members.length; } - - // For MVP, return empty array - actual implementation requires User repository - // console.log('Segment rules would evaluate:', conditions); - - // TODO: Execute query and return real users - return []; - } - - /** - * Build SQL condition from a rule - */ - private buildRuleCondition(rule: SegmentRule): string | null { - const field = this.getFieldMapping(rule.field); - const operator = this.getOperatorMapping(rule.operator); - const value = this.formatValue(rule.value, rule.operator); - - if (!field || !operator) { - return null; + /** + * Get members of a static segment + */ + private async getStaticSegmentMembers(segmentId: string): Promise { + const segment = await this.segmentRepository.findOne({ + where: { id: segmentId }, + }); + if (!segment?.staticMemberIds?.length) { + return []; + } + // TODO: Fetch actual user data from Users module + // For now, return mock data + return segment.staticMemberIds.map((id) => ({ + id, + email: `user-${id}@example.com`, + createdAt: new Date(), + })); } - - return `${field} ${operator} ${value}`; - } - - /** - * Map rule field to database column - */ - private getFieldMapping(field: SegmentRuleField): string | null { - const mapping: Record = { - [SegmentRuleField.EMAIL]: 'user.email', - [SegmentRuleField.FIRST_NAME]: 'user.firstName', - [SegmentRuleField.LAST_NAME]: 'user.lastName', - [SegmentRuleField.CREATED_AT]: 'user.createdAt', - [SegmentRuleField.LAST_LOGIN]: 'user.lastLoginAt', - [SegmentRuleField.COURSE_COUNT]: 'enrollments.count', - [SegmentRuleField.TOTAL_SPENT]: 'payments.total', - [SegmentRuleField.TAG]: 'user.tags', - [SegmentRuleField.COUNTRY]: 'user.country', - [SegmentRuleField.SUBSCRIPTION_STATUS]: 'subscription.status', - }; - - return mapping[field] || null; - } - - /** - * Map rule operator to SQL operator - */ - private getOperatorMapping(operator: SegmentRuleOperator): string | null { - const mapping: Record = { - [SegmentRuleOperator.EQUALS]: '=', - [SegmentRuleOperator.NOT_EQUALS]: '!=', - [SegmentRuleOperator.CONTAINS]: 'LIKE', - [SegmentRuleOperator.NOT_CONTAINS]: 'NOT LIKE', - [SegmentRuleOperator.STARTS_WITH]: 'LIKE', - [SegmentRuleOperator.ENDS_WITH]: 'LIKE', - [SegmentRuleOperator.GREATER_THAN]: '>', - [SegmentRuleOperator.LESS_THAN]: '<', - [SegmentRuleOperator.GREATER_OR_EQUAL]: '>=', - [SegmentRuleOperator.LESS_OR_EQUAL]: '<=', - [SegmentRuleOperator.IS_SET]: 'IS NOT NULL', - [SegmentRuleOperator.IS_NOT_SET]: 'IS NULL', - [SegmentRuleOperator.IN_LIST]: 'IN', - [SegmentRuleOperator.NOT_IN_LIST]: 'NOT IN', - [SegmentRuleOperator.BEFORE]: '<', - [SegmentRuleOperator.AFTER]: '>', - [SegmentRuleOperator.BETWEEN]: 'BETWEEN', - }; - - return mapping[operator] || null; - } - - /** - * Format value based on operator - */ - private formatValue(value: any, operator: SegmentRuleOperator): string { - if ( - operator === SegmentRuleOperator.CONTAINS || - operator === SegmentRuleOperator.NOT_CONTAINS - ) { - return `'%${value}%'`; + /** + * Evaluate segment rules and return matching users + */ + private async evaluateSegmentRules(rules: SegmentRule[]): Promise { + if (!rules.length) { + return []; + } + // TODO: Build actual query against User entity + // This is a placeholder implementation showing the query building logic + // Group rules by AND/OR logic + const sortedRules = [...rules].sort((a, b) => a.order - b.order); + // Build query conditions + const conditions: string[] = []; + for (const rule of sortedRules) { + const condition = this.buildRuleCondition(rule); + if (condition) { + conditions.push(condition); + } + } + // For MVP, return empty array - actual implementation requires User repository + // console.log('Segment rules would evaluate:', conditions); + // TODO: Execute query and return real users + return []; } - - if (operator === SegmentRuleOperator.STARTS_WITH) { - return `'${value}%'`; + /** + * Build SQL condition from a rule + */ + private buildRuleCondition(rule: SegmentRule): string | null { + const field = this.getFieldMapping(rule.field); + const operator = this.getOperatorMapping(rule.operator); + const value = this.formatValue(rule.value, rule.operator); + if (!field || !operator) { + return null; + } + return `${field} ${operator} ${value}`; } - - if (operator === SegmentRuleOperator.ENDS_WITH) { - return `'%${value}'`; + /** + * Map rule field to database column + */ + private getFieldMapping(field: SegmentRuleField): string | null { + const mapping: Record = { + [SegmentRuleField.EMAIL]: 'user.email', + [SegmentRuleField.FIRST_NAME]: 'user.firstName', + [SegmentRuleField.LAST_NAME]: 'user.lastName', + [SegmentRuleField.CREATED_AT]: 'user.createdAt', + [SegmentRuleField.LAST_LOGIN]: 'user.lastLoginAt', + [SegmentRuleField.COURSE_COUNT]: 'enrollments.count', + [SegmentRuleField.TOTAL_SPENT]: 'payments.total', + [SegmentRuleField.TAG]: 'user.tags', + [SegmentRuleField.COUNTRY]: 'user.country', + [SegmentRuleField.SUBSCRIPTION_STATUS]: 'subscription.status', + }; + return mapping[field] || null; } - - if (operator === SegmentRuleOperator.IN_LIST || operator === SegmentRuleOperator.NOT_IN_LIST) { - if (Array.isArray(value)) { - return `(${value.map((v) => `'${v}'`).join(', ')})`; - } + /** + * Map rule operator to SQL operator + */ + private getOperatorMapping(operator: SegmentRuleOperator): string | null { + const mapping: Record = { + [SegmentRuleOperator.EQUALS]: '=', + [SegmentRuleOperator.NOT_EQUALS]: '!=', + [SegmentRuleOperator.CONTAINS]: 'LIKE', + [SegmentRuleOperator.NOT_CONTAINS]: 'NOT LIKE', + [SegmentRuleOperator.STARTS_WITH]: 'LIKE', + [SegmentRuleOperator.ENDS_WITH]: 'LIKE', + [SegmentRuleOperator.GREATER_THAN]: '>', + [SegmentRuleOperator.LESS_THAN]: '<', + [SegmentRuleOperator.GREATER_OR_EQUAL]: '>=', + [SegmentRuleOperator.LESS_OR_EQUAL]: '<=', + [SegmentRuleOperator.IS_SET]: 'IS NOT NULL', + [SegmentRuleOperator.IS_NOT_SET]: 'IS NULL', + [SegmentRuleOperator.IN_LIST]: 'IN', + [SegmentRuleOperator.NOT_IN_LIST]: 'NOT IN', + [SegmentRuleOperator.BEFORE]: '<', + [SegmentRuleOperator.AFTER]: '>', + [SegmentRuleOperator.BETWEEN]: 'BETWEEN', + }; + return mapping[operator] || null; } - - if (typeof value === 'string') { - return `'${value}'`; + /** + * Format value based on operator + */ + private formatValue(value: unknown, operator: SegmentRuleOperator): string { + if (operator === SegmentRuleOperator.CONTAINS || + operator === SegmentRuleOperator.NOT_CONTAINS) { + return `'%${value}%'`; + } + if (operator === SegmentRuleOperator.STARTS_WITH) { + return `'${value}%'`; + } + if (operator === SegmentRuleOperator.ENDS_WITH) { + return `'%${value}'`; + } + if (operator === SegmentRuleOperator.IN_LIST || operator === SegmentRuleOperator.NOT_IN_LIST) { + if (Array.isArray(value)) { + return `(${value.map((v) => `'${v}'`).join(', ')})`; + } + } + if (typeof value === 'string') { + return `'${value}'`; + } + return String(value); } - - return String(value); - } } diff --git a/src/email-marketing/sender/email-sender.service.ts b/src/email-marketing/sender/email-sender.service.ts index 9a32c30b..3d2fc268 100644 --- a/src/email-marketing/sender/email-sender.service.ts +++ b/src/email-marketing/sender/email-sender.service.ts @@ -1,195 +1,156 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import * as nodemailer from 'nodemailer'; - import { TemplateManagementService } from '../templates/template-management.service'; import { EmailAnalyticsService } from '../analytics/email-analytics.service'; import { EmailEventType } from '../enums/email-event-type.enum'; - export interface SendEmailOptions { - to: string; - subject?: string; - templateId?: string; - html?: string; - text?: string; - variables?: Record; - campaignId?: string; - recipientId?: string; - variantId?: string; - trackOpens?: boolean; - trackClicks?: boolean; + to: string; + subject?: string; + templateId?: string; + html?: string; + text?: string; + variables?: Record; + campaignId?: string; + recipientId?: string; + variantId?: string; + trackOpens?: boolean; + trackClicks?: boolean; } - export interface SendEmailResult { - success: boolean; - messageId?: string; - error?: string; + success: boolean; + messageId?: string; + error?: string; } - @Injectable() export class EmailSenderService { - private readonly logger = new Logger(EmailSenderService.name); - private transporter: nodemailer.Transporter; - - constructor( - private readonly configService: ConfigService, - private readonly templateService: TemplateManagementService, - private readonly analyticsService: EmailAnalyticsService, - ) { - this.initializeTransporter(); - } - - private initializeTransporter(): void { - this.transporter = nodemailer.createTransport({ - host: this.configService.get('SMTP_HOST', 'smtp.mailtrap.io'), - port: this.configService.get('SMTP_PORT', 587), - secure: this.configService.get('SMTP_SECURE', false), - auth: { - user: this.configService.get('SMTP_USER'), - pass: this.configService.get('SMTP_PASS'), - }, - }); - } - - /** - * Send a single email - */ - async sendEmail(options: SendEmailOptions): Promise { - try { - let html = options.html; - let text = options.text; - let subject = options.subject; - - // Render template if provided - if (options.templateId) { - const rendered = await this.templateService.renderTemplate( - options.templateId, - options.variables || {}, - ); - html = rendered.html; - text = rendered.text; - subject = rendered.subject; - } - - // Add tracking pixel for opens - if (options.trackOpens && options.campaignId && options.recipientId) { - html = this.addOpenTrackingPixel(html, options.campaignId, options.recipientId); - } - - // Track link clicks - if (options.trackClicks && options.campaignId && options.recipientId) { - html = this.wrapLinksForTracking(html, options.campaignId, options.recipientId); - } - - const fromEmail = this.configService.get('EMAIL_FROM', 'noreply@teachlink.io'); - const fromName = this.configService.get('EMAIL_FROM_NAME', 'TeachLink'); - - const result = await this.transporter.sendMail({ - from: `"${fromName}" <${fromEmail}>`, - to: options.to, - subject, - html, - text, - }); - - // Record sent event - if (options.campaignId && options.recipientId) { - await this.analyticsService.recordEvent( - options.campaignId, - options.recipientId, - EmailEventType.SENT, - { messageId: result.messageId, variantId: options.variantId }, - ); - } - - this.logger.log(`Email sent to ${options.to}, messageId: ${result.messageId}`); - - return { success: true, messageId: result.messageId }; - } catch (error) { - this.logger.error(`Failed to send email to ${options.to}:`, error); - - // Record bounce event - if (options.campaignId && options.recipientId) { - await this.analyticsService.recordEvent( - options.campaignId, - options.recipientId, - EmailEventType.BOUNCED, - { error: error.message }, - ); - } - - return { success: false, error: error.message }; + private readonly logger = new Logger(EmailSenderService.name); + private transporter: nodemailer.Transporter; + constructor(private readonly configService: ConfigService, private readonly templateService: TemplateManagementService, private readonly analyticsService: EmailAnalyticsService) { + this.initializeTransporter(); } - } - - /** - * Send bulk emails - */ - async sendBulkEmails( - recipients: Array<{ email: string; id: string; variables?: Record }>, - options: Omit, - ): Promise<{ sent: number; failed: number; results: SendEmailResult[] }> { - const results: SendEmailResult[] = []; - let sent = 0; - let failed = 0; - - for (const recipient of recipients) { - const result = await this.sendEmail({ - ...options, - to: recipient.email, - recipientId: recipient.id, - variables: { ...options.variables, ...recipient.variables }, - }); - - results.push(result); - if (result.success) sent++; - else failed++; - - // Small delay to avoid rate limiting - await this.delay(50); + private initializeTransporter(): void { + this.transporter = nodemailer.createTransport({ + host: this.configService.get('SMTP_HOST', 'smtp.mailtrap.io'), + port: this.configService.get('SMTP_PORT', 587), + secure: this.configService.get('SMTP_SECURE', false), + auth: { + user: this.configService.get('SMTP_USER'), + pass: this.configService.get('SMTP_PASS'), + }, + }); } - - return { sent, failed, results }; - } - - /** - * Verify SMTP connection - */ - async verifyConnection(): Promise { - try { - await this.transporter.verify(); - return true; - } catch (error) { - this.logger.error('SMTP connection verification failed:', error); - return false; + /** + * Send a single email + */ + async sendEmail(options: SendEmailOptions): Promise { + try { + let html = options.html; + let text = options.text; + let subject = options.subject; + // Render template if provided + if (options.templateId) { + const rendered = await this.templateService.renderTemplate(options.templateId, options.variables || {}); + html = rendered.html; + text = rendered.text; + subject = rendered.subject; + } + // Add tracking pixel for opens + if (options.trackOpens && options.campaignId && options.recipientId) { + html = this.addOpenTrackingPixel(html, options.campaignId, options.recipientId); + } + // Track link clicks + if (options.trackClicks && options.campaignId && options.recipientId) { + html = this.wrapLinksForTracking(html, options.campaignId, options.recipientId); + } + const fromEmail = this.configService.get('EMAIL_FROM', 'noreply@teachlink.io'); + const fromName = this.configService.get('EMAIL_FROM_NAME', 'TeachLink'); + const result = await this.transporter.sendMail({ + from: `"${fromName}" <${fromEmail}>`, + to: options.to, + subject, + html, + text, + }); + // Record sent event + if (options.campaignId && options.recipientId) { + await this.analyticsService.recordEvent(options.campaignId, options.recipientId, EmailEventType.SENT, { messageId: result.messageId, variantId: options.variantId }); + } + this.logger.log(`Email sent to ${options.to}, messageId: ${result.messageId}`); + return { success: true, messageId: result.messageId }; + } + catch (error) { + this.logger.error(`Failed to send email to ${options.to}:`, error); + // Record bounce event + if (options.campaignId && options.recipientId) { + await this.analyticsService.recordEvent(options.campaignId, options.recipientId, EmailEventType.BOUNCED, { error: error.message }); + } + return { success: false, error: error.message }; + } } - } - - // Private helper methods - private addOpenTrackingPixel(html: string, campaignId: string, recipientId: string): string { - const baseUrl = this.configService.get('APP_URL', 'http://localhost:3000'); - const trackingUrl = `${baseUrl}/email-marketing/track/open?c=${campaignId}&r=${recipientId}`; - const pixel = ``; - - return html.replace('', `${pixel}`); - } - - private wrapLinksForTracking(html: string, campaignId: string, recipientId: string): string { - const baseUrl = this.configService.get('APP_URL', 'http://localhost:3000'); - - return html.replace( - /]*href=["'])([^"']+)(["'][^>]*)>/gi, - (match, prefix, url, suffix) => { - if (url.startsWith('mailto:') || url.startsWith('#')) { - return match; + /** + * Send bulk emails + */ + async sendBulkEmails(recipients: Array<{ + email: string; + id: string; + variables?: Record; + }>, options: Omit): Promise<{ + sent: number; + failed: number; + results: SendEmailResult[]; + }> { + const results: SendEmailResult[] = []; + let sent = 0; + let failed = 0; + for (const recipient of recipients) { + const result = await this.sendEmail({ + ...options, + to: recipient.email, + recipientId: recipient.id, + variables: { ...options.variables, ...recipient.variables }, + }); + results.push(result); + if (result.success) + sent++; + else + failed++; + // Small delay to avoid rate limiting + await this.delay(50); + } + return { sent, failed, results }; + } + /** + * Verify SMTP connection + */ + async verifyConnection(): Promise { + try { + await this.transporter.verify(); + return true; + } + catch (error) { + this.logger.error('SMTP connection verification failed:', error); + return false; } - const trackingUrl = `${baseUrl}/email-marketing/track/click?c=${campaignId}&r=${recipientId}&url=${encodeURIComponent(url)}`; - return ``; - }, - ); - } - - private delay(ms: number): Promise { - return new Promise((resolve) => setTimeout(resolve, ms)); - } + } + // Private helper methods + private addOpenTrackingPixel(html: string, campaignId: string, recipientId: string): string { + const baseUrl = this.configService.get('APP_URL', 'http://localhost:3000'); + const trackingUrl = `${baseUrl}/email-marketing/track/open?c=${campaignId}&r=${recipientId}`; + const pixel = ``; + return html.replace('', `${pixel}`); + } + private wrapLinksForTracking(html: string, campaignId: string, recipientId: string): string { + const baseUrl = this.configService.get('APP_URL', 'http://localhost:3000'); + return html.replace(/]*href=["'])([^"']+)(["'][^>]*)>/gi, (match, prefix, url, suffix) => { + if (url.startsWith('mailto:') || url.startsWith('#')) { + return match; + } + const trackingUrl = `${baseUrl}/email-marketing/track/click?c=${campaignId}&r=${recipientId}&url=${encodeURIComponent(url)}`; + return ``; + }); + } + private delay(ms: number): Promise { + return new Promise((resolve) => setTimeout(resolve, ms)); + } } diff --git a/src/email-marketing/templates/template-management.service.ts b/src/email-marketing/templates/template-management.service.ts index eb27d9b6..3450b541 100644 --- a/src/email-marketing/templates/template-management.service.ts +++ b/src/email-marketing/templates/template-management.service.ts @@ -2,196 +2,167 @@ import { Injectable, NotFoundException, BadRequestException } from '@nestjs/comm import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import * as Handlebars from 'handlebars'; - import { EmailTemplate } from '../entities/email-template.entity'; import { CreateTemplateDto } from '../dto/create-template.dto'; import { UpdateTemplateDto } from '../dto/update-template.dto'; - @Injectable() export class TemplateManagementService { - constructor( + constructor( @InjectRepository(EmailTemplate) - private readonly templateRepository: Repository, - ) { - this.registerHandlebarsHelpers(); - } - - /** - * Create a new email template - */ - async create(createTemplateDto: CreateTemplateDto): Promise { - // Validate template syntax - this.validateTemplateSyntax(createTemplateDto.htmlContent); - if (createTemplateDto.textContent) { - this.validateTemplateSyntax(createTemplateDto.textContent); + private readonly templateRepository: Repository) { + this.registerHandlebarsHelpers(); } - - const template = this.templateRepository.create({ - ...createTemplateDto, - variables: this.extractVariables(createTemplateDto.htmlContent), - }); - - return this.templateRepository.save(template); - } - - /** - * Get all templates - */ - async findAll( - page = 1, - limit = 10, - ): Promise<{ - templates: EmailTemplate[]; - total: number; - page: number; - totalPages: number; - }> { - const [templates, total] = await this.templateRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - }); - - return { templates, total, page, totalPages: Math.ceil(total / limit) }; - } - - /** - * Get a single template by ID - */ - async findOne(id: string): Promise { - const template = await this.templateRepository.findOne({ where: { id } }); - if (!template) { - throw new NotFoundException(`Template with ID ${id} not found`); + /** + * Create a new email template + */ + async create(createTemplateDto: CreateTemplateDto): Promise { + // Validate template syntax + this.validateTemplateSyntax(createTemplateDto.htmlContent); + if (createTemplateDto.textContent) { + this.validateTemplateSyntax(createTemplateDto.textContent); + } + const template = this.templateRepository.create({ + ...createTemplateDto, + variables: this.extractVariables(createTemplateDto.htmlContent), + }); + return this.templateRepository.save(template); } - return template; - } - - /** - * Update a template - */ - async update(id: string, updateTemplateDto: UpdateTemplateDto): Promise { - const template = await this.findOne(id); - - if (updateTemplateDto.htmlContent) { - this.validateTemplateSyntax(updateTemplateDto.htmlContent); - updateTemplateDto['variables'] = this.extractVariables(updateTemplateDto.htmlContent); + /** + * Get all templates + */ + async findAll(page = 1, limit = 10): Promise<{ + templates: EmailTemplate[]; + total: number; + page: number; + totalPages: number; + }> { + const [templates, total] = await this.templateRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + }); + return { templates, total, page, totalPages: Math.ceil(total / limit) }; } - - Object.assign(template, updateTemplateDto); - return this.templateRepository.save(template); - } - - /** - * Delete a template - */ - async remove(id: string): Promise { - await this.findOne(id); - await this.templateRepository.softDelete(id); - } - - /** - * Duplicate a template - */ - async duplicate(id: string): Promise { - const original = await this.findOne(id); - - const duplicate = this.templateRepository.create({ - name: `${original.name} (Copy)`, - subject: original.subject, - htmlContent: original.htmlContent, - textContent: original.textContent, - category: original.category, - variables: original.variables, - }); - - return this.templateRepository.save(duplicate); - } - - /** - * Render a template with variables - */ - async renderTemplate( - templateId: string, - variables: Record, - ): Promise<{ html: string; text: string; subject: string }> { - const template = await this.findOne(templateId); - - const htmlTemplate = Handlebars.compile(template.htmlContent); - const subjectTemplate = Handlebars.compile(template.subject); - const textTemplate = template.textContent ? Handlebars.compile(template.textContent) : null; - - return { - html: htmlTemplate(variables), - text: textTemplate ? textTemplate(variables) : this.stripHtml(htmlTemplate(variables)), - subject: subjectTemplate(variables), - }; - } - - /** - * Preview a template with sample data - */ - async previewTemplate(id: string, sampleData?: Record) { - const template = await this.findOne(id); - - const defaultData = this.generateSampleData(template.variables || []); - const data = { ...defaultData, ...sampleData }; - - return this.renderTemplate(id, data); - } - - // Private helper methods - private validateTemplateSyntax(content: string): void { - try { - Handlebars.compile(content); - } catch (error) { - throw new BadRequestException(`Invalid template syntax: ${error.message}`); + /** + * Get a single template by ID + */ + async findOne(id: string): Promise { + const template = await this.templateRepository.findOne({ where: { id } }); + if (!template) { + throw new NotFoundException(`Template with ID ${id} not found`); + } + return template; } - } - - private extractVariables(content: string): string[] { - const regex = /\{\{([^}]+)\}\}/g; - const variables = new Set(); - let match; - - while ((match = regex.exec(content)) !== null) { - const variable = match[1].trim().split(' ')[0]; - if (!variable.startsWith('#') && !variable.startsWith('/')) { - variables.add(variable); - } + /** + * Update a template + */ + async update(id: string, updateTemplateDto: UpdateTemplateDto): Promise { + const template = await this.findOne(id); + if (updateTemplateDto.htmlContent) { + this.validateTemplateSyntax(updateTemplateDto.htmlContent); + updateTemplateDto['variables'] = this.extractVariables(updateTemplateDto.htmlContent); + } + Object.assign(template, updateTemplateDto); + return this.templateRepository.save(template); } - - return Array.from(variables); - } - - private generateSampleData(variables: string[]): Record { - const sampleData: Record = {}; - - for (const variable of variables) { - if (variable.includes('name')) sampleData[variable] = 'John Doe'; - else if (variable.includes('email')) sampleData[variable] = 'john@example.com'; - else if (variable.includes('url') || variable.includes('link')) { - sampleData[variable] = 'https://example.com'; - } else { - sampleData[variable] = `[${variable}]`; - } + /** + * Delete a template + */ + async remove(id: string): Promise { + await this.findOne(id); + await this.templateRepository.softDelete(id); + } + /** + * Duplicate a template + */ + async duplicate(id: string): Promise { + const original = await this.findOne(id); + const duplicate = this.templateRepository.create({ + name: `${original.name} (Copy)`, + subject: original.subject, + htmlContent: original.htmlContent, + textContent: original.textContent, + category: original.category, + variables: original.variables, + }); + return this.templateRepository.save(duplicate); + } + /** + * Render a template with variables + */ + async renderTemplate(templateId: string, variables: Record): Promise<{ + html: string; + text: string; + subject: string; + }> { + const template = await this.findOne(templateId); + const htmlTemplate = Handlebars.compile(template.htmlContent); + const subjectTemplate = Handlebars.compile(template.subject); + const textTemplate = template.textContent ? Handlebars.compile(template.textContent) : null; + return { + html: htmlTemplate(variables), + text: textTemplate ? textTemplate(variables) : this.stripHtml(htmlTemplate(variables)), + subject: subjectTemplate(variables), + }; + } + /** + * Preview a template with sample data + */ + async previewTemplate(id: string, sampleData?: Record) { + const template = await this.findOne(id); + const defaultData = this.generateSampleData(template.variables || []); + const data = { ...defaultData, ...sampleData }; + return this.renderTemplate(id, data); + } + // Private helper methods + private validateTemplateSyntax(content: string): void { + try { + Handlebars.compile(content); + } + catch (error) { + throw new BadRequestException(`Invalid template syntax: ${error.message}`); + } + } + private extractVariables(content: string): string[] { + const regex = /\{\{([^}]+)\}\}/g; + const variables = new Set(); + let match; + while ((match = regex.exec(content)) !== null) { + const variable = match[1].trim().split(' ')[0]; + if (!variable.startsWith('#') && !variable.startsWith('/')) { + variables.add(variable); + } + } + return Array.from(variables); + } + private generateSampleData(variables: string[]): Record { + const sampleData: Record = {}; + for (const variable of variables) { + if (variable.includes('name')) + sampleData[variable] = 'John Doe'; + else if (variable.includes('email')) + sampleData[variable] = 'john@example.com'; + else if (variable.includes('url') || variable.includes('link')) { + sampleData[variable] = 'https://example.com'; + } + else { + sampleData[variable] = `[${variable}]`; + } + } + return sampleData; + } + private stripHtml(html: string): string { + return html + .replace(/<[^>]*>/g, '') + .replace(/\s+/g, ' ') + .trim(); + } + private registerHandlebarsHelpers(): void { + Handlebars.registerHelper('uppercase', (str) => str?.toUpperCase()); + Handlebars.registerHelper('lowercase', (str) => str?.toLowerCase()); + Handlebars.registerHelper('formatDate', (date) => new Date(date).toLocaleDateString()); + Handlebars.registerHelper('ifEquals', function (a, b, options) { + return a === b ? options.fn(this) : options.inverse(this); + }); } - - return sampleData; - } - - private stripHtml(html: string): string { - return html - .replace(/<[^>]*>/g, '') - .replace(/\s+/g, ' ') - .trim(); - } - - private registerHandlebarsHelpers(): void { - Handlebars.registerHelper('uppercase', (str) => str?.toUpperCase()); - Handlebars.registerHelper('lowercase', (str) => str?.toLowerCase()); - Handlebars.registerHelper('formatDate', (date) => new Date(date).toLocaleDateString()); - Handlebars.registerHelper('ifEquals', function (a, b, options) { - return a === b ? options.fn(this) : options.inverse(this); - }); - } } diff --git a/src/email-marketing/templates/template.controller.ts b/src/email-marketing/templates/template.controller.ts index 6e17894a..abeda50d 100644 --- a/src/email-marketing/templates/template.controller.ts +++ b/src/email-marketing/templates/template.controller.ts @@ -1,82 +1,81 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Body, - Param, - Query, - ParseUUIDPipe, - HttpCode, - HttpStatus, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Body, Param, Query, ParseUUIDPipe, HttpCode, HttpStatus, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; - import { TemplateManagementService } from './template-management.service'; import { CreateTemplateDto } from '../dto/create-template.dto'; import { UpdateTemplateDto } from '../dto/update-template.dto'; import { EmailTemplate } from '../entities/email-template.entity'; - @ApiTags('Email Marketing - Templates') @ApiBearerAuth() @Controller('email-marketing/templates') export class TemplateController { - constructor(private readonly templateService: TemplateManagementService) {} - - @Post() - @ApiOperation({ summary: 'Create a new email template' }) - @ApiResponse({ status: 201, description: 'Template created successfully' }) - async create(@Body() createTemplateDto: CreateTemplateDto): Promise { - return this.templateService.create(createTemplateDto); - } - - @Get() - @ApiOperation({ summary: 'Get all email templates' }) - @ApiQuery({ name: 'page', required: false, type: Number }) - @ApiQuery({ name: 'limit', required: false, type: Number }) - async findAll(@Query('page') page = 1, @Query('limit') limit = 10) { - return this.templateService.findAll(page, limit); - } - - @Get(':id') - @ApiOperation({ summary: 'Get a template by ID' }) - @ApiResponse({ status: 404, description: 'Template not found' }) - async findOne(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.templateService.findOne(id); - } - - @Put(':id') - @ApiOperation({ summary: 'Update a template' }) - async update( - @Param('id', ParseUUIDPipe) id: string, - @Body() updateTemplateDto: UpdateTemplateDto, - ): Promise { - return this.templateService.update(id, updateTemplateDto); - } - - @Delete(':id') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete a template' }) - async remove(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.templateService.remove(id); - } - - @Post(':id/duplicate') - @ApiOperation({ summary: 'Duplicate a template' }) - async duplicate(@Param('id', ParseUUIDPipe) id: string): Promise { - return this.templateService.duplicate(id); - } - - @Post(':id/preview') - @ApiOperation({ summary: 'Preview a template with sample data' }) - async preview(@Param('id', ParseUUIDPipe) id: string, @Body() sampleData?: Record) { - return this.templateService.previewTemplate(id, sampleData); - } - - @Post(':id/render') - @ApiOperation({ summary: 'Render a template with provided variables' }) - async render(@Param('id', ParseUUIDPipe) id: string, @Body() variables: Record) { - return this.templateService.renderTemplate(id, variables); - } + constructor(private readonly templateService: TemplateManagementService) { } + @Post() + @ApiOperation({ summary: 'Create a new email template' }) + @ApiResponse({ status: 201, description: 'Template created successfully' }) + async create( + @Body() + createTemplateDto: CreateTemplateDto): Promise { + return this.templateService.create(createTemplateDto); + } + @Get() + @ApiOperation({ summary: 'Get all email templates' }) + @ApiQuery({ name: 'page', required: false, type: Number }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + async findAll( + @Query('page') + page = 1, + @Query('limit') + limit = 10) { + return this.templateService.findAll(page, limit); + } + @Get(':id') + @ApiOperation({ summary: 'Get a template by ID' }) + @ApiResponse({ status: 404, description: 'Template not found' }) + async findOne( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.templateService.findOne(id); + } + @Put(':id') + @ApiOperation({ summary: 'Update a template' }) + async update( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + updateTemplateDto: UpdateTemplateDto): Promise { + return this.templateService.update(id, updateTemplateDto); + } + @Delete(':id') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete a template' }) + async remove( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.templateService.remove(id); + } + @Post(':id/duplicate') + @ApiOperation({ summary: 'Duplicate a template' }) + async duplicate( + @Param('id', ParseUUIDPipe) + id: string): Promise { + return this.templateService.duplicate(id); + } + @Post(':id/preview') + @ApiOperation({ summary: 'Preview a template with sample data' }) + async preview( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + sampleData?: Record) { + return this.templateService.previewTemplate(id, sampleData); + } + @Post(':id/render') + @ApiOperation({ summary: 'Render a template with provided variables' }) + async render( + @Param('id', ParseUUIDPipe) + id: string, + @Body() + variables: Record) { + return this.templateService.renderTemplate(id, variables); + } } diff --git a/src/email-marketing/tracking/tracking.controller.ts b/src/email-marketing/tracking/tracking.controller.ts index 3d4f8564..688a2bc9 100644 --- a/src/email-marketing/tracking/tracking.controller.ts +++ b/src/email-marketing/tracking/tracking.controller.ts @@ -1,10 +1,8 @@ import { Controller, Get, Query, Res } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiExcludeEndpoint } from '@nestjs/swagger'; import { Response } from 'express'; - import { EmailAnalyticsService } from '../analytics/email-analytics.service'; import { EmailEventType } from '../enums/email-event-type.enum'; - /** * Tracking controller for email opens and clicks * These endpoints are called by tracking pixels and wrapped links in emails @@ -12,143 +10,134 @@ import { EmailEventType } from '../enums/email-event-type.enum'; @ApiTags('Email Marketing - Tracking') @Controller('email-marketing/track') export class TrackingController { - // 1x1 transparent GIF pixel - private readonly TRACKING_PIXEL = Buffer.from( - 'R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7', - 'base64', - ); - - constructor(private readonly analyticsService: EmailAnalyticsService) {} - - /** - * Track email open via 1x1 tracking pixel - * Called when email client loads the tracking image - */ - @Get('open') - @ApiExcludeEndpoint() // Hide from Swagger as it's for internal use - async trackOpen( - @Query('c') campaignId: string, - @Query('r') recipientId: string, - @Res() res: Response, - ): Promise { - // Record the open event asynchronously - if (campaignId && recipientId) { - this.analyticsService - .recordEvent(campaignId, recipientId, EmailEventType.OPENED, { - timestamp: new Date().toISOString(), - }) - .catch((error) => { - console.error('Failed to record open event:', error); + // 1x1 transparent GIF pixel + private readonly TRACKING_PIXEL = Buffer.from('R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7', 'base64'); + constructor(private readonly analyticsService: EmailAnalyticsService) { } + /** + * Track email open via 1x1 tracking pixel + * Called when email client loads the tracking image + */ + @Get('open') + @ApiExcludeEndpoint() // Hide from Swagger as it's for internal use + async trackOpen( + @Query('c') + campaignId: string, + @Query('r') + recipientId: string, + @Res() + res: Response): Promise { + // Record the open event asynchronously + if (campaignId && recipientId) { + this.analyticsService + .recordEvent(campaignId, recipientId, EmailEventType.OPENED, { + timestamp: new Date().toISOString(), + }) + .catch((error) => { + console.error('Failed to record open event:', error); + }); + } + // Return the tracking pixel + res.set({ + 'Content-Type': 'image/gif', + 'Cache-Control': 'no-store, no-cache, must-revalidate, proxy-revalidate', + Pragma: 'no-cache', + Expires: '0', }); + res.send(this.TRACKING_PIXEL); } - - // Return the tracking pixel - res.set({ - 'Content-Type': 'image/gif', - 'Cache-Control': 'no-store, no-cache, must-revalidate, proxy-revalidate', - Pragma: 'no-cache', - Expires: '0', - }); - res.send(this.TRACKING_PIXEL); - } - - /** - * Track link click and redirect to original URL - * Called when user clicks a tracked link in the email - */ - @Get('click') - @ApiOperation({ summary: 'Track email link click and redirect' }) - async trackClick( - @Query('c') campaignId: string, - @Query('r') recipientId: string, - @Query('url') url: string, - @Res() res: Response, - ): Promise { - // Validate URL to prevent open redirect vulnerability - const decodedUrl = decodeURIComponent(url || ''); - - if (!this.isValidRedirectUrl(decodedUrl)) { - res.status(400).send('Invalid URL'); - return; + /** + * Track link click and redirect to original URL + * Called when user clicks a tracked link in the email + */ + @Get('click') + @ApiOperation({ summary: 'Track email link click and redirect' }) + async trackClick( + @Query('c') + campaignId: string, + @Query('r') + recipientId: string, + @Query('url') + url: string, + @Res() + res: Response): Promise { + // Validate URL to prevent open redirect vulnerability + const decodedUrl = decodeURIComponent(url || ''); + if (!this.isValidRedirectUrl(decodedUrl)) { + res.status(400).send('Invalid URL'); + return; + } + // Record the click event asynchronously + if (campaignId && recipientId) { + this.analyticsService + .recordEvent(campaignId, recipientId, EmailEventType.CLICKED, { + url: decodedUrl, + timestamp: new Date().toISOString(), + }) + .catch((error) => { + console.error('Failed to record click event:', error); + }); + } + // Redirect to the original URL + res.redirect(302, decodedUrl); } - - // Record the click event asynchronously - if (campaignId && recipientId) { - this.analyticsService - .recordEvent(campaignId, recipientId, EmailEventType.CLICKED, { - url: decodedUrl, - timestamp: new Date().toISOString(), - }) - .catch((error) => { - console.error('Failed to record click event:', error); - }); - } - - // Redirect to the original URL - res.redirect(302, decodedUrl); - } - - /** - * Track email delivery (called by email service provider webhook) - */ - @Get('delivered') - @ApiExcludeEndpoint() - async trackDelivered( - @Query('c') campaignId: string, - @Query('r') recipientId: string, - @Res() res: Response, - ): Promise { - if (campaignId && recipientId) { - await this.analyticsService.recordEvent(campaignId, recipientId, EmailEventType.DELIVERED); + /** + * Track email delivery (called by email service provider webhook) + */ + @Get('delivered') + @ApiExcludeEndpoint() + async trackDelivered( + @Query('c') + campaignId: string, + @Query('r') + recipientId: string, + @Res() + res: Response): Promise { + if (campaignId && recipientId) { + await this.analyticsService.recordEvent(campaignId, recipientId, EmailEventType.DELIVERED); + } + res.status(200).send('OK'); } - res.status(200).send('OK'); - } - - /** - * Track email bounce (called by email service provider webhook) - */ - @Get('bounce') - @ApiExcludeEndpoint() - async trackBounce( - @Query('c') campaignId: string, - @Query('r') recipientId: string, - @Query('type') bounceType: string, - @Res() res: Response, - ): Promise { - if (campaignId && recipientId) { - const eventType = - bounceType === 'soft' ? EmailEventType.SOFT_BOUNCED : EmailEventType.BOUNCED; - - await this.analyticsService.recordEvent(campaignId, recipientId, eventType, { bounceType }); + /** + * Track email bounce (called by email service provider webhook) + */ + @Get('bounce') + @ApiExcludeEndpoint() + async trackBounce( + @Query('c') + campaignId: string, + @Query('r') + recipientId: string, + @Query('type') + bounceType: string, + @Res() + res: Response): Promise { + if (campaignId && recipientId) { + const eventType = bounceType === 'soft' ? EmailEventType.SOFT_BOUNCED : EmailEventType.BOUNCED; + await this.analyticsService.recordEvent(campaignId, recipientId, eventType, { bounceType }); + } + res.status(200).send('OK'); } - res.status(200).send('OK'); - } - - /** - * Handle unsubscribe requests - */ - @Get('unsubscribe') - @ApiOperation({ summary: 'Unsubscribe from email list' }) - async unsubscribe( - @Query('c') campaignId: string, - @Query('r') recipientId: string, - @Query('email') email: string, - @Res() res: Response, - ): Promise { - if (campaignId && recipientId) { - await this.analyticsService.recordEvent( - campaignId, - recipientId, - EmailEventType.UNSUBSCRIBED, - { email }, - ); - } - - // TODO: Actually unsubscribe the user in the subscription service - - // Return a simple confirmation page - res.set('Content-Type', 'text/html'); - res.send(` + /** + * Handle unsubscribe requests + */ + @Get('unsubscribe') + @ApiOperation({ summary: 'Unsubscribe from email list' }) + async unsubscribe( + @Query('c') + campaignId: string, + @Query('r') + recipientId: string, + @Query('email') + email: string, + @Res() + res: Response): Promise { + if (campaignId && recipientId) { + await this.analyticsService.recordEvent(campaignId, recipientId, EmailEventType.UNSUBSCRIBED, { email }); + } + // TODO: Actually unsubscribe the user in the subscription service + // Return a simple confirmation page + res.set('Content-Type', 'text/html'); + res.send(` @@ -166,31 +155,28 @@ export class TrackingController { `); - } - - /** - * Validate redirect URL to prevent open redirect attacks - */ - private isValidRedirectUrl(url: string): boolean { - if (!url) return false; - - try { - const parsed = new URL(url); - - // Only allow http and https protocols - if (!['http:', 'https:'].includes(parsed.protocol)) { - return false; - } - - // Optional: Add domain whitelist for extra security - // const allowedDomains = ['teachlink.io', 'www.teachlink.io']; - // if (!allowedDomains.includes(parsed.hostname)) { - // return false; - // } - - return true; - } catch { - return false; } - } + /** + * Validate redirect URL to prevent open redirect attacks + */ + private isValidRedirectUrl(url: string): boolean { + if (!url) + return false; + try { + const parsed = new URL(url); + // Only allow http and https protocols + if (!['http:', 'https:'].includes(parsed.protocol)) { + return false; + } + // Optional: Add domain whitelist for extra security + // const allowedDomains = ['teachlink.io', 'www.teachlink.io']; + // if (!allowedDomains.includes(parsed.hostname)) { + // return false; + // } + return true; + } + catch { + return false; + } + } } diff --git a/src/feature-flags/analytics/flag-analytics.service.ts b/src/feature-flags/analytics/flag-analytics.service.ts index 6e31e86b..3f29af14 100644 --- a/src/feature-flags/analytics/flag-analytics.service.ts +++ b/src/feature-flags/analytics/flag-analytics.service.ts @@ -1,224 +1,178 @@ import { Injectable } from '@nestjs/common'; -import { - ExperimentStats, - ExperimentVariantStats, - FlagAnalyticsEvent, - FlagEvaluationStats, - FlagSummary, -} from '../interfaces'; - +import { ExperimentStats, ExperimentVariantStats, FlagAnalyticsEvent, FlagEvaluationStats, FlagSummary, } from '../interfaces'; type TrackEvaluationInput = Omit; - @Injectable() export class FlagAnalyticsService { - /** flagKey → events */ - private readonly flagEvents = new Map(); - /** flagKey → Set of unique userIds */ - private readonly flagUsers = new Map>(); - /** experimentId → variantKey → impression count */ - private readonly experimentImpressions = new Map>(); - /** experimentId → variantKey → conversion count */ - private readonly experimentConversions = new Map>(); - - /** - * Records a flag evaluation event. - */ - trackEvaluation(input: TrackEvaluationInput): void { - const event: FlagAnalyticsEvent = { - ...input, - eventId: this.generateEventId(), - timestamp: new Date(), - }; - - if (event.flagKey) { - if (!this.flagEvents.has(event.flagKey)) { - this.flagEvents.set(event.flagKey, []); - } - const flagEvents = this.flagEvents.get(event.flagKey); - if (flagEvents) flagEvents.push(event); - - if (event.userId) { - if (!this.flagUsers.has(event.flagKey)) { - this.flagUsers.set(event.flagKey, new Set()); + /** flagKey → events */ + private readonly flagEvents = new Map(); + /** flagKey → Set of unique userIds */ + private readonly flagUsers = new Map>(); + /** experimentId → variantKey → impression count */ + private readonly experimentImpressions = new Map>(); + /** experimentId → variantKey → conversion count */ + private readonly experimentConversions = new Map>(); + /** + * Records a flag evaluation event. + */ + trackEvaluation(input: TrackEvaluationInput): void { + const event: FlagAnalyticsEvent = { + ...input, + eventId: this.generateEventId(), + timestamp: new Date(), + }; + if (event.flagKey) { + if (!this.flagEvents.has(event.flagKey)) { + this.flagEvents.set(event.flagKey, []); + } + const flagEvents = this.flagEvents.get(event.flagKey); + if (flagEvents) + flagEvents.push(event); + if (event.userId) { + if (!this.flagUsers.has(event.flagKey)) { + this.flagUsers.set(event.flagKey, new Set()); + } + const flagUsers = this.flagUsers.get(event.flagKey); + if (flagUsers) + flagUsers.add(event.userId); + } } - const flagUsers = this.flagUsers.get(event.flagKey); - if (flagUsers) flagUsers.add(event.userId); - } } - } - - /** - * Records an experiment impression (user saw a variant). - */ - trackImpression( - experimentId: string, - variantKey: string, - userId?: string, - flagKey?: string, - ): void { - this.incrementExperimentCounter(this.experimentImpressions, experimentId, variantKey); - - this.trackEvaluation({ - eventType: 'impression', - flagKey, - userId, - experimentId, - experimentVariantKey: variantKey, - }); - } - - /** - * Records an experiment conversion event. - */ - trackConversion( - experimentId: string, - variantKey: string, - userId?: string, - metadata?: Record, - ): void { - this.incrementExperimentCounter(this.experimentConversions, experimentId, variantKey); - - this.trackEvaluation({ - eventType: 'conversion', - userId, - experimentId, - experimentVariantKey: variantKey, - metadata, - }); - } - - /** - * Returns evaluation statistics for a flag. - * Optionally filters to events within the last `sinceHours` hours. - */ - getEvaluationStats(flagKey: string, sinceHours?: number): FlagEvaluationStats { - const allEvents = this.flagEvents.get(flagKey) ?? []; - - const events = sinceHours - ? allEvents.filter((e) => { - const cutoff = new Date(Date.now() - sinceHours * 3_600_000); - return e.timestamp >= cutoff; - }) - : allEvents; - - const evaluationsByVariation: Record = {}; - const evaluationsByReason: Record = {}; - let errorCount = 0; - let evaluationCount = 0; - - for (const event of events) { - if (event.eventType !== 'evaluation') continue; - evaluationCount++; - - if (event.variationKey) { - evaluationsByVariation[event.variationKey] = - (evaluationsByVariation[event.variationKey] ?? 0) + 1; - } - - if (event.reason) { - evaluationsByReason[event.reason] = (evaluationsByReason[event.reason] ?? 0) + 1; - if (event.reason === 'ERROR') errorCount++; - } + /** + * Records an experiment impression (user saw a variant). + */ + trackImpression(experimentId: string, variantKey: string, userId?: string, flagKey?: string): void { + this.incrementExperimentCounter(this.experimentImpressions, experimentId, variantKey); + this.trackEvaluation({ + eventType: 'impression', + flagKey, + userId, + experimentId, + experimentVariantKey: variantKey, + }); } - - return { - flagKey, - totalEvaluations: evaluationCount, - evaluationsByVariation, - evaluationsByReason, - uniqueUsers: this.flagUsers.get(flagKey)?.size ?? 0, - errorRate: evaluationCount > 0 ? errorCount / evaluationCount : 0, - }; - } - - /** - * Returns impression and conversion stats for all variants in an experiment. - */ - getExperimentStats(experimentId: string, controlVariantKey?: string): ExperimentStats { - const impressions = this.experimentImpressions.get(experimentId) ?? new Map(); - const conversions = this.experimentConversions.get(experimentId) ?? new Map(); - - const allVariantKeys = new Set([...impressions.keys(), ...conversions.keys()]); - - let totalImpressions = 0; - const variants: Record = {}; - - for (const variantKey of allVariantKeys) { - const imp = impressions.get(variantKey) ?? 0; - const conv = conversions.get(variantKey) ?? 0; - totalImpressions += imp; - - variants[variantKey] = { - variantKey, - impressions: imp, - conversions: conv, - conversionRate: imp > 0 ? conv / imp : 0, - isControl: variantKey === controlVariantKey, - }; + /** + * Records an experiment conversion event. + */ + trackConversion(experimentId: string, variantKey: string, userId?: string, metadata?: Record): void { + this.incrementExperimentCounter(this.experimentConversions, experimentId, variantKey); + this.trackEvaluation({ + eventType: 'conversion', + userId, + experimentId, + experimentVariantKey: variantKey, + metadata, + }); } - - return { experimentId, totalImpressions, variants }; - } - - /** - * Returns the most evaluated flags, sorted by evaluation count descending. - */ - getTopFlags(limit: number = 10): FlagSummary[] { - const summaries: FlagSummary[] = []; - - for (const [flagKey, events] of this.flagEvents.entries()) { - const evaluations = events.filter((e) => e.eventType === 'evaluation'); - summaries.push({ - flagKey, - totalEvaluations: evaluations.length, - lastEvaluatedAt: events[events.length - 1]?.timestamp, - }); + /** + * Returns evaluation statistics for a flag. + * Optionally filters to events within the last `sinceHours` hours. + */ + getEvaluationStats(flagKey: string, sinceHours?: number): FlagEvaluationStats { + const allEvents = this.flagEvents.get(flagKey) ?? []; + const events = sinceHours + ? allEvents.filter((e) => { + const cutoff = new Date(Date.now() - sinceHours * 3600000); + return e.timestamp >= cutoff; + }) + : allEvents; + const evaluationsByVariation: Record = {}; + const evaluationsByReason: Record = {}; + let errorCount = 0; + let evaluationCount = 0; + for (const event of events) { + if (event.eventType !== 'evaluation') + continue; + evaluationCount++; + if (event.variationKey) { + evaluationsByVariation[event.variationKey] = + (evaluationsByVariation[event.variationKey] ?? 0) + 1; + } + if (event.reason) { + evaluationsByReason[event.reason] = (evaluationsByReason[event.reason] ?? 0) + 1; + if (event.reason === 'ERROR') + errorCount++; + } + } + return { + flagKey, + totalEvaluations: evaluationCount, + evaluationsByVariation, + evaluationsByReason, + uniqueUsers: this.flagUsers.get(flagKey)?.size ?? 0, + errorRate: evaluationCount > 0 ? errorCount / evaluationCount : 0, + }; + } + /** + * Returns impression and conversion stats for all variants in an experiment. + */ + getExperimentStats(experimentId: string, controlVariantKey?: string): ExperimentStats { + const impressions = this.experimentImpressions.get(experimentId) ?? new Map(); + const conversions = this.experimentConversions.get(experimentId) ?? new Map(); + const allVariantKeys = new Set([...impressions.keys(), ...conversions.keys()]); + let totalImpressions = 0; + const variants: Record = {}; + for (const variantKey of allVariantKeys) { + const imp = impressions.get(variantKey) ?? 0; + const conv = conversions.get(variantKey) ?? 0; + totalImpressions += imp; + variants[variantKey] = { + variantKey, + impressions: imp, + conversions: conv, + conversionRate: imp > 0 ? conv / imp : 0, + isControl: variantKey === controlVariantKey, + }; + } + return { experimentId, totalImpressions, variants }; + } + /** + * Returns the most evaluated flags, sorted by evaluation count descending. + */ + getTopFlags(limit: number = 10): FlagSummary[] { + const summaries: FlagSummary[] = []; + for (const [flagKey, events] of this.flagEvents.entries()) { + const evaluations = events.filter((e) => e.eventType === 'evaluation'); + summaries.push({ + flagKey, + totalEvaluations: evaluations.length, + lastEvaluatedAt: events[events.length - 1]?.timestamp, + }); + } + return summaries.sort((a, b) => b.totalEvaluations - a.totalEvaluations).slice(0, limit); } - - return summaries.sort((a, b) => b.totalEvaluations - a.totalEvaluations).slice(0, limit); - } - - /** - * Returns the most recent evaluation events for a flag in reverse-chronological order. - */ - getFlagEvaluationHistory(flagKey: string, limit: number = 100): FlagAnalyticsEvent[] { - const events = this.flagEvents.get(flagKey) ?? []; - return events - .filter((e) => e.eventType === 'evaluation') - .slice(-limit) - .reverse(); - } - - /** - * Clears stored analytics. Pass a flagKey to clear only that flag's data, - * or call without arguments to wipe all analytics. - */ - clearAnalytics(flagKey?: string): void { - if (flagKey) { - this.flagEvents.delete(flagKey); - this.flagUsers.delete(flagKey); - return; + /** + * Returns the most recent evaluation events for a flag in reverse-chronological order. + */ + getFlagEvaluationHistory(flagKey: string, limit: number = 100): FlagAnalyticsEvent[] { + const events = this.flagEvents.get(flagKey) ?? []; + return events + .filter((e) => e.eventType === 'evaluation') + .slice(-limit) + .reverse(); + } + /** + * Clears stored analytics. Pass a flagKey to clear only that flag's data, + * or call without arguments to wipe all analytics. + */ + clearAnalytics(flagKey?: string): void { + if (flagKey) { + this.flagEvents.delete(flagKey); + this.flagUsers.delete(flagKey); + return; + } + this.flagEvents.clear(); + this.flagUsers.clear(); + this.experimentImpressions.clear(); + this.experimentConversions.clear(); + } + private incrementExperimentCounter(store: Map>, experimentId: string, variantKey: string): void { + if (!store.has(experimentId)) { + store.set(experimentId, new Map()); + } + const inner = store.get(experimentId) ?? new Map(); + inner.set(variantKey, (inner.get(variantKey) ?? 0) + 1); } - - this.flagEvents.clear(); - this.flagUsers.clear(); - this.experimentImpressions.clear(); - this.experimentConversions.clear(); - } - - private incrementExperimentCounter( - store: Map>, - experimentId: string, - variantKey: string, - ): void { - if (!store.has(experimentId)) { - store.set(experimentId, new Map()); + private generateEventId(): string { + return `evt_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`; } - const inner = store.get(experimentId) ?? new Map(); - inner.set(variantKey, (inner.get(variantKey) ?? 0) + 1); - } - - private generateEventId(): string { - return `evt_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`; - } } diff --git a/src/feature-flags/evaluation/flag-evaluation.service.ts b/src/feature-flags/evaluation/flag-evaluation.service.ts index a593d035..f542a680 100644 --- a/src/feature-flags/evaluation/flag-evaluation.service.ts +++ b/src/feature-flags/evaluation/flag-evaluation.service.ts @@ -1,260 +1,201 @@ import { Injectable } from '@nestjs/common'; -import { - EvaluationReason, - FeatureFlag, - FlagEvaluationResult, - FlagValueType, - UserContext, -} from '../interfaces'; +import { EvaluationReason, FeatureFlag, FlagEvaluationResult, FlagValueType, UserContext, } from '../interfaces'; import { FlagAnalyticsService } from '../analytics/flag-analytics.service'; import { ExperimentationService } from '../experimentation/experimentation.service'; import { RolloutService } from '../rollout/rollout.service'; import { TargetingService } from '../targeting/targeting.service'; - @Injectable() export class FlagEvaluationService { - private readonly flags = new Map(); - - constructor( - private readonly targetingService: TargetingService, - private readonly rolloutService: RolloutService, - private readonly experimentationService: ExperimentationService, - private readonly analyticsService: FlagAnalyticsService, - ) {} - - /** - * Evaluates a single feature flag for the given user context. - * - * Evaluation order: - * 1. Flag disabled / archived → off variation - * 2. Prerequisites → off variation if unmet - * 3. Targeting rules → matched variation - * 4. A/B experiment → assigned variant - * 5. Gradual rollout → default variation - * 6. Default → default variation - */ - evaluate(flagKey: string, userContext: UserContext): FlagEvaluationResult { - try { - const flag = this.flags.get(flagKey); - - if (!flag) { - const result = this.errorResult(flagKey); - this.recordEvaluation(result, userContext); - return result; - } - - if (flag.archived || !flag.enabled) { - const result = this.buildResult(flag, flag.offVariationKey, 'FLAG_DISABLED'); - this.recordEvaluation(result, userContext); - return result; - } - - // Prerequisites - if (flag.prerequisites?.length) { - for (const prereq of flag.prerequisites) { - const prereqResult = this.evaluate(prereq.flagKey, userContext); - if (prereqResult.variationKey !== prereq.requiredVariationKey) { - const result = this.buildResult(flag, flag.offVariationKey, 'PREREQUISITE_FAILED'); + private readonly flags = new Map(); + constructor(private readonly targetingService: TargetingService, private readonly rolloutService: RolloutService, private readonly experimentationService: ExperimentationService, private readonly analyticsService: FlagAnalyticsService) { } + /** + * Evaluates a single feature flag for the given user context. + * + * Evaluation order: + * 1. Flag disabled / archived → off variation + * 2. Prerequisites → off variation if unmet + * 3. Targeting rules → matched variation + * 4. A/B experiment → assigned variant + * 5. Gradual rollout → default variation + * 6. Default → default variation + */ + evaluate(flagKey: string, userContext: UserContext): FlagEvaluationResult { + try { + const flag = this.flags.get(flagKey); + if (!flag) { + const result = this.errorResult(flagKey); + this.recordEvaluation(result, userContext); + return result; + } + if (flag.archived || !flag.enabled) { + const result = this.buildResult(flag, flag.offVariationKey, 'FLAG_DISABLED'); + this.recordEvaluation(result, userContext); + return result; + } + // Prerequisites + if (flag.prerequisites?.length) { + for (const prereq of flag.prerequisites) { + const prereqResult = this.evaluate(prereq.flagKey, userContext); + if (prereqResult.variationKey !== prereq.requiredVariationKey) { + const result = this.buildResult(flag, flag.offVariationKey, 'PREREQUISITE_FAILED'); + this.recordEvaluation(result, userContext); + return result; + } + } + } + // Targeting rules + if (flag.targeting) { + const matchedVariationKey = this.targetingService.evaluateTargeting(flag.targeting, userContext); + if (matchedVariationKey !== null) { + const result = this.buildResult(flag, matchedVariationKey, 'TARGETING_MATCH'); + this.recordEvaluation(result, userContext); + return result; + } + } + // A/B experiment + if (flag.experiment) { + const experimentResult = this.experimentationService.assignVariant(flag.experiment, flagKey, userContext); + if (experimentResult) { + const result: FlagEvaluationResult = { + flagKey, + value: experimentResult.value, + variationKey: this.variationKeyForValue(flag, experimentResult.value), + reason: 'EXPERIMENT', + experimentId: experimentResult.experimentId, + experimentVariantKey: experimentResult.variantKey, + timestamp: new Date(), + }; + this.recordEvaluation(result, userContext); + this.analyticsService.trackImpression(experimentResult.experimentId, experimentResult.variantKey, userContext.userId, flagKey); + return result; + } + } + // Gradual rollout + if (flag.rollout) { + const inRollout = this.rolloutService.isUserInRollout(flag.rollout, flagKey, userContext); + if (inRollout) { + const result = this.buildResult(flag, flag.defaultVariationKey, 'ROLLOUT'); + this.recordEvaluation(result, userContext); + return result; + } + else { + // User is outside the rollout window → serve off variation + const result = this.buildResult(flag, flag.offVariationKey, 'DEFAULT'); + this.recordEvaluation(result, userContext); + return result; + } + } + // Default + const result = this.buildResult(flag, flag.defaultVariationKey, 'DEFAULT'); this.recordEvaluation(result, userContext); return result; - } } - } - - // Targeting rules - if (flag.targeting) { - const matchedVariationKey = this.targetingService.evaluateTargeting( - flag.targeting, - userContext, - ); - if (matchedVariationKey !== null) { - const result = this.buildResult(flag, matchedVariationKey, 'TARGETING_MATCH'); - this.recordEvaluation(result, userContext); - return result; + catch { + const flag = this.flags.get(flagKey); + const result = flag + ? this.buildResult(flag, flag.offVariationKey, 'ERROR') + : this.errorResult(flagKey); + this.recordEvaluation(result, userContext); + return result; } - } - - // A/B experiment - if (flag.experiment) { - const experimentResult = this.experimentationService.assignVariant( - flag.experiment, - flagKey, - userContext, - ); - if (experimentResult) { - const result: FlagEvaluationResult = { - flagKey, - value: experimentResult.value, - variationKey: this.variationKeyForValue(flag, experimentResult.value), - reason: 'EXPERIMENT', - experimentId: experimentResult.experimentId, - experimentVariantKey: experimentResult.variantKey, - timestamp: new Date(), - }; - - this.recordEvaluation(result, userContext); - this.analyticsService.trackImpression( - experimentResult.experimentId, - experimentResult.variantKey, - userContext.userId, - flagKey, - ); - - return result; + } + /** + * Evaluates all registered flags for the given user context. + */ + evaluateAll(userContext: UserContext): Record { + const results: Record = {}; + for (const flagKey of this.flags.keys()) { + results[flagKey] = this.evaluate(flagKey, userContext); } - } - - // Gradual rollout - if (flag.rollout) { - const inRollout = this.rolloutService.isUserInRollout(flag.rollout, flagKey, userContext); - if (inRollout) { - const result = this.buildResult(flag, flag.defaultVariationKey, 'ROLLOUT'); - this.recordEvaluation(result, userContext); - return result; - } else { - // User is outside the rollout window → serve off variation - const result = this.buildResult(flag, flag.offVariationKey, 'DEFAULT'); - this.recordEvaluation(result, userContext); - return result; + return results; + } + /** + * Convenience method — returns the boolean value of a flag. + */ + evaluateBoolean(flagKey: string, userContext: UserContext, defaultValue = false): boolean { + const result = this.evaluate(flagKey, userContext); + return result.reason === 'ERROR' ? defaultValue : Boolean(result.value); + } + /** + * Convenience method — returns the string value of a flag. + */ + evaluateString(flagKey: string, userContext: UserContext, defaultValue = ''): string { + const result = this.evaluate(flagKey, userContext); + return result.reason === 'ERROR' ? defaultValue : String(result.value); + } + /** + * Convenience method — returns the numeric value of a flag. + */ + evaluateNumber(flagKey: string, userContext: UserContext, defaultValue = 0): number { + const result = this.evaluate(flagKey, userContext); + return result.reason === 'ERROR' ? defaultValue : Number(result.value); + } + // --------------------------------------------------------------------------- + // Flag management + // --------------------------------------------------------------------------- + setFlag(flag: FeatureFlag): void { + this.flags.set(flag.key, { ...flag, updatedAt: new Date() }); + } + setFlags(flags: FeatureFlag[]): void { + for (const flag of flags) { + this.setFlag(flag); } - } - - // Default - const result = this.buildResult(flag, flag.defaultVariationKey, 'DEFAULT'); - this.recordEvaluation(result, userContext); - return result; - } catch { - const flag = this.flags.get(flagKey); - const result = flag - ? this.buildResult(flag, flag.offVariationKey, 'ERROR') - : this.errorResult(flagKey); - this.recordEvaluation(result, userContext); - return result; } - } - - /** - * Evaluates all registered flags for the given user context. - */ - evaluateAll(userContext: UserContext): Record { - const results: Record = {}; - for (const flagKey of this.flags.keys()) { - results[flagKey] = this.evaluate(flagKey, userContext); + updateFlag(flagKey: string, updates: Partial>): FeatureFlag | null { + const existing = this.flags.get(flagKey); + if (!existing) + return null; + const updated: FeatureFlag = { + ...existing, + ...updates, + key: existing.key, + id: existing.id, + version: existing.version + 1, + updatedAt: new Date(), + }; + this.flags.set(flagKey, updated); + return updated; + } + removeFlag(flagKey: string): boolean { + return this.flags.delete(flagKey); + } + getFlag(flagKey: string): FeatureFlag | undefined { + return this.flags.get(flagKey); + } + getAllFlags(): FeatureFlag[] { + return Array.from(this.flags.values()); + } + // --------------------------------------------------------------------------- + // Helpers + // --------------------------------------------------------------------------- + private buildResult(flag: FeatureFlag, variationKey: string, reason: EvaluationReason, ruleId?: string): FlagEvaluationResult { + const variation = flag.variations.find((v) => v.key === variationKey); + const value: FlagValueType = variation?.value ?? flag.variations[0]?.value ?? false; + return { flagKey: flag.key, value, variationKey, reason, ruleId, timestamp: new Date() }; + } + private errorResult(flagKey: string): FlagEvaluationResult { + return { + flagKey, + value: false, + variationKey: 'off', + reason: 'ERROR', + timestamp: new Date(), + }; + } + private variationKeyForValue(flag: FeatureFlag, value: FlagValueType): string { + return flag.variations.find((v) => v.value === value)?.key ?? flag.defaultVariationKey; } - return results; - } - - /** - * Convenience method — returns the boolean value of a flag. - */ - evaluateBoolean(flagKey: string, userContext: UserContext, defaultValue = false): boolean { - const result = this.evaluate(flagKey, userContext); - return result.reason === 'ERROR' ? defaultValue : Boolean(result.value); - } - - /** - * Convenience method — returns the string value of a flag. - */ - evaluateString(flagKey: string, userContext: UserContext, defaultValue = ''): string { - const result = this.evaluate(flagKey, userContext); - return result.reason === 'ERROR' ? defaultValue : String(result.value); - } - - /** - * Convenience method — returns the numeric value of a flag. - */ - evaluateNumber(flagKey: string, userContext: UserContext, defaultValue = 0): number { - const result = this.evaluate(flagKey, userContext); - return result.reason === 'ERROR' ? defaultValue : Number(result.value); - } - - // --------------------------------------------------------------------------- - // Flag management - // --------------------------------------------------------------------------- - - setFlag(flag: FeatureFlag): void { - this.flags.set(flag.key, { ...flag, updatedAt: new Date() }); - } - - setFlags(flags: FeatureFlag[]): void { - for (const flag of flags) { - this.setFlag(flag); + private recordEvaluation(result: FlagEvaluationResult, userContext: UserContext): void { + this.analyticsService.trackEvaluation({ + eventType: 'evaluation', + flagKey: result.flagKey, + userId: userContext.userId, + sessionId: userContext.sessionId, + variationKey: result.variationKey, + experimentId: result.experimentId, + experimentVariantKey: result.experimentVariantKey, + reason: result.reason, + value: result.value, + }); } - } - - updateFlag( - flagKey: string, - updates: Partial>, - ): FeatureFlag | null { - const existing = this.flags.get(flagKey); - if (!existing) return null; - - const updated: FeatureFlag = { - ...existing, - ...updates, - key: existing.key, - id: existing.id, - version: existing.version + 1, - updatedAt: new Date(), - }; - - this.flags.set(flagKey, updated); - return updated; - } - - removeFlag(flagKey: string): boolean { - return this.flags.delete(flagKey); - } - - getFlag(flagKey: string): FeatureFlag | undefined { - return this.flags.get(flagKey); - } - - getAllFlags(): FeatureFlag[] { - return Array.from(this.flags.values()); - } - - // --------------------------------------------------------------------------- - // Helpers - // --------------------------------------------------------------------------- - - private buildResult( - flag: FeatureFlag, - variationKey: string, - reason: EvaluationReason, - ruleId?: string, - ): FlagEvaluationResult { - const variation = flag.variations.find((v) => v.key === variationKey); - const value: FlagValueType = variation?.value ?? flag.variations[0]?.value ?? false; - - return { flagKey: flag.key, value, variationKey, reason, ruleId, timestamp: new Date() }; - } - - private errorResult(flagKey: string): FlagEvaluationResult { - return { - flagKey, - value: false, - variationKey: 'off', - reason: 'ERROR', - timestamp: new Date(), - }; - } - - private variationKeyForValue(flag: FeatureFlag, value: FlagValueType): string { - return flag.variations.find((v) => v.value === value)?.key ?? flag.defaultVariationKey; - } - - private recordEvaluation(result: FlagEvaluationResult, userContext: UserContext): void { - this.analyticsService.trackEvaluation({ - eventType: 'evaluation', - flagKey: result.flagKey, - userId: userContext.userId, - sessionId: userContext.sessionId, - variationKey: result.variationKey, - experimentId: result.experimentId, - experimentVariantKey: result.experimentVariantKey, - reason: result.reason, - value: result.value, - }); - } } diff --git a/src/feature-flags/experimentation/experimentation.service.ts b/src/feature-flags/experimentation/experimentation.service.ts index af77d6e1..bf722b08 100644 --- a/src/feature-flags/experimentation/experimentation.service.ts +++ b/src/feature-flags/experimentation/experimentation.service.ts @@ -1,147 +1,109 @@ import { Injectable } from '@nestjs/common'; import { ExperimentConfig, ExperimentResult, ExperimentVariant, UserContext } from '../interfaces'; import { RolloutService } from '../rollout/rollout.service'; - interface ConversionRecord { - eventName: string; - metadata?: Record; - timestamp: Date; + eventName: string; + metadata?: Record; + timestamp: Date; } - @Injectable() export class ExperimentationService { - /** variantKey → userId → ConversionRecord[] */ - private readonly conversions = new Map>(); - - constructor(private readonly rolloutService: RolloutService) {} - - /** - * Assigns a user to an experiment variant using consistent hashing. - * Returns null if the experiment is inactive or the user is outside traffic allocation. - */ - assignVariant( - config: ExperimentConfig, - flagKey: string, - userContext: UserContext, - ): ExperimentResult | null { - if (config.status !== 'running') return null; - - const now = new Date(); - if (config.startDate && now < config.startDate) return null; - if (config.endDate && now > config.endDate) return null; - - if (!this.isInExperimentTraffic(config, flagKey, userContext)) return null; - - const variant = this.selectVariant(config, flagKey, userContext); - if (!variant) return null; - - return { - experimentId: config.experimentId, - variantKey: variant.key, - value: variant.value, - isControl: variant.isControl ?? false, - }; - } - - /** - * Records a conversion event for a user in an experiment. - */ - trackConversion( - experimentId: string, - userId: string, - eventName: string, - metadata?: Record, - ): void { - if (!this.conversions.has(experimentId)) { - this.conversions.set(experimentId, new Map()); + /** variantKey → userId → ConversionRecord[] */ + private readonly conversions = new Map>(); + constructor(private readonly rolloutService: RolloutService) { } + /** + * Assigns a user to an experiment variant using consistent hashing. + * Returns null if the experiment is inactive or the user is outside traffic allocation. + */ + assignVariant(config: ExperimentConfig, flagKey: string, userContext: UserContext): ExperimentResult | null { + if (config.status !== 'running') + return null; + const now = new Date(); + if (config.startDate && now < config.startDate) + return null; + if (config.endDate && now > config.endDate) + return null; + if (!this.isInExperimentTraffic(config, flagKey, userContext)) + return null; + const variant = this.selectVariant(config, flagKey, userContext); + if (!variant) + return null; + return { + experimentId: config.experimentId, + variantKey: variant.key, + value: variant.value, + isControl: variant.isControl ?? false, + }; } - - const expConversions = this.conversions.get(experimentId) ?? new Map(); - - if (!expConversions.has(userId)) { - expConversions.set(userId, []); + /** + * Records a conversion event for a user in an experiment. + */ + trackConversion(experimentId: string, userId: string, eventName: string, metadata?: Record): void { + if (!this.conversions.has(experimentId)) { + this.conversions.set(experimentId, new Map()); + } + const expConversions = this.conversions.get(experimentId) ?? new Map(); + if (!expConversions.has(userId)) { + expConversions.set(userId, []); + } + const userConversions = expConversions.get(userId) ?? []; + userConversions.push({ eventName, metadata, timestamp: new Date() }); } - - const userConversions = expConversions.get(userId) ?? []; - userConversions.push({ eventName, metadata, timestamp: new Date() }); - } - - /** - * Returns all recorded conversion records for an experiment. - */ - getConversions(experimentId: string): Map { - return this.conversions.get(experimentId) ?? new Map(); - } - - /** - * Returns the list of experiment IDs that currently have conversion data. - */ - getActiveExperimentIds(): string[] { - return Array.from(this.conversions.keys()); - } - - /** - * Checks whether a user falls within the experiment's traffic allocation percentage. - * Uses a separate hash seed from variant assignment to avoid correlation. - */ - private isInExperimentTraffic( - config: ExperimentConfig, - flagKey: string, - userContext: UserContext, - ): boolean { - const bucketValue = this.resolveBucketAttributeValue( - config.bucketByAttribute ?? 'userId', - userContext, - ); - const trafficBucketKey = `${flagKey}:${config.experimentId}:traffic:${bucketValue}`; - const bucket = this.rolloutService.computeBucketValue(trafficBucketKey); - return bucket < config.trafficAllocation; - } - - /** - * Selects a variant for the user using weighted bucket assignment. - * The same user always receives the same variant for the same experiment. - */ - private selectVariant( - config: ExperimentConfig, - flagKey: string, - userContext: UserContext, - ): ExperimentVariant | null { - if (!config.variants || config.variants.length === 0) return null; - - const bucketValue = this.resolveBucketAttributeValue( - config.bucketByAttribute ?? 'userId', - userContext, - ); - const variantBucketKey = `${flagKey}:${config.experimentId}:variant:${bucketValue}`; - const bucket = this.rolloutService.computeBucketValue(variantBucketKey); - - const totalWeight = config.variants.reduce((sum, v) => sum + v.weight, 0); - if (totalWeight === 0) return null; - - const normalizedBucket = (bucket / 100) * totalWeight; - let cumulative = 0; - - for (const variant of config.variants) { - cumulative += variant.weight; - if (normalizedBucket < cumulative) { - return variant; - } + /** + * Returns all recorded conversion records for an experiment. + */ + getConversions(experimentId: string): Map { + return this.conversions.get(experimentId) ?? new Map(); } - - return config.variants[config.variants.length - 1]; - } - - private resolveBucketAttributeValue(attribute: string, userContext: UserContext): string { - switch (attribute) { - case 'userId': - return userContext.userId; - case 'sessionId': - return userContext.sessionId ?? userContext.userId; - case 'email': - return userContext.email ?? userContext.userId; - default: - return userContext.attributes?.[attribute]?.toString() ?? userContext.userId; + /** + * Returns the list of experiment IDs that currently have conversion data. + */ + getActiveExperimentIds(): string[] { + return Array.from(this.conversions.keys()); + } + /** + * Checks whether a user falls within the experiment's traffic allocation percentage. + * Uses a separate hash seed from variant assignment to avoid correlation. + */ + private isInExperimentTraffic(config: ExperimentConfig, flagKey: string, userContext: UserContext): boolean { + const bucketValue = this.resolveBucketAttributeValue(config.bucketByAttribute ?? 'userId', userContext); + const trafficBucketKey = `${flagKey}:${config.experimentId}:traffic:${bucketValue}`; + const bucket = this.rolloutService.computeBucketValue(trafficBucketKey); + return bucket < config.trafficAllocation; + } + /** + * Selects a variant for the user using weighted bucket assignment. + * The same user always receives the same variant for the same experiment. + */ + private selectVariant(config: ExperimentConfig, flagKey: string, userContext: UserContext): ExperimentVariant | null { + if (!config.variants || config.variants.length === 0) + return null; + const bucketValue = this.resolveBucketAttributeValue(config.bucketByAttribute ?? 'userId', userContext); + const variantBucketKey = `${flagKey}:${config.experimentId}:variant:${bucketValue}`; + const bucket = this.rolloutService.computeBucketValue(variantBucketKey); + const totalWeight = config.variants.reduce((sum, v) => sum + v.weight, 0); + if (totalWeight === 0) + return null; + const normalizedBucket = (bucket / 100) * totalWeight; + let cumulative = 0; + for (const variant of config.variants) { + cumulative += variant.weight; + if (normalizedBucket < cumulative) { + return variant; + } + } + return config.variants[config.variants.length - 1]; + } + private resolveBucketAttributeValue(attribute: string, userContext: UserContext): string { + switch (attribute) { + case 'userId': + return userContext.userId; + case 'sessionId': + return userContext.sessionId ?? userContext.userId; + case 'email': + return userContext.email ?? userContext.userId; + default: + return userContext.attributes?.[attribute]?.toString() ?? userContext.userId; + } } - } } diff --git a/src/feature-flags/feature-flags.module.ts b/src/feature-flags/feature-flags.module.ts index 06e39cb2..255bf527 100644 --- a/src/feature-flags/feature-flags.module.ts +++ b/src/feature-flags/feature-flags.module.ts @@ -4,21 +4,21 @@ import { TargetingService } from './targeting/targeting.service'; import { RolloutService } from './rollout/rollout.service'; import { ExperimentationService } from './experimentation/experimentation.service'; import { FlagAnalyticsService } from './analytics/flag-analytics.service'; - @Module({ - providers: [ - FlagAnalyticsService, - TargetingService, - RolloutService, - ExperimentationService, - FlagEvaluationService, - ], - exports: [ - FlagEvaluationService, - TargetingService, - RolloutService, - ExperimentationService, - FlagAnalyticsService, - ], + providers: [ + FlagAnalyticsService, + TargetingService, + RolloutService, + ExperimentationService, + FlagEvaluationService, + ], + exports: [ + FlagEvaluationService, + TargetingService, + RolloutService, + ExperimentationService, + FlagAnalyticsService, + ], }) -export class FeatureFlagsModule {} +export class FeatureFlagsModule { +} diff --git a/src/feature-flags/interfaces/index.ts b/src/feature-flags/interfaces/index.ts index 279de7e4..4ddfa65f 100644 --- a/src/feature-flags/interfaces/index.ts +++ b/src/feature-flags/interfaces/index.ts @@ -1,199 +1,155 @@ export type FlagValueType = boolean | string | number; - -export type EvaluationReason = - | 'FLAG_DISABLED' - | 'PREREQUISITE_FAILED' - | 'TARGETING_MATCH' - | 'ROLLOUT' - | 'EXPERIMENT' - | 'DEFAULT' - | 'ERROR'; - -export type ConditionOperator = - | 'equals' - | 'notEquals' - | 'contains' - | 'notContains' - | 'startsWith' - | 'endsWith' - | 'greaterThan' - | 'greaterThanOrEqual' - | 'lessThan' - | 'lessThanOrEqual' - | 'in' - | 'notIn' - | 'regex' - | 'exists' - | 'notExists'; - +export type EvaluationReason = 'FLAG_DISABLED' | 'PREREQUISITE_FAILED' | 'TARGETING_MATCH' | 'ROLLOUT' | 'EXPERIMENT' | 'DEFAULT' | 'ERROR'; +export type ConditionOperator = 'equals' | 'notEquals' | 'contains' | 'notContains' | 'startsWith' | 'endsWith' | 'greaterThan' | 'greaterThanOrEqual' | 'lessThan' | 'lessThanOrEqual' | 'in' | 'notIn' | 'regex' | 'exists' | 'notExists'; export type FlagType = 'boolean' | 'string' | 'number'; - export type ExperimentStatus = 'draft' | 'running' | 'paused' | 'completed'; - export interface UserContext { - userId: string; - email?: string; - country?: string; - plan?: string; - roles?: string[]; - groups?: string[]; - attributes?: Record; - sessionId?: string; - ipAddress?: string; -} - + userId: string; + email?: string; + country?: string; + plan?: string; + roles?: string[]; + groups?: string[]; + attributes?: Record; + sessionId?: string; + ipAddress?: string; +} export interface TargetingCondition { - attribute: string; - operator: ConditionOperator; - value?: FlagValueType | FlagValueType[]; + attribute: string; + operator: ConditionOperator; + value?: FlagValueType | FlagValueType[]; } - export interface TargetingRule { - id: string; - name?: string; - conditions: TargetingCondition[]; - conditionsOperator: 'AND' | 'OR'; - serveVariationKey: string; - priority: number; -} - + id: string; + name?: string; + conditions: TargetingCondition[]; + conditionsOperator: 'AND' | 'OR'; + serveVariationKey: string; + priority: number; +} export interface TargetingConfig { - rules: TargetingRule[]; - defaultServeVariationKey: string; + rules: TargetingRule[]; + defaultServeVariationKey: string; } - export interface RampStep { - at: Date; - percentage: number; + at: Date; + percentage: number; } - export interface RolloutConfig { - percentage: number; - bucketByAttribute?: string; - sticky?: boolean; - startDate?: Date; - endDate?: Date; - rampSchedule?: RampStep[]; -} - + percentage: number; + bucketByAttribute?: string; + sticky?: boolean; + startDate?: Date; + endDate?: Date; + rampSchedule?: RampStep[]; +} export interface ExperimentVariant { - key: string; - name: string; - weight: number; - value: FlagValueType; - isControl?: boolean; - description?: string; -} - + key: string; + name: string; + weight: number; + value: FlagValueType; + isControl?: boolean; + description?: string; +} export interface ExperimentConfig { - experimentId: string; - name: string; - hypothesis?: string; - variants: ExperimentVariant[]; - trafficAllocation: number; - bucketByAttribute?: string; - startDate?: Date; - endDate?: Date; - status: ExperimentStatus; - winnerVariantKey?: string; - metrics?: string[]; -} - + experimentId: string; + name: string; + hypothesis?: string; + variants: ExperimentVariant[]; + trafficAllocation: number; + bucketByAttribute?: string; + startDate?: Date; + endDate?: Date; + status: ExperimentStatus; + winnerVariantKey?: string; + metrics?: string[]; +} export interface FlagVariation { - key: string; - name: string; - value: FlagValueType; - description?: string; + key: string; + name: string; + value: FlagValueType; + description?: string; } - export interface FlagPrerequisite { - flagKey: string; - requiredVariationKey: string; + flagKey: string; + requiredVariationKey: string; } - export interface FeatureFlag { - id: string; - key: string; - name: string; - description?: string; - type: FlagType; - enabled: boolean; - defaultVariationKey: string; - offVariationKey: string; - variations: FlagVariation[]; - targeting?: TargetingConfig; - rollout?: RolloutConfig; - experiment?: ExperimentConfig; - prerequisites?: FlagPrerequisite[]; - tags?: string[]; - projectId?: string; - environmentId?: string; - version: number; - archived: boolean; - createdAt: Date; - updatedAt: Date; - createdBy?: string; - updatedBy?: string; -} - + id: string; + key: string; + name: string; + description?: string; + type: FlagType; + enabled: boolean; + defaultVariationKey: string; + offVariationKey: string; + variations: FlagVariation[]; + targeting?: TargetingConfig; + rollout?: RolloutConfig; + experiment?: ExperimentConfig; + prerequisites?: FlagPrerequisite[]; + tags?: string[]; + projectId?: string; + environmentId?: string; + version: number; + archived: boolean; + createdAt: Date; + updatedAt: Date; + createdBy?: string; + updatedBy?: string; +} export interface FlagEvaluationResult { - flagKey: string; - value: FlagValueType; - variationKey: string; - reason: EvaluationReason; - ruleId?: string; - experimentId?: string; - experimentVariantKey?: string; - timestamp: Date; -} - + flagKey: string; + value: FlagValueType; + variationKey: string; + reason: EvaluationReason; + ruleId?: string; + experimentId?: string; + experimentVariantKey?: string; + timestamp: Date; +} export interface ExperimentResult { - experimentId: string; - variantKey: string; - value: FlagValueType; - isControl: boolean; + experimentId: string; + variantKey: string; + value: FlagValueType; + isControl: boolean; } - export interface FlagAnalyticsEvent { - eventId: string; - eventType: 'evaluation' | 'impression' | 'conversion' | 'error'; - flagKey?: string; - userId?: string; - sessionId?: string; - variationKey?: string; - experimentId?: string; - experimentVariantKey?: string; - reason?: EvaluationReason; - value?: FlagValueType; - metadata?: Record; - timestamp: Date; -} - + eventId: string; + eventType: 'evaluation' | 'impression' | 'conversion' | 'error'; + flagKey?: string; + userId?: string; + sessionId?: string; + variationKey?: string; + experimentId?: string; + experimentVariantKey?: string; + reason?: EvaluationReason; + value?: FlagValueType; + metadata?: Record; + timestamp: Date; +} export interface FlagEvaluationStats { - flagKey: string; - totalEvaluations: number; - evaluationsByVariation: Record; - evaluationsByReason: Record; - uniqueUsers: number; - errorRate: number; -} - + flagKey: string; + totalEvaluations: number; + evaluationsByVariation: Record; + evaluationsByReason: Record; + uniqueUsers: number; + errorRate: number; +} export interface ExperimentVariantStats { - variantKey: string; - impressions: number; - conversions: number; - conversionRate: number; - isControl: boolean; + variantKey: string; + impressions: number; + conversions: number; + conversionRate: number; + isControl: boolean; } - export interface ExperimentStats { - experimentId: string; - totalImpressions: number; - variants: Record; + experimentId: string; + totalImpressions: number; + variants: Record; } - export interface FlagSummary { - flagKey: string; - totalEvaluations: number; - lastEvaluatedAt?: Date; + flagKey: string; + totalEvaluations: number; + lastEvaluatedAt?: Date; } diff --git a/src/feature-flags/rollout/rollout.service.ts b/src/feature-flags/rollout/rollout.service.ts index 0576401c..8946177b 100644 --- a/src/feature-flags/rollout/rollout.service.ts +++ b/src/feature-flags/rollout/rollout.service.ts @@ -1,74 +1,68 @@ import { Injectable } from '@nestjs/common'; import { RolloutConfig, UserContext } from '../interfaces'; - @Injectable() export class RolloutService { - /** - * Determines whether a user falls within the configured rollout percentage. - * Uses consistent hashing so the same user always gets the same result. - */ - isUserInRollout(config: RolloutConfig, flagKey: string, userContext: UserContext): boolean { - const now = new Date(); - - if (config.startDate && now < config.startDate) return false; - if (config.endDate && now > config.endDate) return false; - - const currentPercentage = this.getCurrentPercentage(config); - if (currentPercentage <= 0) return false; - if (currentPercentage >= 100) return true; - - const bucketKey = this.resolveBucketKey(config.bucketByAttribute ?? 'userId', userContext); - const bucketValue = this.computeBucketValue(`${flagKey}:${bucketKey}`); - - return bucketValue < currentPercentage; - } - - /** - * Returns the effective rollout percentage at the current time, - * accounting for any ramp schedule defined on the config. - */ - getCurrentPercentage(config: RolloutConfig): number { - if (!config.rampSchedule || config.rampSchedule.length === 0) { - return config.percentage; + /** + * Determines whether a user falls within the configured rollout percentage. + * Uses consistent hashing so the same user always gets the same result. + */ + isUserInRollout(config: RolloutConfig, flagKey: string, userContext: UserContext): boolean { + const now = new Date(); + if (config.startDate && now < config.startDate) + return false; + if (config.endDate && now > config.endDate) + return false; + const currentPercentage = this.getCurrentPercentage(config); + if (currentPercentage <= 0) + return false; + if (currentPercentage >= 100) + return true; + const bucketKey = this.resolveBucketKey(config.bucketByAttribute ?? 'userId', userContext); + const bucketValue = this.computeBucketValue(`${flagKey}:${bucketKey}`); + return bucketValue < currentPercentage; } - - const now = new Date(); - const sortedSteps = [...config.rampSchedule].sort((a, b) => a.at.getTime() - b.at.getTime()); - - let effective = 0; - for (const step of sortedSteps) { - if (now >= step.at) { - effective = step.percentage; - } else { - break; - } + /** + * Returns the effective rollout percentage at the current time, + * accounting for any ramp schedule defined on the config. + */ + getCurrentPercentage(config: RolloutConfig): number { + if (!config.rampSchedule || config.rampSchedule.length === 0) { + return config.percentage; + } + const now = new Date(); + const sortedSteps = [...config.rampSchedule].sort((a, b) => a.at.getTime() - b.at.getTime()); + let effective = 0; + for (const step of sortedSteps) { + if (now >= step.at) { + effective = step.percentage; + } + else { + break; + } + } + return Math.min(effective, config.percentage); } - - return Math.min(effective, config.percentage); - } - - /** - * DJB2 hash — fast, deterministic, and well-distributed. - * Returns a value in the range [0, 99]. - */ - computeBucketValue(key: string): number { - let hash = 5381; - for (let i = 0; i < key.length; i++) { - hash = ((hash << 5) + hash + key.charCodeAt(i)) >>> 0; + /** + * DJB2 hash — fast, deterministic, and well-distributed. + * Returns a value in the range [0, 99]. + */ + computeBucketValue(key: string): number { + let hash = 5381; + for (let i = 0; i < key.length; i++) { + hash = ((hash << 5) + hash + key.charCodeAt(i)) >>> 0; + } + return hash % 100; } - return hash % 100; - } - - private resolveBucketKey(attribute: string, userContext: UserContext): string { - switch (attribute) { - case 'userId': - return userContext.userId; - case 'sessionId': - return userContext.sessionId ?? userContext.userId; - case 'email': - return userContext.email ?? userContext.userId; - default: - return userContext.attributes?.[attribute]?.toString() ?? userContext.userId; + private resolveBucketKey(attribute: string, userContext: UserContext): string { + switch (attribute) { + case 'userId': + return userContext.userId; + case 'sessionId': + return userContext.sessionId ?? userContext.userId; + case 'email': + return userContext.email ?? userContext.userId; + default: + return userContext.attributes?.[attribute]?.toString() ?? userContext.userId; + } } - } } diff --git a/src/feature-flags/targeting/targeting.service.ts b/src/feature-flags/targeting/targeting.service.ts index 3fdc6098..c1d3e248 100644 --- a/src/feature-flags/targeting/targeting.service.ts +++ b/src/feature-flags/targeting/targeting.service.ts @@ -1,158 +1,113 @@ import { Injectable } from '@nestjs/common'; -import { - ConditionOperator, - FlagValueType, - TargetingCondition, - TargetingConfig, - TargetingRule, - UserContext, -} from '../interfaces'; - +import { ConditionOperator, FlagValueType, TargetingCondition, TargetingConfig, TargetingRule, UserContext, } from '../interfaces'; @Injectable() export class TargetingService { - /** - * Evaluates targeting rules against a user context. - * Returns the matched variation key, or null if no rule matches. - */ - evaluateTargeting(config: TargetingConfig, userContext: UserContext): string | null { - const sortedRules = [...config.rules].sort((a, b) => a.priority - b.priority); - - for (const rule of sortedRules) { - if (this.evaluateRule(rule, userContext)) { - return rule.serveVariationKey; - } - } - - return null; - } - - private evaluateRule(rule: TargetingRule, userContext: UserContext): boolean { - if (!rule.conditions || rule.conditions.length === 0) return false; - - if (rule.conditionsOperator === 'OR') { - return rule.conditions.some((c) => this.evaluateCondition(c, userContext)); - } - - return rule.conditions.every((c) => this.evaluateCondition(c, userContext)); - } - - private evaluateCondition(condition: TargetingCondition, userContext: UserContext): boolean { - const attributeValue = this.resolveAttribute(condition.attribute, userContext); - - return this.applyOperator(condition.operator, attributeValue, condition.value); - } - - private applyOperator( - operator: ConditionOperator, - attributeValue: unknown, - conditionValue?: FlagValueType | FlagValueType[], - ): boolean { - switch (operator) { - case 'exists': - return attributeValue !== null && attributeValue !== undefined; - - case 'notExists': - return attributeValue === null || attributeValue === undefined; - - case 'equals': - return String(attributeValue) === String(conditionValue); - - case 'notEquals': - return String(attributeValue) !== String(conditionValue); - - case 'contains': - return ( - typeof attributeValue === 'string' && - typeof conditionValue === 'string' && - attributeValue.toLowerCase().includes(conditionValue.toLowerCase()) - ); - - case 'notContains': - return ( - typeof attributeValue === 'string' && - typeof conditionValue === 'string' && - !attributeValue.toLowerCase().includes(conditionValue.toLowerCase()) - ); - - case 'startsWith': - return ( - typeof attributeValue === 'string' && - typeof conditionValue === 'string' && - attributeValue.toLowerCase().startsWith(conditionValue.toLowerCase()) - ); - - case 'endsWith': - return ( - typeof attributeValue === 'string' && - typeof conditionValue === 'string' && - attributeValue.toLowerCase().endsWith(conditionValue.toLowerCase()) - ); - - case 'greaterThan': - return Number(attributeValue) > Number(conditionValue); - - case 'greaterThanOrEqual': - return Number(attributeValue) >= Number(conditionValue); - - case 'lessThan': - return Number(attributeValue) < Number(conditionValue); - - case 'lessThanOrEqual': - return Number(attributeValue) <= Number(conditionValue); - - case 'in': - if (!Array.isArray(conditionValue)) return false; - return conditionValue.map(String).includes(String(attributeValue)); - - case 'notIn': - if (!Array.isArray(conditionValue)) return false; - return !conditionValue.map(String).includes(String(attributeValue)); - - case 'regex': - if (typeof conditionValue !== 'string' || typeof attributeValue !== 'string') { - return false; - } - try { - return new RegExp(conditionValue).test(attributeValue); - } catch { - return false; + /** + * Evaluates targeting rules against a user context. + * Returns the matched variation key, or null if no rule matches. + */ + evaluateTargeting(config: TargetingConfig, userContext: UserContext): string | null { + const sortedRules = [...config.rules].sort((a, b) => a.priority - b.priority); + for (const rule of sortedRules) { + if (this.evaluateRule(rule, userContext)) { + return rule.serveVariationKey; + } } - - default: - return false; + return null; } - } - - /** - * Resolves an attribute name from the user context. - * Checks top-level properties first, then custom attributes map. - */ - private resolveAttribute(attribute: string, userContext: UserContext): unknown { - const topLevel: Record = { - userId: userContext.userId, - email: userContext.email, - country: userContext.country, - plan: userContext.plan, - sessionId: userContext.sessionId, - ipAddress: userContext.ipAddress, - }; - - if (attribute in topLevel) { - return topLevel[attribute]; + private evaluateRule(rule: TargetingRule, userContext: UserContext): boolean { + if (!rule.conditions || rule.conditions.length === 0) + return false; + if (rule.conditionsOperator === 'OR') { + return rule.conditions.some((c) => this.evaluateCondition(c, userContext)); + } + return rule.conditions.every((c) => this.evaluateCondition(c, userContext)); } - - if (attribute === 'roles') { - return userContext.roles?.join(',') ?? null; + private evaluateCondition(condition: TargetingCondition, userContext: UserContext): boolean { + const attributeValue = this.resolveAttribute(condition.attribute, userContext); + return this.applyOperator(condition.operator, attributeValue, condition.value); } - - if (attribute === 'groups') { - return userContext.groups?.join(',') ?? null; + private applyOperator(operator: ConditionOperator, attributeValue: unknown, conditionValue?: FlagValueType | FlagValueType[]): boolean { + switch (operator) { + case 'exists': + return attributeValue !== null && attributeValue !== undefined; + case 'notExists': + return attributeValue === null || attributeValue === undefined; + case 'equals': + return String(attributeValue) === String(conditionValue); + case 'notEquals': + return String(attributeValue) !== String(conditionValue); + case 'contains': + return (typeof attributeValue === 'string' && + typeof conditionValue === 'string' && + attributeValue.toLowerCase().includes(conditionValue.toLowerCase())); + case 'notContains': + return (typeof attributeValue === 'string' && + typeof conditionValue === 'string' && + !attributeValue.toLowerCase().includes(conditionValue.toLowerCase())); + case 'startsWith': + return (typeof attributeValue === 'string' && + typeof conditionValue === 'string' && + attributeValue.toLowerCase().startsWith(conditionValue.toLowerCase())); + case 'endsWith': + return (typeof attributeValue === 'string' && + typeof conditionValue === 'string' && + attributeValue.toLowerCase().endsWith(conditionValue.toLowerCase())); + case 'greaterThan': + return Number(attributeValue) > Number(conditionValue); + case 'greaterThanOrEqual': + return Number(attributeValue) >= Number(conditionValue); + case 'lessThan': + return Number(attributeValue) < Number(conditionValue); + case 'lessThanOrEqual': + return Number(attributeValue) <= Number(conditionValue); + case 'in': + if (!Array.isArray(conditionValue)) + return false; + return conditionValue.map(String).includes(String(attributeValue)); + case 'notIn': + if (!Array.isArray(conditionValue)) + return false; + return !conditionValue.map(String).includes(String(attributeValue)); + case 'regex': + if (typeof conditionValue !== 'string' || typeof attributeValue !== 'string') { + return false; + } + try { + return new RegExp(conditionValue).test(attributeValue); + } + catch { + return false; + } + default: + return false; + } } - - if (userContext.attributes && attribute in userContext.attributes) { - return userContext.attributes[attribute]; + /** + * Resolves an attribute name from the user context. + * Checks top-level properties first, then custom attributes map. + */ + private resolveAttribute(attribute: string, userContext: UserContext): unknown { + const topLevel: Record = { + userId: userContext.userId, + email: userContext.email, + country: userContext.country, + plan: userContext.plan, + sessionId: userContext.sessionId, + ipAddress: userContext.ipAddress, + }; + if (attribute in topLevel) { + return topLevel[attribute]; + } + if (attribute === 'roles') { + return userContext.roles?.join(',') ?? null; + } + if (attribute === 'groups') { + return userContext.groups?.join(',') ?? null; + } + if (userContext.attributes && attribute in userContext.attributes) { + return userContext.attributes[attribute]; + } + return null; } - - return null; - } } diff --git a/src/gamification/badges/badges.service.ts b/src/gamification/badges/badges.service.ts index 1fbd63f4..2174fed9 100644 --- a/src/gamification/badges/badges.service.ts +++ b/src/gamification/badges/badges.service.ts @@ -4,41 +4,33 @@ import { Repository } from 'typeorm'; import { Badge } from '../entities/badge.entity'; import { UserBadge } from '../entities/user-badge.entity'; import { User } from '../../users/entities/user.entity'; - @Injectable() export class BadgesService { - constructor( + constructor( @InjectRepository(Badge) - private badgeRepository: Repository, + private badgeRepository: Repository, @InjectRepository(UserBadge) - private userBadgeRepository: Repository, - ) {} - - async awardBadge(userId: string, badgeId: string): Promise { - const existingBadge = await this.userBadgeRepository.findOne({ - where: { user: { id: userId }, badge: { id: badgeId } }, - }); - - if (existingBadge) { - return existingBadge; + private userBadgeRepository: Repository) { } + async awardBadge(userId: string, badgeId: string): Promise { + const existingBadge = await this.userBadgeRepository.findOne({ + where: { user: { id: userId }, badge: { id: badgeId } }, + }); + if (existingBadge) { + return existingBadge; + } + const userBadge = this.userBadgeRepository.create({ + user: { id: userId } as User, + badge: { id: badgeId } as Badge, + }); + return await this.userBadgeRepository.save(userBadge); + } + async getUserBadges(userId: string): Promise { + return await this.userBadgeRepository.find({ + where: { user: { id: userId } }, + relations: ['badge'], + }); + } + async getAllBadges(): Promise { + return await this.badgeRepository.find(); } - - const userBadge = this.userBadgeRepository.create({ - user: { id: userId } as User, - badge: { id: badgeId } as Badge, - }); - - return await this.userBadgeRepository.save(userBadge); - } - - async getUserBadges(userId: string): Promise { - return await this.userBadgeRepository.find({ - where: { user: { id: userId } }, - relations: ['badge'], - }); - } - - async getAllBadges(): Promise { - return await this.badgeRepository.find(); - } } diff --git a/src/gamification/challenges/challenges.service.ts b/src/gamification/challenges/challenges.service.ts index fa293c90..89b98b21 100644 --- a/src/gamification/challenges/challenges.service.ts +++ b/src/gamification/challenges/challenges.service.ts @@ -4,51 +4,43 @@ import { Repository } from 'typeorm'; import { Challenge } from '../entities/challenge.entity'; import { UserChallenge } from '../entities/user-challenge.entity'; import { User } from '../../users/entities/user.entity'; - @Injectable() export class ChallengesService { - constructor( + constructor( @InjectRepository(Challenge) - private challengeRepository: Repository, + private challengeRepository: Repository, @InjectRepository(UserChallenge) - private userChallengeRepository: Repository, - ) {} - - async updateProgress(userId: string, challengeId: string, increment: number): Promise { - let userChallenge = await this.userChallengeRepository.findOne({ - where: { user: { id: userId }, challenge: { id: challengeId } }, - relations: ['challenge'], - }); - - if (!userChallenge) { - const challenge = await this.challengeRepository.findOne({ where: { id: challengeId } }); - if (!challenge) throw new Error('Challenge not found'); - - userChallenge = this.userChallengeRepository.create({ - user: { id: userId } as User, - challenge, - progressValue: 0, - isCompleted: false, - }); + private userChallengeRepository: Repository) { } + async updateProgress(userId: string, challengeId: string, increment: number): Promise { + let userChallenge = await this.userChallengeRepository.findOne({ + where: { user: { id: userId }, challenge: { id: challengeId } }, + relations: ['challenge'], + }); + if (!userChallenge) { + const challenge = await this.challengeRepository.findOne({ where: { id: challengeId } }); + if (!challenge) + throw new Error('Challenge not found'); + userChallenge = this.userChallengeRepository.create({ + user: { id: userId } as User, + challenge, + progressValue: 0, + isCompleted: false, + }); + } + if (userChallenge.isCompleted) + return userChallenge; + userChallenge.progressValue += increment; + if (userChallenge.progressValue >= userChallenge.challenge.goalValue) { + userChallenge.isCompleted = true; + userChallenge.completedAt = new Date(); + // TODO: Reward user points via PointsService + } + return await this.userChallengeRepository.save(userChallenge); } - - if (userChallenge.isCompleted) return userChallenge; - - userChallenge.progressValue += increment; - - if (userChallenge.progressValue >= userChallenge.challenge.goalValue) { - userChallenge.isCompleted = true; - userChallenge.completedAt = new Date(); - // TODO: Reward user points via PointsService + async getUserChallenges(userId: string): Promise { + return await this.userChallengeRepository.find({ + where: { user: { id: userId } }, + relations: ['challenge'], + }); } - - return await this.userChallengeRepository.save(userChallenge); - } - - async getUserChallenges(userId: string): Promise { - return await this.userChallengeRepository.find({ - where: { user: { id: userId } }, - relations: ['challenge'], - }); - } } diff --git a/src/gamification/entities/badge.entity.ts b/src/gamification/entities/badge.entity.ts index 5cf7ef5b..0856a782 100644 --- a/src/gamification/entities/badge.entity.ts +++ b/src/gamification/entities/badge.entity.ts @@ -1,22 +1,16 @@ import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm'; - @Entity('badges') export class Badge { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - name: string; - - @Column() - description: string; - - @Column() - iconUrl: string; - - @Column() - criteriaType: string; // e.g., 'POINTS_REACHED', 'CHALLENGE_COMPLETED' - - @Column('jsonb', { nullable: true }) - criteriaValue: any; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + name: string; + @Column() + description: string; + @Column() + iconUrl: string; + @Column() + criteriaType: string; // e.g., 'POINTS_REACHED', 'CHALLENGE_COMPLETED' + @Column('jsonb', { nullable: true }) + criteriaValue: unknown; } diff --git a/src/gamification/entities/challenge.entity.ts b/src/gamification/entities/challenge.entity.ts index de0e4b59..8c3d9a86 100644 --- a/src/gamification/entities/challenge.entity.ts +++ b/src/gamification/entities/challenge.entity.ts @@ -1,22 +1,16 @@ import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm'; - @Entity('challenges') export class Challenge { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - title: string; - - @Column() - description: string; - - @Column() - rewardPoints: number; - - @Column() - goalValue: number; - - @Column() - type: string; // e.g., 'READ_ARTICLES', 'WATCH_VIDEOS' + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + title: string; + @Column() + description: string; + @Column() + rewardPoints: number; + @Column() + goalValue: number; + @Column() + type: string; // e.g., 'READ_ARTICLES', 'WATCH_VIDEOS' } diff --git a/src/gamification/entities/point-transaction.entity.ts b/src/gamification/entities/point-transaction.entity.ts index e04bd8ed..a96aa793 100644 --- a/src/gamification/entities/point-transaction.entity.ts +++ b/src/gamification/entities/point-transaction.entity.ts @@ -1,20 +1,15 @@ import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, CreateDateColumn } from 'typeorm'; import { User } from '../../users/entities/user.entity'; - @Entity('point_transactions') export class PointTransaction { - @PrimaryGeneratedColumn('uuid') - id: string; - - @ManyToOne(() => User) - user: User; - - @Column() - points: number; - - @Column() - activityType: string; // e.g., 'COURSE_COMPLETED', 'DAILY_LOGIN' - - @CreateDateColumn() - createdAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @ManyToOne(() => User) + user: User; + @Column() + points: number; + @Column() + activityType: string; // e.g., 'COURSE_COMPLETED', 'DAILY_LOGIN' + @CreateDateColumn() + createdAt: Date; } diff --git a/src/gamification/entities/user-badge.entity.ts b/src/gamification/entities/user-badge.entity.ts index fcbc89a9..45c3e3f7 100644 --- a/src/gamification/entities/user-badge.entity.ts +++ b/src/gamification/entities/user-badge.entity.ts @@ -1,18 +1,14 @@ import { Entity, PrimaryGeneratedColumn, ManyToOne, CreateDateColumn } from 'typeorm'; import { User } from '../../users/entities/user.entity'; import { Badge } from './badge.entity'; - @Entity('user_badges') export class UserBadge { - @PrimaryGeneratedColumn('uuid') - id: string; - - @ManyToOne(() => User) - user: User; - - @ManyToOne(() => Badge) - badge: Badge; - - @CreateDateColumn() - earnedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @ManyToOne(() => User) + user: User; + @ManyToOne(() => Badge) + badge: Badge; + @CreateDateColumn() + earnedAt: Date; } diff --git a/src/gamification/entities/user-challenge.entity.ts b/src/gamification/entities/user-challenge.entity.ts index 157ba6a0..68946964 100644 --- a/src/gamification/entities/user-challenge.entity.ts +++ b/src/gamification/entities/user-challenge.entity.ts @@ -1,24 +1,18 @@ import { Entity, PrimaryGeneratedColumn, ManyToOne, Column } from 'typeorm'; import { User } from '../../users/entities/user.entity'; import { Challenge } from './challenge.entity'; - @Entity('user_challenges') export class UserChallenge { - @PrimaryGeneratedColumn('uuid') - id: string; - - @ManyToOne(() => User) - user: User; - - @ManyToOne(() => Challenge) - challenge: Challenge; - - @Column({ default: 0 }) - progressValue: number; - - @Column({ default: false }) - isCompleted: boolean; - - @Column({ nullable: true }) - completedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @ManyToOne(() => User) + user: User; + @ManyToOne(() => Challenge) + challenge: Challenge; + @Column({ default: 0 }) + progressValue: number; + @Column({ default: false }) + isCompleted: boolean; + @Column({ nullable: true }) + completedAt: Date; } diff --git a/src/gamification/entities/user-progress.entity.ts b/src/gamification/entities/user-progress.entity.ts index fbe04ccd..4333691d 100644 --- a/src/gamification/entities/user-progress.entity.ts +++ b/src/gamification/entities/user-progress.entity.ts @@ -1,25 +1,20 @@ import { Entity, PrimaryGeneratedColumn, Column, OneToOne, JoinColumn, Index } from 'typeorm'; import { User } from '../../users/entities/user.entity'; - @Entity('user_progress') export class UserProgress { - @PrimaryGeneratedColumn('uuid') - id: string; - - @OneToOne(() => User) - @JoinColumn() - @Index() - user: User; - - @Column({ default: 0 }) - @Index() - totalPoints: number; - - @Column({ default: 1 }) - @Index() - level: number; - - @Column({ default: 0 }) - @Index() - xp: number; + @PrimaryGeneratedColumn('uuid') + id: string; + @OneToOne(() => User) + @JoinColumn() + @Index() + user: User; + @Column({ default: 0 }) + @Index() + totalPoints: number; + @Column({ default: 1 }) + @Index() + level: number; + @Column({ default: 0 }) + @Index() + xp: number; } diff --git a/src/gamification/gamification.controller.ts b/src/gamification/gamification.controller.ts index b8f8305a..93b8ae0b 100644 --- a/src/gamification/gamification.controller.ts +++ b/src/gamification/gamification.controller.ts @@ -4,43 +4,41 @@ import { PointsService } from './points/points.service'; import { BadgesService } from './badges/badges.service'; import { LeaderboardService } from './leaderboards/leaderboards.service'; import { ChallengesService } from './challenges/challenges.service'; - @Controller('gamification') export class GamificationController { - constructor( - private readonly gamificationService: GamificationService, - private readonly pointsService: PointsService, - private readonly badgesService: BadgesService, - private readonly leaderboardService: LeaderboardService, - private readonly challengesService: ChallengesService, - ) {} - - @Get('progress/:userId') - async getProgress(@Param('userId') userId: string) { - return this.pointsService.getUserProgress(userId); - } - - @Get('badges/:userId') - async getBadges(@Param('userId') userId: string) { - return this.badgesService.getUserBadges(userId); - } - - @Get('challenges/:userId') - async getChallenges(@Param('userId') userId: string) { - return this.challengesService.getUserChallenges(userId); - } - - @Get('leaderboard') - async getLeaderboard(@Query('limit') limit?: number) { - return this.leaderboardService.getTopPlayers(limit); - } - - @Post('activity/:userId') - async recordActivity( - @Param('userId') userId: string, - @Body('type') type: string, - @Body('points') points?: number, - ) { - return this.gamificationService.handleActivity(userId, type, points); - } + constructor(private readonly gamificationService: GamificationService, private readonly pointsService: PointsService, private readonly badgesService: BadgesService, private readonly leaderboardService: LeaderboardService, private readonly challengesService: ChallengesService) { } + @Get('progress/:userId') + async getProgress( + @Param('userId') + userId: string) { + return this.pointsService.getUserProgress(userId); + } + @Get('badges/:userId') + async getBadges( + @Param('userId') + userId: string) { + return this.badgesService.getUserBadges(userId); + } + @Get('challenges/:userId') + async getChallenges( + @Param('userId') + userId: string) { + return this.challengesService.getUserChallenges(userId); + } + @Get('leaderboard') + async getLeaderboard( + @Query('limit') + limit?: number) { + return this.leaderboardService.getTopPlayers(limit); + } + @Post('activity/:userId') + async recordActivity( + @Param('userId') + userId: string, + @Body('type') + type: string, + @Body('points') + points?: number) { + return this.gamificationService.handleActivity(userId, type, points); + } } diff --git a/src/gamification/gamification.module.ts b/src/gamification/gamification.module.ts index eed57136..080779d1 100644 --- a/src/gamification/gamification.module.ts +++ b/src/gamification/gamification.module.ts @@ -12,26 +12,26 @@ import { LeaderboardService } from './leaderboards/leaderboards.service'; import { ChallengesService } from './challenges/challenges.service'; import { GamificationService } from './gamification.service'; import { GamificationController } from './gamification.controller'; - @Module({ - imports: [ - TypeOrmModule.forFeature([ - PointTransaction, - UserProgress, - Badge, - UserBadge, - Challenge, - UserChallenge, - ]), - ], - providers: [ - PointsService, - BadgesService, - LeaderboardService, - ChallengesService, - GamificationService, - ], - controllers: [GamificationController], - exports: [GamificationService, PointsService, BadgesService, ChallengesService], + imports: [ + TypeOrmModule.forFeature([ + PointTransaction, + UserProgress, + Badge, + UserBadge, + Challenge, + UserChallenge, + ]), + ], + providers: [ + PointsService, + BadgesService, + LeaderboardService, + ChallengesService, + GamificationService, + ], + controllers: [GamificationController], + exports: [GamificationService, PointsService, BadgesService, ChallengesService], }) -export class GamificationModule {} +export class GamificationModule { +} diff --git a/src/gamification/gamification.service.ts b/src/gamification/gamification.service.ts index a712bf38..65b7bef4 100644 --- a/src/gamification/gamification.service.ts +++ b/src/gamification/gamification.service.ts @@ -2,28 +2,19 @@ import { Injectable } from '@nestjs/common'; import { PointsService } from './points/points.service'; import { BadgesService } from './badges/badges.service'; import { ChallengesService } from './challenges/challenges.service'; - @Injectable() export class GamificationService { - constructor( - private pointsService: PointsService, - private badgesService: BadgesService, - private challengesService: ChallengesService, - ) {} - - async handleActivity(userId: string, activityType: string, points: number = 10): Promise { - // 1. Add points - const progress = await this.pointsService.addPoints(userId, points, activityType); - - // 2. Check for badge awarding (simple example: award badge if points > 500) - if (progress.totalPoints >= 500) { - // Assuming a badge with ID 'early-achiever' exists - // await this.badgesService.awardBadge(userId, 'early-achiever-id'); + constructor(private pointsService: PointsService, private badgesService: BadgesService, private challengesService: ChallengesService) { } + async handleActivity(userId: string, activityType: string, points: number = 10): Promise { + // 1. Add points + const progress = await this.pointsService.addPoints(userId, points, activityType); + // 2. Check for badge awarding (simple example: award badge if points > 500) + if (progress.totalPoints >= 500) { + // Assuming a badge with ID 'early-achiever' exists + // await this.badgesService.awardBadge(userId, 'early-achiever-id'); + } + // 3. Update active challenges based on activity + // This would normally involve complex logic to match activityType to challenge goals + return progress; } - - // 3. Update active challenges based on activity - // This would normally involve complex logic to match activityType to challenge goals - - return progress; - } } diff --git a/src/gamification/leaderboards/leaderboards.service.ts b/src/gamification/leaderboards/leaderboards.service.ts index 26d3c755..d1764aae 100644 --- a/src/gamification/leaderboards/leaderboards.service.ts +++ b/src/gamification/leaderboards/leaderboards.service.ts @@ -2,28 +2,23 @@ import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { UserProgress } from '../entities/user-progress.entity'; - @Injectable() export class LeaderboardService { - constructor( + constructor( @InjectRepository(UserProgress) - private userProgressRepository: Repository, - ) {} - - async getTopPlayers(limit: number = 10): Promise { - return await this.userProgressRepository.find({ - order: { totalPoints: 'DESC' }, - take: limit, - relations: ['user'], - }); - } - - async getUserRank(userId: string): Promise { - const allProgress = await this.userProgressRepository.find({ - order: { totalPoints: 'DESC' }, - }); - - const rank = allProgress.findIndex((p) => p.user?.id === userId) + 1; - return rank > 0 ? rank : null; - } + private userProgressRepository: Repository) { } + async getTopPlayers(limit: number = 10): Promise { + return await this.userProgressRepository.find({ + order: { totalPoints: 'DESC' }, + take: limit, + relations: ['user'], + }); + } + async getUserRank(userId: string): Promise { + const allProgress = await this.userProgressRepository.find({ + order: { totalPoints: 'DESC' }, + }); + const rank = allProgress.findIndex((p) => p.user?.id === userId) + 1; + return rank > 0 ? rank : null; + } } diff --git a/src/gamification/points/points.service.ts b/src/gamification/points/points.service.ts index c3816e22..17327bf9 100644 --- a/src/gamification/points/points.service.ts +++ b/src/gamification/points/points.service.ts @@ -4,55 +4,46 @@ import { Repository } from 'typeorm'; import { UserProgress } from '../entities/user-progress.entity'; import { PointTransaction } from '../entities/point-transaction.entity'; import { User } from '../../users/entities/user.entity'; - @Injectable() export class PointsService { - constructor( + constructor( @InjectRepository(UserProgress) - private userProgressRepository: Repository, + private userProgressRepository: Repository, @InjectRepository(PointTransaction) - private pointTransactionRepository: Repository, - ) {} - - async addPoints(userId: string, points: number, activityType: string): Promise { - // Log the transaction - const transaction = this.pointTransactionRepository.create({ - user: { id: userId } as User, - points, - activityType, - }); - await this.pointTransactionRepository.save(transaction); - - // Update user progress - let progress = await this.userProgressRepository.findOne({ - where: { user: { id: userId } }, - }); - - if (!progress) { - progress = this.userProgressRepository.create({ - user: { id: userId } as User, - totalPoints: 0, - level: 1, - xp: 0, - }); + private pointTransactionRepository: Repository) { } + async addPoints(userId: string, points: number, activityType: string): Promise { + // Log the transaction + const transaction = this.pointTransactionRepository.create({ + user: { id: userId } as User, + points, + activityType, + }); + await this.pointTransactionRepository.save(transaction); + // Update user progress + let progress = await this.userProgressRepository.findOne({ + where: { user: { id: userId } }, + }); + if (!progress) { + progress = this.userProgressRepository.create({ + user: { id: userId } as User, + totalPoints: 0, + level: 1, + xp: 0, + }); + } + progress.totalPoints += points; + progress.xp += points; + // Basic level progression logic: level up every 1000 XP + const newLevel = Math.floor(progress.xp / 1000) + 1; + if (newLevel > progress.level) { + progress.level = newLevel; + // TODO: Emit level up event + } + return await this.userProgressRepository.save(progress); } - - progress.totalPoints += points; - progress.xp += points; - - // Basic level progression logic: level up every 1000 XP - const newLevel = Math.floor(progress.xp / 1000) + 1; - if (newLevel > progress.level) { - progress.level = newLevel; - // TODO: Emit level up event + async getUserProgress(userId: string): Promise { + return await this.userProgressRepository.findOne({ + where: { user: { id: userId } }, + }); } - - return await this.userProgressRepository.save(progress); - } - - async getUserProgress(userId: string): Promise { - return await this.userProgressRepository.findOne({ - where: { user: { id: userId } }, - }); - } } diff --git a/src/gateways/messaging.gateway.ts b/src/gateways/messaging.gateway.ts index 31f04e3f..28e33e22 100644 --- a/src/gateways/messaging.gateway.ts +++ b/src/gateways/messaging.gateway.ts @@ -1,11 +1,4 @@ -import { - WebSocketGateway, - SubscribeMessage, - MessageBody, - ConnectedSocket, - OnGatewayConnection, - OnGatewayDisconnect, -} from '@nestjs/websockets'; +import { WebSocketGateway, SubscribeMessage, MessageBody, ConnectedSocket, OnGatewayConnection, OnGatewayDisconnect, } from '@nestjs/websockets'; import { UseGuards, Logger } from '@nestjs/common'; import { Socket } from 'socket.io'; import { WsJwtAuthGuard } from '../auth/guards/ws-jwt-auth.guard'; @@ -13,45 +6,42 @@ import { WsThrottlerGuard } from '../common/guards/ws-throttler.guard'; import { wsManager } from '../common/utils/websocket.utils'; import { JwtService } from '@nestjs/jwt'; import { MESSAGING_GATEWAY_EVENTS } from '../collaboration/constants/collaboration-events.constants'; - @WebSocketGateway({ namespace: '/messaging' }) @UseGuards(WsThrottlerGuard) export class MessagingGateway implements OnGatewayConnection, OnGatewayDisconnect { - private readonly logger = new Logger(MessagingGateway.name); - - constructor(private readonly jwtService: JwtService) {} - - async handleConnection(client: Socket) { - try { - const token = - client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; - if (!token) { - throw new Error('No token provided'); - } - - const payload = await this.jwtService.verifyAsync(token); - const userId = payload.sub || payload.id; - - const registered = wsManager.registerConnection(userId, client); - if (!registered) { - return; - } - this.logger.log(`Client connected: ${client.id}`); - } catch (_error) { - client.emit('error', { message: 'Unauthorized' }); - client.disconnect(true); + private readonly logger = new Logger(MessagingGateway.name); + constructor(private readonly jwtService: JwtService) { } + async handleConnection(client: Socket) { + try { + const token = client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; + if (!token) { + throw new Error('No token provided'); + } + const payload = await this.jwtService.verifyAsync(token); + const userId = payload.sub || payload.id; + const registered = wsManager.registerConnection(userId, client); + if (!registered) { + return; + } + this.logger.log(`Client connected: ${client.id}`); + } + catch (_error) { + client.emit('error', { message: 'Unauthorized' }); + client.disconnect(true); + } + } + handleDisconnect(client: Socket) { + wsManager.cleanupSocket(client); + this.logger.log(`Client disconnected: ${client.id}`); + } + @UseGuards(WsJwtAuthGuard) + @SubscribeMessage(MESSAGING_GATEWAY_EVENTS.SEND_MESSAGE) + async handleMessage( + @MessageBody() + data: unknown, + @ConnectedSocket() + client: Socket) { + const user = (client as unknown).user; + return { userId: user.sub, message: data }; } - } - - handleDisconnect(client: Socket) { - wsManager.cleanupSocket(client); - this.logger.log(`Client disconnected: ${client.id}`); - } - - @UseGuards(WsJwtAuthGuard) - @SubscribeMessage(MESSAGING_GATEWAY_EVENTS.SEND_MESSAGE) - async handleMessage(@MessageBody() data: any, @ConnectedSocket() client: Socket) { - const user = (client as any).user; - return { userId: user.sub, message: data }; - } } diff --git a/src/gateways/notifications.gateway.ts b/src/gateways/notifications.gateway.ts index ffb912f3..def7f37f 100644 --- a/src/gateways/notifications.gateway.ts +++ b/src/gateways/notifications.gateway.ts @@ -1,10 +1,4 @@ -import { - WebSocketGateway, - SubscribeMessage, - ConnectedSocket, - OnGatewayConnection, - OnGatewayDisconnect, -} from '@nestjs/websockets'; +import { WebSocketGateway, SubscribeMessage, ConnectedSocket, OnGatewayConnection, OnGatewayDisconnect, } from '@nestjs/websockets'; import { UseGuards, Logger } from '@nestjs/common'; import { Socket } from 'socket.io'; import { WsJwtAuthGuard } from '../auth/guards/ws-jwt-auth.guard'; @@ -12,46 +6,41 @@ import { WsThrottlerGuard } from '../common/guards/ws-throttler.guard'; import { wsManager } from '../common/utils/websocket.utils'; import { JwtService } from '@nestjs/jwt'; import { NOTIFICATION_GATEWAY_EVENTS } from '../collaboration/constants/collaboration-events.constants'; - @WebSocketGateway({ namespace: '/notifications' }) @UseGuards(WsThrottlerGuard) export class NotificationsGateway implements OnGatewayConnection, OnGatewayDisconnect { - private readonly logger = new Logger(NotificationsGateway.name); - - constructor(private readonly jwtService: JwtService) {} - - async handleConnection(client: Socket) { - try { - const token = - client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; - if (!token) { - throw new Error('No token provided'); - } - - const payload = await this.jwtService.verifyAsync(token); - const userId = payload.sub || payload.id; - - const registered = wsManager.registerConnection(userId, client); - if (!registered) { - // Connection rejected by global limits - return; - } - this.logger.log(`Client connected: ${client.id}`); - } catch (_error) { - client.emit('error', { message: 'Unauthorized' }); - client.disconnect(true); + private readonly logger = new Logger(NotificationsGateway.name); + constructor(private readonly jwtService: JwtService) { } + async handleConnection(client: Socket) { + try { + const token = client.handshake.auth?.token || client.handshake.headers?.authorization?.split(' ')[1]; + if (!token) { + throw new Error('No token provided'); + } + const payload = await this.jwtService.verifyAsync(token); + const userId = payload.sub || payload.id; + const registered = wsManager.registerConnection(userId, client); + if (!registered) { + // Connection rejected by global limits + return; + } + this.logger.log(`Client connected: ${client.id}`); + } + catch (_error) { + client.emit('error', { message: 'Unauthorized' }); + client.disconnect(true); + } + } + handleDisconnect(client: Socket) { + wsManager.cleanupSocket(client); + this.logger.log(`Client disconnected: ${client.id}`); + } + @UseGuards(WsJwtAuthGuard) + @SubscribeMessage(NOTIFICATION_GATEWAY_EVENTS.SUBSCRIBE_NOTIFICATIONS) + async handleSubscribe( + @ConnectedSocket() + client: Socket) { + const user = (client as unknown).user; + return { userId: user.sub, subscribed: true }; } - } - - handleDisconnect(client: Socket) { - wsManager.cleanupSocket(client); - this.logger.log(`Client disconnected: ${client.id}`); - } - - @UseGuards(WsJwtAuthGuard) - @SubscribeMessage(NOTIFICATION_GATEWAY_EVENTS.SUBSCRIBE_NOTIFICATIONS) - async handleSubscribe(@ConnectedSocket() client: Socket) { - const user = (client as any).user; - return { userId: user.sub, subscribed: true }; - } } diff --git a/src/graphql/graphql.module.ts b/src/graphql/graphql.module.ts index 3f6d3579..ff768f72 100644 --- a/src/graphql/graphql.module.ts +++ b/src/graphql/graphql.module.ts @@ -15,58 +15,58 @@ import { CourseResolver } from './resolvers/course.resolver'; import { AssessmentResolver } from './resolvers/assessment.resolver'; import { DataLoaderService } from './services/dataloader.service'; import { QueryComplexityService } from './services/query-complexity.service'; - @Module({ - imports: [ - NestGraphQLModule.forRoot({ - driver: ApolloDriver, - autoSchemaFile: join(process.cwd(), 'src/graphql/schema/schema.graphql'), - sortSchema: true, - playground: true, - subscriptions: { - 'graphql-ws': true, - 'subscriptions-transport-ws': true, - }, - // Enable query complexity validation - validationRules: [], - // Custom plugins for query complexity - plugins: [], - context: ({ req, connection }, _, { injector }) => { - if (connection) { - return { req: connection.context }; - } - // Attach DataLoaders to context for N+1 prevention - const dataLoaderService = injector?.get(DataLoaderService); - const loaders = dataLoaderService?.createLoaders() || {}; - return { req, loaders }; - }, - formatError: (error) => { - return { - message: error.message, - code: error.extensions?.code, - path: error.path, - }; - }, - }), - UsersModule, - CoursesModule, - AssessmentsModule, - AuthModule, - ], - providers: [ - QueryResolver, - MutationResolver, - SubscriptionResolver, - UserResolver, - CourseResolver, - AssessmentResolver, - DataLoaderService, - QueryComplexityService, - { - provide: 'PUB_SUB', - useValue: new PubSub(), - }, - ], - exports: [DataLoaderService, QueryComplexityService, 'PUB_SUB'], + imports: [ + NestGraphQLModule.forRoot({ + driver: ApolloDriver, + autoSchemaFile: join(process.cwd(), 'src/graphql/schema/schema.graphql'), + sortSchema: true, + playground: true, + subscriptions: { + 'graphql-ws': true, + 'subscriptions-transport-ws': true, + }, + // Enable query complexity validation + validationRules: [], + // Custom plugins for query complexity + plugins: [], + context: ({ req, connection }, _, { injector }) => { + if (connection) { + return { req: connection.context }; + } + // Attach DataLoaders to context for N+1 prevention + const dataLoaderService = injector?.get(DataLoaderService); + const loaders = dataLoaderService?.createLoaders() || {}; + return { req, loaders }; + }, + formatError: (error) => { + return { + message: error.message, + code: error.extensions?.code, + path: error.path, + }; + }, + }), + UsersModule, + CoursesModule, + AssessmentsModule, + AuthModule, + ], + providers: [ + QueryResolver, + MutationResolver, + SubscriptionResolver, + UserResolver, + CourseResolver, + AssessmentResolver, + DataLoaderService, + QueryComplexityService, + { + provide: 'PUB_SUB', + useValue: new PubSub(), + }, + ], + exports: [DataLoaderService, QueryComplexityService, 'PUB_SUB'], }) -export class GraphQLModule {} +export class GraphQLModule { +} diff --git a/src/graphql/inputs/assessment.input.ts b/src/graphql/inputs/assessment.input.ts index a8d0964d..f0a8b065 100644 --- a/src/graphql/inputs/assessment.input.ts +++ b/src/graphql/inputs/assessment.input.ts @@ -1,68 +1,56 @@ import { InputType, Field, Int } from '@nestjs/graphql'; import { IsString, IsNumber, IsOptional, IsArray, Min } from 'class-validator'; - @InputType() export class QuestionInput { - @Field() - @IsString() - prompt: string; - - @Field() - @IsString() - type: string; - - @Field(() => [String], { nullable: true }) - @IsOptional() - @IsArray() - options?: string[]; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - correctAnswer?: string; - - @Field(() => Int) - @IsNumber() - @Min(1) - points: number; + @Field() + @IsString() + prompt: string; + @Field() + @IsString() + type: string; + @Field(() => [String], { nullable: true }) + @IsOptional() + @IsArray() + options?: string[]; + @Field({ nullable: true }) + @IsOptional() + @IsString() + correctAnswer?: string; + @Field(() => Int) + @IsNumber() + @Min(1) + points: number; } - @InputType() export class CreateAssessmentInput { - @Field() - @IsString() - title: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - description?: string; - - @Field(() => Int) - @IsNumber() - @Min(1) - durationMinutes: number; - - @Field(() => [QuestionInput]) - @IsArray() - questions: QuestionInput[]; + @Field() + @IsString() + title: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + description?: string; + @Field(() => Int) + @IsNumber() + @Min(1) + durationMinutes: number; + @Field(() => [QuestionInput]) + @IsArray() + questions: QuestionInput[]; } - @InputType() export class UpdateAssessmentInput { - @Field({ nullable: true }) - @IsOptional() - @IsString() - title?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - description?: string; - - @Field(() => Int, { nullable: true }) - @IsOptional() - @IsNumber() - @Min(1) - durationMinutes?: number; + @Field({ nullable: true }) + @IsOptional() + @IsString() + title?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + description?: string; + @Field(() => Int, { nullable: true }) + @IsOptional() + @IsNumber() + @Min(1) + durationMinutes?: number; } diff --git a/src/graphql/inputs/course.input.ts b/src/graphql/inputs/course.input.ts index bc49b37c..072161cc 100644 --- a/src/graphql/inputs/course.input.ts +++ b/src/graphql/inputs/course.input.ts @@ -1,74 +1,61 @@ import { InputType, Field, Float } from '@nestjs/graphql'; import { IsString, IsNumber, IsOptional, Min } from 'class-validator'; - @InputType() export class CreateCourseInput { - @Field() - @IsString() - title: string; - - @Field() - @IsString() - description: string; - - @Field(() => Float) - @IsNumber() - @Min(0) - price: number; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - thumbnailUrl?: string; - - @Field() - @IsString() - instructorId: string; + @Field() + @IsString() + title: string; + @Field() + @IsString() + description: string; + @Field(() => Float) + @IsNumber() + @Min(0) + price: number; + @Field({ nullable: true }) + @IsOptional() + @IsString() + thumbnailUrl?: string; + @Field() + @IsString() + instructorId: string; } - @InputType() export class UpdateCourseInput { - @Field({ nullable: true }) - @IsOptional() - @IsString() - title?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - description?: string; - - @Field(() => Float, { nullable: true }) - @IsOptional() - @IsNumber() - @Min(0) - price?: number; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - status?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - thumbnailUrl?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + title?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + description?: string; + @Field(() => Float, { nullable: true }) + @IsOptional() + @IsNumber() + @Min(0) + price?: number; + @Field({ nullable: true }) + @IsOptional() + @IsString() + status?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + thumbnailUrl?: string; } - @InputType() export class CourseFilterInput { - @Field({ nullable: true }) - @IsOptional() - @IsString() - status?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - instructorId?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - search?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + status?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + instructorId?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + search?: string; } diff --git a/src/graphql/inputs/user.input.ts b/src/graphql/inputs/user.input.ts index c6bdf1bd..555d965e 100644 --- a/src/graphql/inputs/user.input.ts +++ b/src/graphql/inputs/user.input.ts @@ -1,79 +1,65 @@ import { InputType, Field } from '@nestjs/graphql'; import { IsEmail, IsString, MinLength, IsOptional, IsEnum } from 'class-validator'; import { UserRole, UserStatus } from '../../users/entities/user.entity'; - @InputType() export class CreateUserInput { - @Field() - @IsEmail() - email: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - username?: string; - - @Field() - @IsString() - @MinLength(8) - password: string; - - @Field() - @IsString() - firstName: string; - - @Field() - @IsString() - lastName: string; - - @Field(() => UserRole, { nullable: true }) - @IsOptional() - @IsEnum(UserRole) - role?: UserRole; + @Field() + @IsEmail() + email: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + username?: string; + @Field() + @IsString() + @MinLength(8) + password: string; + @Field() + @IsString() + firstName: string; + @Field() + @IsString() + lastName: string; + @Field(() => UserRole, { nullable: true }) + @IsOptional() + @IsEnum(UserRole) + role?: UserRole; } - @InputType() export class UpdateUserInput { - @Field({ nullable: true }) - @IsOptional() - @IsString() - username?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - firstName?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - lastName?: string; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - profilePicture?: string; - - @Field(() => UserStatus, { nullable: true }) - @IsOptional() - @IsEnum(UserStatus) - status?: UserStatus; + @Field({ nullable: true }) + @IsOptional() + @IsString() + username?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + firstName?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + lastName?: string; + @Field({ nullable: true }) + @IsOptional() + @IsString() + profilePicture?: string; + @Field(() => UserStatus, { nullable: true }) + @IsOptional() + @IsEnum(UserStatus) + status?: UserStatus; } - @InputType() export class UserFilterInput { - @Field({ nullable: true }) - @IsOptional() - @IsEnum(UserRole) - role?: UserRole; - - @Field({ nullable: true }) - @IsOptional() - @IsEnum(UserStatus) - status?: UserStatus; - - @Field({ nullable: true }) - @IsOptional() - @IsString() - search?: string; + @Field({ nullable: true }) + @IsOptional() + @IsEnum(UserRole) + role?: UserRole; + @Field({ nullable: true }) + @IsOptional() + @IsEnum(UserStatus) + status?: UserStatus; + @Field({ nullable: true }) + @IsOptional() + @IsString() + search?: string; } diff --git a/src/graphql/middleware/dataloader.middleware.ts b/src/graphql/middleware/dataloader.middleware.ts index 58cc027e..1cd341eb 100644 --- a/src/graphql/middleware/dataloader.middleware.ts +++ b/src/graphql/middleware/dataloader.middleware.ts @@ -1,18 +1,16 @@ import { Injectable, NestMiddleware } from '@nestjs/common'; import { Request, Response, NextFunction } from 'express'; import { DataLoaderService } from '../services/dataloader.service'; - /** * Middleware to inject DataLoaders into request context * Creates fresh loaders for each request to ensure proper caching scope */ @Injectable() export class DataLoaderMiddleware implements NestMiddleware { - constructor(private readonly dataLoaderService: DataLoaderService) {} - - use(req: Request, res: Response, next: NextFunction) { - // Attach loaders to request object for GraphQL context - (req as any).loaders = this.dataLoaderService.createLoaders(); - next(); - } + constructor(private readonly dataLoaderService: DataLoaderService) { } + use(req: Request, res: Response, next: NextFunction) { + // Attach loaders to request object for GraphQL context + (req as unknown).loaders = this.dataLoaderService.createLoaders(); + next(); + } } diff --git a/src/graphql/resolvers/assessment.resolver.ts b/src/graphql/resolvers/assessment.resolver.ts index 58877133..2c91bd38 100644 --- a/src/graphql/resolvers/assessment.resolver.ts +++ b/src/graphql/resolvers/assessment.resolver.ts @@ -1,16 +1,17 @@ import { Resolver, ResolveField, Parent } from '@nestjs/graphql'; import { AssessmentType, QuestionType } from '../types/assessment.type'; - /** * Field Resolver for Assessment type * Handles nested field resolution for assessment questions */ @Resolver(() => AssessmentType) export class AssessmentResolver { - @ResolveField(() => [QuestionType]) - async questions(@Parent() assessment: AssessmentType): Promise { - // Questions are typically loaded with the assessment - // This resolver ensures proper GraphQL type resolution - return assessment.questions || []; - } + @ResolveField(() => [QuestionType]) + async questions( + @Parent() + assessment: AssessmentType): Promise { + // Questions are typically loaded with the assessment + // This resolver ensures proper GraphQL type resolution + return assessment.questions || []; + } } diff --git a/src/graphql/resolvers/course.resolver.ts b/src/graphql/resolvers/course.resolver.ts index f1ab6a65..22f25c17 100644 --- a/src/graphql/resolvers/course.resolver.ts +++ b/src/graphql/resolvers/course.resolver.ts @@ -2,39 +2,32 @@ import { Resolver, ResolveField, Parent, Context } from '@nestjs/graphql'; import { CourseType } from '../types/course.type'; import { UserType } from '../types/user.type'; import { UsersService } from '../../users/users.service'; - /** * Field Resolver for Course type * Handles nested field resolution with DataLoader optimization */ @Resolver(() => CourseType) export class CourseResolver { - constructor(private readonly usersService: UsersService) {} - - @ResolveField(() => UserType, { nullable: true }) - async instructor( - @Parent() course: CourseType, - @Context() context: any, - ): Promise { - if (!course.instructor) { - return null; + constructor(private readonly usersService: UsersService) { } + @ResolveField(() => UserType, { nullable: true }) + async instructor( + @Parent() + course: CourseType, + @Context() + context: unknown): Promise { + if (!course.instructor) { + return null; + } + const { userLoader } = context.loaders || {}; + // If instructor is already loaded with full data, return it + if (typeof course.instructor === 'object' && course.instructor.id) { + return course.instructor; + } + // Otherwise, use DataLoader to fetch instructor + const instructorId = typeof course.instructor === 'string' ? course.instructor : course.instructor.id; + if (userLoader) { + return userLoader.load(instructorId); + } + return this.usersService.findOne(instructorId); } - - const { userLoader } = context.loaders || {}; - - // If instructor is already loaded with full data, return it - if (typeof course.instructor === 'object' && course.instructor.id) { - return course.instructor; - } - - // Otherwise, use DataLoader to fetch instructor - const instructorId = - typeof course.instructor === 'string' ? course.instructor : course.instructor.id; - - if (userLoader) { - return userLoader.load(instructorId); - } - - return this.usersService.findOne(instructorId); - } } diff --git a/src/graphql/resolvers/mutation.resolver.ts b/src/graphql/resolvers/mutation.resolver.ts index c99fa1be..a26fd915 100644 --- a/src/graphql/resolvers/mutation.resolver.ts +++ b/src/graphql/resolvers/mutation.resolver.ts @@ -11,106 +11,108 @@ import { CreateUserInput, UpdateUserInput } from '../inputs/user.input'; import { CreateCourseInput, UpdateCourseInput } from '../inputs/course.input'; import { CreateAssessmentInput, UpdateAssessmentInput } from '../inputs/assessment.input'; import { JwtAuthGuard } from '../../auth/guards/jwt-auth.guard'; - /** * Main Mutation Resolver for GraphQL API * Handles all write operations with real-time notifications */ @Resolver() export class MutationResolver { - constructor( - private readonly usersService: UsersService, - private readonly coursesService: CoursesService, - private readonly assessmentsService: AssessmentsService, - @Inject('PUB_SUB') private readonly pubSub: PubSub, - ) {} - - // User Mutations - @Mutation(() => UserType) - async createUser(@Args('input') input: CreateUserInput): Promise { - const user = await this.usersService.create(input); - await this.pubSub.publish('userCreated', { userCreated: user }); - return user; - } - - @Mutation(() => UserType) - @UseGuards(JwtAuthGuard) - async updateUser( - @Args('id', { type: () => ID }) id: string, - @Args('input') input: UpdateUserInput, - ): Promise { - const user = await this.usersService.update(id, input); - await this.pubSub.publish('userUpdated', { userUpdated: user }); - return user; - } - - @Mutation(() => Boolean) - @UseGuards(JwtAuthGuard) - async deleteUser(@Args('id', { type: () => ID }) id: string): Promise { - await this.usersService.remove(id); - await this.pubSub.publish('userDeleted', { userDeleted: { id } }); - return true; - } - - // Course Mutations - @Mutation(() => CourseType) - @UseGuards(JwtAuthGuard) - async createCourse(@Args('input') input: CreateCourseInput): Promise { - const course = await this.coursesService.create(input); - await this.pubSub.publish('courseCreated', { courseCreated: course }); - return course; - } - - @Mutation(() => CourseType) - @UseGuards(JwtAuthGuard) - async updateCourse( - @Args('id', { type: () => ID }) id: string, - @Args('input') input: UpdateCourseInput, - ): Promise { - const course = await this.coursesService.update(id, input); - await this.pubSub.publish('courseUpdated', { courseUpdated: course }); - return course; - } - - @Mutation(() => Boolean) - @UseGuards(JwtAuthGuard) - async deleteCourse(@Args('id', { type: () => ID }) id: string): Promise { - await this.coursesService.remove(id); - await this.pubSub.publish('courseDeleted', { courseDeleted: { id } }); - return true; - } - - // Assessment Mutations - @Mutation(() => AssessmentType) - @UseGuards(JwtAuthGuard) - async createAssessment(@Args('input') input: CreateAssessmentInput): Promise { - const assessment = await this.assessmentsService.create(input); - await this.pubSub.publish('assessmentCreated', { - assessmentCreated: assessment, - }); - return assessment; - } - - @Mutation(() => AssessmentType) - @UseGuards(JwtAuthGuard) - async updateAssessment( - @Args('id', { type: () => ID }) id: string, - @Args('input') input: UpdateAssessmentInput, - ): Promise { - const assessment = await this.assessmentsService.update(id, input); - await this.pubSub.publish('assessmentUpdated', { - assessmentUpdated: assessment, - }); - return assessment; - } - - @Mutation(() => Boolean) - @UseGuards(JwtAuthGuard) - async deleteAssessment(@Args('id', { type: () => ID }) id: string): Promise { - await this.assessmentsService.remove(id); - await this.pubSub.publish('assessmentDeleted', { - assessmentDeleted: { id }, - }); - return true; - } + constructor(private readonly usersService: UsersService, private readonly coursesService: CoursesService, private readonly assessmentsService: AssessmentsService, + @Inject('PUB_SUB') + private readonly pubSub: PubSub) { } + // User Mutations + @Mutation(() => UserType) + async createUser( + @Args('input') + input: CreateUserInput): Promise { + const user = await this.usersService.create(input); + await this.pubSub.publish('userCreated', { userCreated: user }); + return user; + } + @Mutation(() => UserType) + @UseGuards(JwtAuthGuard) + async updateUser( + @Args('id', { type: () => ID }) + id: string, + @Args('input') + input: UpdateUserInput): Promise { + const user = await this.usersService.update(id, input); + await this.pubSub.publish('userUpdated', { userUpdated: user }); + return user; + } + @Mutation(() => Boolean) + @UseGuards(JwtAuthGuard) + async deleteUser( + @Args('id', { type: () => ID }) + id: string): Promise { + await this.usersService.remove(id); + await this.pubSub.publish('userDeleted', { userDeleted: { id } }); + return true; + } + // Course Mutations + @Mutation(() => CourseType) + @UseGuards(JwtAuthGuard) + async createCourse( + @Args('input') + input: CreateCourseInput): Promise { + const course = await this.coursesService.create(input); + await this.pubSub.publish('courseCreated', { courseCreated: course }); + return course; + } + @Mutation(() => CourseType) + @UseGuards(JwtAuthGuard) + async updateCourse( + @Args('id', { type: () => ID }) + id: string, + @Args('input') + input: UpdateCourseInput): Promise { + const course = await this.coursesService.update(id, input); + await this.pubSub.publish('courseUpdated', { courseUpdated: course }); + return course; + } + @Mutation(() => Boolean) + @UseGuards(JwtAuthGuard) + async deleteCourse( + @Args('id', { type: () => ID }) + id: string): Promise { + await this.coursesService.remove(id); + await this.pubSub.publish('courseDeleted', { courseDeleted: { id } }); + return true; + } + // Assessment Mutations + @Mutation(() => AssessmentType) + @UseGuards(JwtAuthGuard) + async createAssessment( + @Args('input') + input: CreateAssessmentInput): Promise { + const assessment = await this.assessmentsService.create(input); + await this.pubSub.publish('assessmentCreated', { + assessmentCreated: assessment, + }); + return assessment; + } + @Mutation(() => AssessmentType) + @UseGuards(JwtAuthGuard) + async updateAssessment( + @Args('id', { type: () => ID }) + id: string, + @Args('input') + input: UpdateAssessmentInput): Promise { + const assessment = await this.assessmentsService.update(id, input); + await this.pubSub.publish('assessmentUpdated', { + assessmentUpdated: assessment, + }); + return assessment; + } + @Mutation(() => Boolean) + @UseGuards(JwtAuthGuard) + async deleteAssessment( + @Args('id', { type: () => ID }) + id: string): Promise { + await this.assessmentsService.remove(id); + await this.pubSub.publish('assessmentDeleted', { + assessmentDeleted: { id }, + }); + return true; + } } diff --git a/src/graphql/resolvers/query.resolver.ts b/src/graphql/resolvers/query.resolver.ts index 1ae99137..54fd92c4 100644 --- a/src/graphql/resolvers/query.resolver.ts +++ b/src/graphql/resolvers/query.resolver.ts @@ -10,100 +10,92 @@ import { AssessmentType } from '../types/assessment.type'; import { UserFilterInput } from '../inputs/user.input'; import { CourseFilterInput } from '../inputs/course.input'; import { JwtAuthGuard } from '../../auth/guards/jwt-auth.guard'; - /** * Main Query Resolver for GraphQL API * Handles all read operations with optimized data fetching */ @Resolver() export class QueryResolver { - constructor( - private readonly usersService: UsersService, - private readonly coursesService: CoursesService, - private readonly assessmentsService: AssessmentsService, - ) {} - - // User Queries - @Query(() => UserType, { nullable: true }) - @UseGuards(JwtAuthGuard) - async user( - @Args('id', { type: () => ID }) id: string, - @Context() context: any, - ): Promise { - const { userLoader } = context.loaders || {}; - if (userLoader) { - return userLoader.load(id); + constructor(private readonly usersService: UsersService, private readonly coursesService: CoursesService, private readonly assessmentsService: AssessmentsService) { } + // User Queries + @Query(() => UserType, { nullable: true }) + @UseGuards(JwtAuthGuard) + async user( + @Args('id', { type: () => ID }) + id: string, + @Context() + context: unknown): Promise { + const { userLoader } = context.loaders || {}; + if (userLoader) { + return userLoader.load(id); + } + return this.usersService.findOne(id); } - return this.usersService.findOne(id); - } - - @Query(() => [UserType]) - @UseGuards(JwtAuthGuard) - async users( + @Query(() => [UserType]) + @UseGuards(JwtAuthGuard) + async users( @Args('filter', { type: () => UserFilterInput, nullable: true }) - filter?: UserFilterInput, - ): Promise> { - return this.usersService.findAll(filter); - } - - @Query(() => UserType) - @UseGuards(JwtAuthGuard) - async me(@Context() context: any): Promise { - const userId = context.req.user?.userId; - if (!userId) { - throw new Error('User not authenticated'); + filter?: UserFilterInput): Promise> { + return this.usersService.findAll(filter); } - return this.usersService.findOne(userId); - } - - // Course Queries - @Query(() => CourseType, { nullable: true }) - async course( - @Args('id', { type: () => ID }) id: string, - @Context() context: any, - ): Promise { - const { courseLoader } = context.loaders || {}; - if (courseLoader) { - return courseLoader.load(id); + @Query(() => UserType) + @UseGuards(JwtAuthGuard) + async me( + @Context() + context: unknown): Promise { + const userId = context.req.user?.userId; + if (!userId) { + throw new Error('User not authenticated'); + } + return this.usersService.findOne(userId); } - return this.coursesService.findOne(id); - } - - @Query(() => [CourseType]) - async courses( + // Course Queries + @Query(() => CourseType, { nullable: true }) + async course( + @Args('id', { type: () => ID }) + id: string, + @Context() + context: unknown): Promise { + const { courseLoader } = context.loaders || {}; + if (courseLoader) { + return courseLoader.load(id); + } + return this.coursesService.findOne(id); + } + @Query(() => [CourseType]) + async courses( @Args('filter', { type: () => CourseFilterInput, nullable: true }) - filter?: CourseFilterInput, - ): Promise> { - return this.coursesService.findAll(filter); - } - - @Query(() => [CourseType]) - @UseGuards(JwtAuthGuard) - async myCourses(@Context() context: any): Promise { - const userId = context.req.user?.userId; - if (!userId) { - throw new Error('User not authenticated'); + filter?: CourseFilterInput): Promise> { + return this.coursesService.findAll(filter); + } + @Query(() => [CourseType]) + @UseGuards(JwtAuthGuard) + async myCourses( + @Context() + context: unknown): Promise { + const userId = context.req.user?.userId; + if (!userId) { + throw new Error('User not authenticated'); + } + return this.coursesService.findByInstructor(userId); + } + // Assessment Queries + @Query(() => AssessmentType, { nullable: true }) + @UseGuards(JwtAuthGuard) + async assessment( + @Args('id', { type: () => ID }) + id: string, + @Context() + context: unknown): Promise { + const { assessmentLoader } = context.loaders || {}; + if (assessmentLoader) { + return assessmentLoader.load(id); + } + return this.assessmentsService.findOne(id); } - return this.coursesService.findByInstructor(userId); - } - - // Assessment Queries - @Query(() => AssessmentType, { nullable: true }) - @UseGuards(JwtAuthGuard) - async assessment( - @Args('id', { type: () => ID }) id: string, - @Context() context: any, - ): Promise { - const { assessmentLoader } = context.loaders || {}; - if (assessmentLoader) { - return assessmentLoader.load(id); + @Query(() => [AssessmentType]) + @UseGuards(JwtAuthGuard) + async assessments(): Promise { + return this.assessmentsService.findAll(); } - return this.assessmentsService.findOne(id); - } - - @Query(() => [AssessmentType]) - @UseGuards(JwtAuthGuard) - async assessments(): Promise { - return this.assessmentsService.findAll(); - } } diff --git a/src/graphql/resolvers/subscription.resolver.ts b/src/graphql/resolvers/subscription.resolver.ts index d6ff2cf4..2bb7e634 100644 --- a/src/graphql/resolvers/subscription.resolver.ts +++ b/src/graphql/resolvers/subscription.resolver.ts @@ -4,87 +4,79 @@ import { PubSub } from 'graphql-subscriptions'; import { UserType } from '../types/user.type'; import { CourseType } from '../types/course.type'; import { AssessmentType } from '../types/assessment.type'; - /** * Subscription Resolver for real-time GraphQL updates * Provides live data streams for entity changes */ @Resolver() export class SubscriptionResolver { - constructor(@Inject('PUB_SUB') private readonly pubSub: PubSub) {} - - // User Subscriptions - @Subscription(() => UserType, { - name: 'userCreated', - description: 'Subscribe to new user creation events', - }) - userCreated() { - return this.pubSub.asyncIterableIterator('userCreated'); - } - - @Subscription(() => UserType, { - name: 'userUpdated', - description: 'Subscribe to user update events', - }) - userUpdated() { - return this.pubSub.asyncIterableIterator('userUpdated'); - } - - @Subscription(() => UserType, { - name: 'userDeleted', - description: 'Subscribe to user deletion events', - }) - userDeleted() { - return this.pubSub.asyncIterableIterator('userDeleted'); - } - - // Course Subscriptions - @Subscription(() => CourseType, { - name: 'courseCreated', - description: 'Subscribe to new course creation events', - }) - courseCreated() { - return this.pubSub.asyncIterableIterator('courseCreated'); - } - - @Subscription(() => CourseType, { - name: 'courseUpdated', - description: 'Subscribe to course update events', - }) - courseUpdated() { - return this.pubSub.asyncIterableIterator('courseUpdated'); - } - - @Subscription(() => CourseType, { - name: 'courseDeleted', - description: 'Subscribe to course deletion events', - }) - courseDeleted() { - return this.pubSub.asyncIterableIterator('courseDeleted'); - } - - // Assessment Subscriptions - @Subscription(() => AssessmentType, { - name: 'assessmentCreated', - description: 'Subscribe to new assessment creation events', - }) - assessmentCreated() { - return this.pubSub.asyncIterableIterator('assessmentCreated'); - } - - @Subscription(() => AssessmentType, { - name: 'assessmentUpdated', - description: 'Subscribe to assessment update events', - }) - assessmentUpdated() { - return this.pubSub.asyncIterableIterator('assessmentUpdated'); - } - - @Subscription(() => AssessmentType, { - name: 'assessmentDeleted', - description: 'Subscribe to assessment deletion events', - }) - assessmentDeleted() { - return this.pubSub.asyncIterableIterator('assessmentDeleted'); - } + constructor( + @Inject('PUB_SUB') + private readonly pubSub: PubSub) { } + // User Subscriptions + @Subscription(() => UserType, { + name: 'userCreated', + description: 'Subscribe to new user creation events', + }) + userCreated() { + return this.pubSub.asyncIterableIterator('userCreated'); + } + @Subscription(() => UserType, { + name: 'userUpdated', + description: 'Subscribe to user update events', + }) + userUpdated() { + return this.pubSub.asyncIterableIterator('userUpdated'); + } + @Subscription(() => UserType, { + name: 'userDeleted', + description: 'Subscribe to user deletion events', + }) + userDeleted() { + return this.pubSub.asyncIterableIterator('userDeleted'); + } + // Course Subscriptions + @Subscription(() => CourseType, { + name: 'courseCreated', + description: 'Subscribe to new course creation events', + }) + courseCreated() { + return this.pubSub.asyncIterableIterator('courseCreated'); + } + @Subscription(() => CourseType, { + name: 'courseUpdated', + description: 'Subscribe to course update events', + }) + courseUpdated() { + return this.pubSub.asyncIterableIterator('courseUpdated'); + } + @Subscription(() => CourseType, { + name: 'courseDeleted', + description: 'Subscribe to course deletion events', + }) + courseDeleted() { + return this.pubSub.asyncIterableIterator('courseDeleted'); + } + // Assessment Subscriptions + @Subscription(() => AssessmentType, { + name: 'assessmentCreated', + description: 'Subscribe to new assessment creation events', + }) + assessmentCreated() { + return this.pubSub.asyncIterableIterator('assessmentCreated'); + } + @Subscription(() => AssessmentType, { + name: 'assessmentUpdated', + description: 'Subscribe to assessment update events', + }) + assessmentUpdated() { + return this.pubSub.asyncIterableIterator('assessmentUpdated'); + } + @Subscription(() => AssessmentType, { + name: 'assessmentDeleted', + description: 'Subscribe to assessment deletion events', + }) + assessmentDeleted() { + return this.pubSub.asyncIterableIterator('assessmentDeleted'); + } } diff --git a/src/graphql/resolvers/user.resolver.ts b/src/graphql/resolvers/user.resolver.ts index b588a0d0..2db028a1 100644 --- a/src/graphql/resolvers/user.resolver.ts +++ b/src/graphql/resolvers/user.resolver.ts @@ -2,23 +2,23 @@ import { Resolver, ResolveField, Parent, Context } from '@nestjs/graphql'; import { UserType } from '../types/user.type'; import { CourseType } from '../types/course.type'; import { CoursesService } from '../../courses/courses.service'; - /** * Field Resolver for User type * Handles nested field resolution with DataLoader optimization */ @Resolver(() => UserType) export class UserResolver { - constructor(private readonly coursesService: CoursesService) {} - - @ResolveField(() => [CourseType]) - async courses(@Parent() user: UserType, @Context() context: any): Promise { - const { coursesByInstructorLoader } = context.loaders || {}; - - if (coursesByInstructorLoader) { - return coursesByInstructorLoader.load(user.id); + constructor(private readonly coursesService: CoursesService) { } + @ResolveField(() => [CourseType]) + async courses( + @Parent() + user: UserType, + @Context() + context: unknown): Promise { + const { coursesByInstructorLoader } = context.loaders || {}; + if (coursesByInstructorLoader) { + return coursesByInstructorLoader.load(user.id); + } + return this.coursesService.findByInstructor(user.id); } - - return this.coursesService.findByInstructor(user.id); - } } diff --git a/src/graphql/services/dataloader.service.ts b/src/graphql/services/dataloader.service.ts index 4703931d..a2dbc5fc 100644 --- a/src/graphql/services/dataloader.service.ts +++ b/src/graphql/services/dataloader.service.ts @@ -6,83 +6,71 @@ import { AssessmentsService } from '../../assessment/assessments.service'; import { User } from '../../users/entities/user.entity'; import { Course } from '../../courses/entities/course.entity'; import { Assessment } from '../../assessment/entities/assessment.entity'; - /** * DataLoader service to prevent N+1 query problems * Batches and caches database requests within a single GraphQL request */ @Injectable() export class DataLoaderService { - constructor( - private readonly usersService: UsersService, - private readonly coursesService: CoursesService, - private readonly assessmentsService: AssessmentsService, - ) {} - - /** - * Create a new DataLoader for batching user queries by ID - */ - createUserLoader(): DataLoader { - return new DataLoader(async (userIds: readonly string[]) => { - const users = await this.usersService.findByIds(Array.from(userIds)); - const userMap = new Map(users.map((user) => [user.id, user])); - return userIds.map((id) => userMap.get(id) || null); - }); - } - - /** - * Create a new DataLoader for batching course queries by ID - */ - createCourseLoader(): DataLoader { - return new DataLoader(async (courseIds: readonly string[]) => { - const courses = await this.coursesService.findByIds(Array.from(courseIds)); - const courseMap = new Map(courses.map((course) => [course.id, course])); - return courseIds.map((id) => courseMap.get(id) || null); - }); - } - - /** - * Create a new DataLoader for batching assessment queries by ID - */ - createAssessmentLoader(): DataLoader { - return new DataLoader(async (assessmentIds: readonly string[]) => { - const assessments = await this.assessmentsService.findByIds(Array.from(assessmentIds)); - const assessmentMap = new Map(assessments.map((assessment) => [assessment.id, assessment])); - return assessmentIds.map((id) => assessmentMap.get(id) || null); - }); - } - - /** - * Create a new DataLoader for batching courses by instructor ID - */ - createCoursesByInstructorLoader(): DataLoader { - return new DataLoader(async (instructorIds: readonly string[]) => { - const courses = await this.coursesService.findByInstructorIds(Array.from(instructorIds)); - - const coursesByInstructor = new Map(); - courses.forEach((course) => { - const instructorId = course.instructor?.id; - if (instructorId) { - if (!coursesByInstructor.has(instructorId)) { - coursesByInstructor.set(instructorId, []); - } - coursesByInstructor.get(instructorId).push(course); - } - }); - - return instructorIds.map((id) => coursesByInstructor.get(id) || []); - }); - } - - /** - * Create all loaders for a GraphQL request context - */ - createLoaders() { - return { - userLoader: this.createUserLoader(), - courseLoader: this.createCourseLoader(), - assessmentLoader: this.createAssessmentLoader(), - coursesByInstructorLoader: this.createCoursesByInstructorLoader(), - }; - } + constructor(private readonly usersService: UsersService, private readonly coursesService: CoursesService, private readonly assessmentsService: AssessmentsService) { } + /** + * Create a new DataLoader for batching user queries by ID + */ + createUserLoader(): DataLoader { + return new DataLoader(async (userIds: readonly string[]) => { + const users = await this.usersService.findByIds(Array.from(userIds)); + const userMap = new Map(users.map((user) => [user.id, user])); + return userIds.map((id) => userMap.get(id) || null); + }); + } + /** + * Create a new DataLoader for batching course queries by ID + */ + createCourseLoader(): DataLoader { + return new DataLoader(async (courseIds: readonly string[]) => { + const courses = await this.coursesService.findByIds(Array.from(courseIds)); + const courseMap = new Map(courses.map((course) => [course.id, course])); + return courseIds.map((id) => courseMap.get(id) || null); + }); + } + /** + * Create a new DataLoader for batching assessment queries by ID + */ + createAssessmentLoader(): DataLoader { + return new DataLoader(async (assessmentIds: readonly string[]) => { + const assessments = await this.assessmentsService.findByIds(Array.from(assessmentIds)); + const assessmentMap = new Map(assessments.map((assessment) => [assessment.id, assessment])); + return assessmentIds.map((id) => assessmentMap.get(id) || null); + }); + } + /** + * Create a new DataLoader for batching courses by instructor ID + */ + createCoursesByInstructorLoader(): DataLoader { + return new DataLoader(async (instructorIds: readonly string[]) => { + const courses = await this.coursesService.findByInstructorIds(Array.from(instructorIds)); + const coursesByInstructor = new Map(); + courses.forEach((course) => { + const instructorId = course.instructor?.id; + if (instructorId) { + if (!coursesByInstructor.has(instructorId)) { + coursesByInstructor.set(instructorId, []); + } + coursesByInstructor.get(instructorId).push(course); + } + }); + return instructorIds.map((id) => coursesByInstructor.get(id) || []); + }); + } + /** + * Create all loaders for a GraphQL request context + */ + createLoaders() { + return { + userLoader: this.createUserLoader(), + courseLoader: this.createCourseLoader(), + assessmentLoader: this.createAssessmentLoader(), + coursesByInstructorLoader: this.createCoursesByInstructorLoader(), + }; + } } diff --git a/src/graphql/services/query-complexity.service.ts b/src/graphql/services/query-complexity.service.ts index 9ddadb78..abaf7f86 100644 --- a/src/graphql/services/query-complexity.service.ts +++ b/src/graphql/services/query-complexity.service.ts @@ -1,36 +1,33 @@ import { Injectable, Logger } from '@nestjs/common'; import { GraphQLSchema, FieldNode, OperationDefinitionNode, visit, parse } from 'graphql'; - /** * Query complexity analysis configuration */ export interface ComplexityConfig { - /** Maximum query depth allowed (default: 10) */ - maxDepth: number; - /** Maximum complexity score allowed (default: 1000) */ - maxComplexity: number; - /** Complexity scalar multipliers for list fields */ - listScalarMultiplier: number; - /** Default complexity per field */ - defaultFieldComplexity: number; - /** Custom complexity values for specific types/fields */ - fieldComplexityMap: Record; + /** Maximum query depth allowed (default: 10) */ + maxDepth: number; + /** Maximum complexity score allowed (default: 1000) */ + maxComplexity: number; + /** Complexity scalar multipliers for list fields */ + listScalarMultiplier: number; + /** Default complexity per field */ + defaultFieldComplexity: number; + /** Custom complexity values for specific types/fields */ + fieldComplexityMap: Record; } - /** * Complexity analysis result */ export interface ComplexityResult { - /** Total complexity score */ - complexity: number; - /** Query depth */ - depth: number; - /** Whether the query exceeds limits */ - allowed: boolean; - /** Error message if not allowed */ - error?: string; + /** Total complexity score */ + complexity: number; + /** Query depth */ + depth: number; + /** Whether the query exceeds limits */ + allowed: boolean; + /** Error message if not allowed */ + error?: string; } - /** * Query Complexity Analysis Service * @@ -41,265 +38,226 @@ export interface ComplexityResult { */ @Injectable() export class QueryComplexityService { - private readonly logger = new Logger(QueryComplexityService.name); - - private readonly defaultConfig: ComplexityConfig = { - maxDepth: 10, - maxComplexity: 1000, - listScalarMultiplier: 10, - defaultFieldComplexity: 1, - fieldComplexityMap: { - // User queries - User: 5, - users: 10, - currentUser: 2, - - // Course queries - higher cost for list queries - Course: 5, - courses: 15, - courseById: 3, - popularCourses: 20, - searchCourses: 25, - - // Assessment queries - Assessment: 5, - assessments: 15, - assessmentById: 3, - questions: 20, - - // Results and analytics - results: 30, - analytics: 50, - statistics: 40, - - // Connection/N+1 patterns - edges: 2, - node: 1, - pageInfo: 1, - }, - }; - - private config: ComplexityConfig; - private schema: GraphQLSchema | null = null; - - constructor() { - // Initialize with environment-based configuration - this.config = { - ...this.defaultConfig, - maxDepth: parseInt(process.env.GRAPHQL_MAX_DEPTH || '10', 10), - maxComplexity: parseInt(process.env.GRAPHQL_MAX_COMPLEXITY || '1000', 10), - listScalarMultiplier: parseInt(process.env.GRAPHQL_LIST_MULTIPLIER || '10', 10), + private readonly logger = new Logger(QueryComplexityService.name); + private readonly defaultConfig: ComplexityConfig = { + maxDepth: 10, + maxComplexity: 1000, + listScalarMultiplier: 10, + defaultFieldComplexity: 1, + fieldComplexityMap: { + // User queries + User: 5, + users: 10, + currentUser: 2, + // Course queries - higher cost for list queries + Course: 5, + courses: 15, + courseById: 3, + popularCourses: 20, + searchCourses: 25, + // Assessment queries + Assessment: 5, + assessments: 15, + assessmentById: 3, + questions: 20, + // Results and analytics + results: 30, + analytics: 50, + statistics: 40, + // Connection/N+1 patterns + edges: 2, + node: 1, + pageInfo: 1, + }, }; - } - - /** - * Set the GraphQL schema for complexity analysis - */ - setSchema(schema: GraphQLSchema): void { - this.schema = schema; - } - - /** - * Update complexity configuration - */ - setConfig(config: Partial): void { - this.config = { ...this.defaultConfig, ...config }; - } - - /** - * Analyze a query for complexity - */ - analyze(query: string, variables?: Record): ComplexityResult { - try { - // Parse the query to get AST - const ast = parse(query); - - // Find the operation - const operation = this.findOperation(ast); - if (!operation) { - return { - complexity: 0, - depth: 0, - allowed: true, + private config: ComplexityConfig; + private schema: GraphQLSchema | null = null; + constructor() { + // Initialize with environment-based configuration + this.config = { + ...this.defaultConfig, + maxDepth: parseInt(process.env.GRAPHQL_MAX_DEPTH || '10', 10), + maxComplexity: parseInt(process.env.GRAPHQL_MAX_COMPLEXITY || '1000', 10), + listScalarMultiplier: parseInt(process.env.GRAPHQL_LIST_MULTIPLIER || '10', 10), }; - } - - // Calculate depth and complexity - const depth = this.calculateDepth(operation); - const complexity = this.calculateComplexity(operation, variables); - - // Check if within limits - const allowed = depth <= this.config.maxDepth && complexity <= this.config.maxComplexity; - - let error: string | undefined; - if (!allowed) { - if (depth > this.config.maxDepth) { - error = `Query depth ${depth} exceeds maximum allowed depth ${this.config.maxDepth}`; - } else { - error = `Query complexity ${complexity} exceeds maximum allowed complexity ${this.config.maxComplexity}`; + } + /** + * Set the GraphQL schema for complexity analysis + */ + setSchema(schema: GraphQLSchema): void { + this.schema = schema; + } + /** + * Update complexity configuration + */ + setConfig(config: Partial): void { + this.config = { ...this.defaultConfig, ...config }; + } + /** + * Analyze a query for complexity + */ + analyze(query: string, variables?: Record): ComplexityResult { + try { + // Parse the query to get AST + const ast = parse(query); + // Find the operation + const operation = this.findOperation(ast); + if (!operation) { + return { + complexity: 0, + depth: 0, + allowed: true, + }; + } + // Calculate depth and complexity + const depth = this.calculateDepth(operation); + const complexity = this.calculateComplexity(operation, variables); + // Check if within limits + const allowed = depth <= this.config.maxDepth && complexity <= this.config.maxComplexity; + let error: string | undefined; + if (!allowed) { + if (depth > this.config.maxDepth) { + error = `Query depth ${depth} exceeds maximum allowed depth ${this.config.maxDepth}`; + } + else { + error = `Query complexity ${complexity} exceeds maximum allowed complexity ${this.config.maxComplexity}`; + } + this.logger.warn(`Query rejected: ${error}`); + } + return { + complexity, + depth, + allowed, + error, + }; + } + catch (error) { + this.logger.error(`Error analyzing query complexity: ${error.message}`); + // On parse error, allow the query to proceed (it will fail at execution anyway) + return { + complexity: 0, + depth: 0, + allowed: true, + error: undefined, + }; } - this.logger.warn(`Query rejected: ${error}`); - } - - return { - complexity, - depth, - allowed, - error, - }; - } catch (error) { - this.logger.error(`Error analyzing query complexity: ${error.message}`); - // On parse error, allow the query to proceed (it will fail at execution anyway) - return { - complexity: 0, - depth: 0, - allowed: true, - error: undefined, - }; } - } - - /** - * Calculate the depth of a query - */ - private calculateDepth(operation: OperationDefinitionNode): number { - let maxDepth = 0; - - const visitor = { - Field: { - enter: (node: FieldNode) => { - const depth = this.getFieldDepth(node); - if (depth > maxDepth) { - maxDepth = depth; - } - }, - }, - }; - - visit(operation, visitor); - return maxDepth; - } - - /** - * Get the depth of a specific field - */ - private getFieldDepth(field: FieldNode, currentDepth: number = 0): number { - if (!field.selectionSet) { - return currentDepth; + /** + * Calculate the depth of a query + */ + private calculateDepth(operation: OperationDefinitionNode): number { + let maxDepth = 0; + const visitor = { + Field: { + enter: (node: FieldNode) => { + const depth = this.getFieldDepth(node); + if (depth > maxDepth) { + maxDepth = depth; + } + }, + }, + }; + visit(operation, visitor); + return maxDepth; } - - let maxDepth = currentDepth; - for (const selection of field.selectionSet.selections) { - if (selection.kind === 'Field') { - const depth = this.getFieldDepth(selection, currentDepth + 1); - if (depth > maxDepth) { - maxDepth = depth; + /** + * Get the depth of a specific field + */ + private getFieldDepth(field: FieldNode, currentDepth: number = 0): number { + if (!field.selectionSet) { + return currentDepth; } - } + let maxDepth = currentDepth; + for (const selection of field.selectionSet.selections) { + if (selection.kind === 'Field') { + const depth = this.getFieldDepth(selection, currentDepth + 1); + if (depth > maxDepth) { + maxDepth = depth; + } + } + } + return maxDepth; + } + /** + * Calculate the complexity score of a query + */ + private calculateComplexity(operation: OperationDefinitionNode, variables?: Record): number { + let totalComplexity = 0; + const visitor = { + Field: { + enter: (node: FieldNode): void => { + const fieldName = node.name.value; + const args = node.arguments || []; + // Get complexity for this field + let fieldComplexity = this.config.fieldComplexityMap[fieldName] || this.config.defaultFieldComplexity; + // Check for list arguments (like limit, first, last) + for (const arg of args) { + if (arg.name.value === 'limit' || + arg.name.value === 'first' || + arg.name.value === 'last') { + // Get the actual value from variables or literal + let listSize = 10; // Default + if (arg.value.kind === 'IntValue') { + listSize = parseInt(arg.value.value, 10); + } + else if (arg.value.kind === 'Variable' && variables) { + listSize = variables[arg.value.name.value] || 10; + } + // Multiply complexity by list size with scalar + fieldComplexity += (listSize - 1) * this.config.listScalarMultiplier; + } + } + totalComplexity += fieldComplexity; + // Stop if we exceeds max complexity (early termination) + if (totalComplexity > this.config.maxComplexity) { + // Early termination by throwing an error + throw new Error(`Query complexity ${totalComplexity} exceeds maximum allowed complexity of ${this.config.maxComplexity}`); + } + }, + }, + }; + visit(operation, visitor); + return totalComplexity; } - - return maxDepth; - } - - /** - * Calculate the complexity score of a query - */ - private calculateComplexity( - operation: OperationDefinitionNode, - variables?: Record, - ): number { - let totalComplexity = 0; - - const visitor = { - Field: { - enter: (node: FieldNode): void => { - const fieldName = node.name.value; - const args = node.arguments || []; - - // Get complexity for this field - let fieldComplexity = - this.config.fieldComplexityMap[fieldName] || this.config.defaultFieldComplexity; - - // Check for list arguments (like limit, first, last) - for (const arg of args) { - if ( - arg.name.value === 'limit' || - arg.name.value === 'first' || - arg.name.value === 'last' - ) { - // Get the actual value from variables or literal - let listSize = 10; // Default - if (arg.value.kind === 'IntValue') { - listSize = parseInt(arg.value.value, 10); - } else if (arg.value.kind === 'Variable' && variables) { - listSize = variables[arg.value.name.value] || 10; - } - // Multiply complexity by list size with scalar - fieldComplexity += (listSize - 1) * this.config.listScalarMultiplier; + /** + * Find the main operation in the query + */ + private findOperation(ast: unknown): OperationDefinitionNode | null { + const definitions = ast.definitions || []; + for (const def of definitions) { + if (def.kind === 'OperationDefinition') { + // Return first query operation + if (def.operation === 'query') { + return def; + } } - } - - totalComplexity += fieldComplexity; - - // Stop if we exceeds max complexity (early termination) - if (totalComplexity > this.config.maxComplexity) { - // Early termination by throwing an error - throw new Error( - `Query complexity ${totalComplexity} exceeds maximum allowed complexity of ${this.config.maxComplexity}`, - ); - } - }, - }, - }; - - visit(operation, visitor); - return totalComplexity; - } - - /** - * Find the main operation in the query - */ - private findOperation(ast: any): OperationDefinitionNode | null { - const definitions = ast.definitions || []; - - for (const def of definitions) { - if (def.kind === 'OperationDefinition') { - // Return first query operation - if (def.operation === 'query') { - return def; } - } + return null; + } + /** + * Get current configuration + */ + getConfig(): ComplexityConfig { + return { ...this.config }; + } + /** + * Validate a query against complexity limits + * Returns true if query is allowed, false otherwise + */ + validate(query: string, variables?: Record): boolean { + const result = this.analyze(query, variables); + return result.allowed; } - - return null; - } - - /** - * Get current configuration - */ - getConfig(): ComplexityConfig { - return { ...this.config }; - } - - /** - * Validate a query against complexity limits - * Returns true if query is allowed, false otherwise - */ - validate(query: string, variables?: Record): boolean { - const result = this.analyze(query, variables); - return result.allowed; - } } - /** * GraphQL complexity validator function for Apollo Server */ export function createComplexityValidator(service: QueryComplexityService) { - return (query: string, variables?: Record): Error | undefined => { - const result = service.analyze(query, variables); - if (!result.allowed && result.error) { - return new Error(result.error); - } - return undefined; - }; + return (query: string, variables?: Record): Error | undefined => { + const result = service.analyze(query, variables); + if (!result.allowed && result.error) { + return new Error(result.error); + } + return undefined; + }; } diff --git a/src/graphql/types/assessment.type.ts b/src/graphql/types/assessment.type.ts index 8d43d3e5..97f4a47a 100644 --- a/src/graphql/types/assessment.type.ts +++ b/src/graphql/types/assessment.type.ts @@ -1,43 +1,31 @@ import { ObjectType, Field, ID, Int } from '@nestjs/graphql'; - @ObjectType() export class QuestionType { - @Field(() => ID) - id: string; - - @Field() - prompt: string; - - @Field() - type: string; - - @Field(() => [String], { nullable: 'itemsAndList' }) - options?: string[]; - - @Field({ nullable: true }) - correctAnswer?: string; - - @Field(() => Int) - points: number; + @Field(() => ID) + id: string; + @Field() + prompt: string; + @Field() + type: string; + @Field(() => [String], { nullable: 'itemsAndList' }) + options?: string[]; + @Field({ nullable: true }) + correctAnswer?: string; + @Field(() => Int) + points: number; } - @ObjectType() export class AssessmentType { - @Field(() => ID) - id: string; - - @Field() - title: string; - - @Field({ nullable: true }) - description?: string; - - @Field(() => Int) - durationMinutes: number; - - @Field(() => [QuestionType], { nullable: 'itemsAndList' }) - questions?: QuestionType[]; - - @Field() - createdAt: Date; + @Field(() => ID) + id: string; + @Field() + title: string; + @Field({ nullable: true }) + description?: string; + @Field(() => Int) + durationMinutes: number; + @Field(() => [QuestionType], { nullable: 'itemsAndList' }) + questions?: QuestionType[]; + @Field() + createdAt: Date; } diff --git a/src/graphql/types/course.type.ts b/src/graphql/types/course.type.ts index 153ab5c6..e570e737 100644 --- a/src/graphql/types/course.type.ts +++ b/src/graphql/types/course.type.ts @@ -1,32 +1,23 @@ import { ObjectType, Field, ID, Float } from '@nestjs/graphql'; import { UserType } from './user.type'; - @ObjectType() export class CourseType { - @Field(() => ID) - id: string; - - @Field() - title: string; - - @Field() - description: string; - - @Field(() => Float) - price: number; - - @Field() - status: string; - - @Field({ nullable: true }) - thumbnailUrl?: string; - - @Field(() => UserType, { nullable: true }) - instructor?: UserType; - - @Field() - createdAt: Date; - - @Field() - updatedAt: Date; + @Field(() => ID) + id: string; + @Field() + title: string; + @Field() + description: string; + @Field(() => Float) + price: number; + @Field() + status: string; + @Field({ nullable: true }) + thumbnailUrl?: string; + @Field(() => UserType, { nullable: true }) + instructor?: UserType; + @Field() + createdAt: Date; + @Field() + updatedAt: Date; } diff --git a/src/graphql/types/user.type.ts b/src/graphql/types/user.type.ts index 9b98792e..dacfff4f 100644 --- a/src/graphql/types/user.type.ts +++ b/src/graphql/types/user.type.ts @@ -1,58 +1,42 @@ import { ObjectType, Field, ID, registerEnumType } from '@nestjs/graphql'; import { UserRole, UserStatus } from '../../users/entities/user.entity'; import { CourseType } from './course.type'; - registerEnumType(UserRole, { - name: 'UserRole', - description: 'User role in the system', + name: 'UserRole', + description: 'User role in the system', }); - registerEnumType(UserStatus, { - name: 'UserStatus', - description: 'User account status', + name: 'UserStatus', + description: 'User account status', }); - @ObjectType() export class UserType { - @Field(() => ID) - id: string; - - @Field() - email: string; - - @Field({ nullable: true }) - username?: string; - - @Field() - firstName: string; - - @Field() - lastName: string; - - @Field(() => UserRole) - role: UserRole; - - @Field(() => UserStatus) - status: UserStatus; - - @Field({ nullable: true }) - tenantId?: string; - - @Field({ nullable: true }) - profilePicture?: string; - - @Field() - isEmailVerified: boolean; - - @Field({ nullable: true }) - lastLoginAt?: Date; - - @Field(() => [CourseType], { nullable: 'itemsAndList' }) - courses?: CourseType[]; - - @Field() - createdAt: Date; - - @Field() - updatedAt: Date; + @Field(() => ID) + id: string; + @Field() + email: string; + @Field({ nullable: true }) + username?: string; + @Field() + firstName: string; + @Field() + lastName: string; + @Field(() => UserRole) + role: UserRole; + @Field(() => UserStatus) + status: UserStatus; + @Field({ nullable: true }) + tenantId?: string; + @Field({ nullable: true }) + profilePicture?: string; + @Field() + isEmailVerified: boolean; + @Field({ nullable: true }) + lastLoginAt?: Date; + @Field(() => [CourseType], { nullable: 'itemsAndList' }) + courses?: CourseType[]; + @Field() + createdAt: Date; + @Field() + updatedAt: Date; } diff --git a/src/health/health.controller.ts b/src/health/health.controller.ts index efa517ca..faa0b9be 100644 --- a/src/health/health.controller.ts +++ b/src/health/health.controller.ts @@ -1,12 +1,4 @@ -import { - Controller, - Get, - HttpStatus, - OnModuleDestroy, - Query, - Res, - UseGuards, -} from '@nestjs/common'; +import { Controller, Get, HttpStatus, OnModuleDestroy, Query, Res, UseGuards, } from '@nestjs/common'; import { ApiBearerAuth, ApiResponse, ApiTags } from '@nestjs/swagger'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { DataSource } from 'typeorm'; @@ -15,178 +7,155 @@ import { SkipThrottle } from '@nestjs/throttler'; import { Response } from 'express'; import { HealthService } from './health.service'; import { ShutdownStateService } from '../common/services/shutdown-state.service'; - @SkipThrottle() @ApiTags('health') @ApiBearerAuth() @UseGuards(JwtAuthGuard) @Controller('health') export class HealthController implements OnModuleDestroy { - private redis: Redis; - - constructor( - private readonly dataSource: DataSource, - private readonly healthService: HealthService, - private readonly shutdownState: ShutdownStateService, - ) { - this.redis = new Redis({ - host: process.env.REDIS_HOST, - port: Number(process.env.REDIS_PORT), - }); - - this.redis.on('error', () => { - // Health endpoint handles Redis failures explicitly in checkHealth. - }); - } - - async onModuleDestroy(): Promise { - if (this.redis.status !== 'end') { - await this.redis.quit(); + private redis: Redis; + constructor(private readonly dataSource: DataSource, private readonly healthService: HealthService, private readonly shutdownState: ShutdownStateService) { + this.redis = new Redis({ + host: process.env.REDIS_HOST, + port: Number(process.env.REDIS_PORT), + }); + this.redis.on('error', () => { + // Health endpoint handles Redis failures explicitly in checkHealth. + }); } - } - - @Get() - @ApiResponse({ status: HttpStatus.OK, description: 'Health check response' }) - async checkHealth() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - return healthStatus; - } - - @Get('liveness') - @ApiResponse({ status: HttpStatus.OK, description: 'Liveness check response' }) - @ApiResponse({ - status: HttpStatus.SERVICE_UNAVAILABLE, - description: 'Application is shutting down', - }) - checkLiveness(@Res() res: Response) { - if (this.shutdownState.isShuttingDown()) { - return res.status(HttpStatus.SERVICE_UNAVAILABLE).json({ - status: 'shutting_down', - timestamp: new Date().toISOString(), - }); + async onModuleDestroy(): Promise { + if (this.redis.status !== 'end') { + await this.redis.quit(); + } } - - return res.status(HttpStatus.OK).json({ - status: 'ok', - timestamp: new Date().toISOString(), - }); - } - - @Get('readiness') - @ApiResponse({ - status: HttpStatus.OK, - description: 'Readiness check response', - }) - @ApiResponse({ - status: HttpStatus.SERVICE_UNAVAILABLE, - description: 'Application is shutting down', - }) - async checkReadiness(@Res() res: Response) { - if (this.shutdownState.isShuttingDown()) { - return res.status(HttpStatus.SERVICE_UNAVAILABLE).json({ - status: 'shutting_down', - timestamp: new Date().toISOString(), - }); + @Get() + @ApiResponse({ status: HttpStatus.OK, description: 'Health check response' }) + async checkHealth() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + return healthStatus; } - - const healthStatus = await this.healthService.checkReadiness(this.dataSource, this.redis); - return res.status(HttpStatus.OK).json(healthStatus); - } - - @Get('dependencies') - async checkDependencies(@Query('service') service?: string) { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - - if (service) { - return { - service, - status: healthStatus.services[service] || 'unknown', - details: healthStatus.details[service] || null, - timestamp: healthStatus.timestamp, - }; + @Get('liveness') + @ApiResponse({ status: HttpStatus.OK, description: 'Liveness check response' }) + @ApiResponse({ + status: HttpStatus.SERVICE_UNAVAILABLE, + description: 'Application is shutting down', + }) + checkLiveness( + @Res() + res: Response) { + if (this.shutdownState.isShuttingDown()) { + return res.status(HttpStatus.SERVICE_UNAVAILABLE).json({ + status: 'shutting_down', + timestamp: new Date().toISOString(), + }); + } + return res.status(HttpStatus.OK).json({ + status: 'ok', + timestamp: new Date().toISOString(), + }); + } + @Get('readiness') + @ApiResponse({ + status: HttpStatus.OK, + description: 'Readiness check response', + }) + @ApiResponse({ + status: HttpStatus.SERVICE_UNAVAILABLE, + description: 'Application is shutting down', + }) + async checkReadiness( + @Res() + res: Response) { + if (this.shutdownState.isShuttingDown()) { + return res.status(HttpStatus.SERVICE_UNAVAILABLE).json({ + status: 'shutting_down', + timestamp: new Date().toISOString(), + }); + } + const healthStatus = await this.healthService.checkReadiness(this.dataSource, this.redis); + return res.status(HttpStatus.OK).json(healthStatus); + } + @Get('dependencies') + async checkDependencies( + @Query('service') + service?: string) { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + if (service) { + return { + service, + status: healthStatus.services[service] || 'unknown', + details: healthStatus.details[service] || null, + timestamp: healthStatus.timestamp, + }; + } + return { + dependencies: healthStatus.services, + details: healthStatus.details, + timestamp: healthStatus.timestamp, + overallStatus: healthStatus.status, + }; + } + @Get('database') + async checkDatabase() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + return { + service: 'database', + status: healthStatus.services.database, + details: healthStatus.details.database, + timestamp: healthStatus.timestamp, + }; + } + @Get('redis') + async checkRedis() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + return { + service: 'redis', + status: healthStatus.services.redis, + details: healthStatus.details.redis, + timestamp: healthStatus.timestamp, + }; + } + @Get('queue') + async checkQueue() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + return { + service: 'queue', + status: healthStatus.services.queue, + details: healthStatus.details.queue, + timestamp: healthStatus.timestamp, + }; + } + @Get('cache') + async checkCache() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + return { + service: 'cache', + status: healthStatus.services.cache, + details: healthStatus.details.cache, + timestamp: healthStatus.timestamp, + }; + } + @Get('summary') + async getHealthSummary() { + const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); + const serviceCount = Object.keys(healthStatus.services).length; + const healthyCount = Object.values(healthStatus.services).filter((status) => status === 'up').length; + const degradedCount = Object.values(healthStatus.services).filter((status) => status === 'degraded' || status === 'warning').length; + const criticalCount = Object.values(healthStatus.services).filter((status) => status === 'down' || status === 'critical').length; + return { + overall: healthStatus.status, + timestamp: healthStatus.timestamp, + uptime: healthStatus.uptime, + version: healthStatus.version, + environment: healthStatus.environment, + summary: { + total: serviceCount, + healthy: healthyCount, + degraded: degradedCount, + critical: criticalCount, + healthScore: Math.round((healthyCount / serviceCount) * 100), + }, + services: healthStatus.services, + }; } - - return { - dependencies: healthStatus.services, - details: healthStatus.details, - timestamp: healthStatus.timestamp, - overallStatus: healthStatus.status, - }; - } - - @Get('database') - async checkDatabase() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - return { - service: 'database', - status: healthStatus.services.database, - details: healthStatus.details.database, - timestamp: healthStatus.timestamp, - }; - } - - @Get('redis') - async checkRedis() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - return { - service: 'redis', - status: healthStatus.services.redis, - details: healthStatus.details.redis, - timestamp: healthStatus.timestamp, - }; - } - - @Get('queue') - async checkQueue() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - return { - service: 'queue', - status: healthStatus.services.queue, - details: healthStatus.details.queue, - timestamp: healthStatus.timestamp, - }; - } - - @Get('cache') - async checkCache() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - return { - service: 'cache', - status: healthStatus.services.cache, - details: healthStatus.details.cache, - timestamp: healthStatus.timestamp, - }; - } - - @Get('summary') - async getHealthSummary() { - const healthStatus = await this.healthService.checkHealth(this.dataSource, this.redis); - - const serviceCount = Object.keys(healthStatus.services).length; - const healthyCount = Object.values(healthStatus.services).filter( - (status) => status === 'up', - ).length; - const degradedCount = Object.values(healthStatus.services).filter( - (status) => status === 'degraded' || status === 'warning', - ).length; - const criticalCount = Object.values(healthStatus.services).filter( - (status) => status === 'down' || status === 'critical', - ).length; - - return { - overall: healthStatus.status, - timestamp: healthStatus.timestamp, - uptime: healthStatus.uptime, - version: healthStatus.version, - environment: healthStatus.environment, - summary: { - total: serviceCount, - healthy: healthyCount, - degraded: degradedCount, - critical: criticalCount, - healthScore: Math.round((healthyCount / serviceCount) * 100), - }, - services: healthStatus.services, - }; - } } diff --git a/src/health/health.module.ts b/src/health/health.module.ts index a38cb2c8..60e1294b 100644 --- a/src/health/health.module.ts +++ b/src/health/health.module.ts @@ -1,10 +1,10 @@ import { Module } from '@nestjs/common'; import { HealthController } from './health.controller'; import { HealthService } from './health.service'; - @Module({ - controllers: [HealthController], - providers: [HealthService], - exports: [HealthService], + controllers: [HealthController], + providers: [HealthService], + exports: [HealthService], }) -export class HealthModule {} +export class HealthModule { +} diff --git a/src/health/health.service.ts b/src/health/health.service.ts index e2cf21e7..76aeefec 100644 --- a/src/health/health.service.ts +++ b/src/health/health.service.ts @@ -4,535 +4,519 @@ import { Redis } from 'ioredis'; import * as fs from 'fs'; import * as _path from 'path'; import axios from 'axios'; - export interface HealthStatus { - status: 'ok' | 'degraded' | 'critical'; - timestamp: string; - uptime: number; - version?: string; - environment?: string; - services: { - database: string; - redis: string; - externalApis: Record; - disk: string; - queue?: string; - cache?: string; - featureFlags?: string; - bull?: string; - }; - details?: { - database?: { - responseTime: number; - connectionStatus: string; - connectionCount?: number; - maxConnections?: number; + status: 'ok' | 'degraded' | 'critical'; + timestamp: string; + uptime: number; + version?: string; + environment?: string; + services: { + database: string; + redis: string; + externalApis: Record; + disk: string; + queue?: string; + cache?: string; + featureFlags?: string; + bull?: string; }; - redis?: { - responseTime: number; - connectionStatus: string; - memory?: { - used: number; - total: number; - percentage: number; - }; - }; - disk?: { - used: number; - total: number; - percentage: number; - }; - externalApis?: Record; - queue?: { - activeJobs: number; - waitingJobs: number; - failedJobs: number; - responseTime: number; - }; - cache?: { - hitRate: number; - missRate: number; - responseTime: number; - }; - featureFlags?: { - responseTime: number; - status: string; - }; - bull?: { - activeQueues: number; - totalJobs: number; - responseTime: number; + details?: { + database?: { + responseTime: number; + connectionStatus: string; + connectionCount?: number; + maxConnections?: number; + }; + redis?: { + responseTime: number; + connectionStatus: string; + memory?: { + used: number; + total: number; + percentage: number; + }; + }; + disk?: { + used: number; + total: number; + percentage: number; + }; + externalApis?: Record; + queue?: { + activeJobs: number; + waitingJobs: number; + failedJobs: number; + responseTime: number; + }; + cache?: { + hitRate: number; + missRate: number; + responseTime: number; + }; + featureFlags?: { + responseTime: number; + status: string; + }; + bull?: { + activeQueues: number; + totalJobs: number; + responseTime: number; + }; }; - }; } - @Injectable() export class HealthService { - private readonly logger = new Logger(HealthService.name); - private readonly startTime = Date.now(); - - // External API endpoints to check - private readonly externalApiEndpoints = [ - { name: 'stripe', url: process.env.STRIPE_HEALTH_URL, key: 'STRIPE_SECRET_KEY' }, - { name: 'sendgrid', url: process.env.SENDGRID_HEALTH_URL, key: 'SENDGRID_API_KEY' }, - { name: 'aws', url: process.env.AWS_HEALTH_URL, key: 'AWS_ACCESS_KEY_ID' }, - ]; - - // Disk space thresholds - private readonly diskWarningThreshold = 85; // 85% - private readonly diskCriticalThreshold = 95; // 95% - - async checkHealth(dataSource: DataSource, redis: Redis): Promise { - const healthStatus: HealthStatus = { - status: 'ok', - timestamp: new Date().toISOString(), - uptime: Date.now() - this.startTime, - version: process.env.npm_package_version || 'unknown', - environment: process.env.NODE_ENV || 'development', - services: { - database: 'unknown', - redis: 'unknown', - externalApis: {}, - disk: 'unknown', - queue: 'unknown', - cache: 'unknown', - featureFlags: 'unknown', - bull: 'unknown', - }, - details: {}, - }; - - // Check database - const dbCheck = await this.checkDatabase(dataSource); - healthStatus.services.database = dbCheck.status; - healthStatus.details.database = { - responseTime: dbCheck.responseTime, - connectionStatus: dbCheck.status, - }; - if (dbCheck.status === 'down') { - healthStatus.status = 'degraded'; - } - - // Check Redis - const redisCheck = await this.checkRedis(redis); - healthStatus.services.redis = redisCheck.status; - healthStatus.details.redis = { - responseTime: redisCheck.responseTime, - connectionStatus: redisCheck.status, - }; - if (redisCheck.status === 'down') { - healthStatus.status = 'degraded'; - } - - // Check external APIs - const apiChecks = await this.checkExternalApis(); - healthStatus.services.externalApis = {}; - for (const [apiName, check] of Object.entries(apiChecks)) { - healthStatus.services.externalApis[apiName] = check.status; - } - healthStatus.details.externalApis = apiChecks; - - // Check if any external API is down - const anyApiDown = Object.values(apiChecks).some((check) => check.status === 'down'); - if (anyApiDown && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - // Check disk space - const diskCheck = await this.checkDiskSpace(); - healthStatus.services.disk = diskCheck.status; - healthStatus.details.disk = { - used: diskCheck.used, - total: diskCheck.total, - percentage: diskCheck.percentage, - }; - if (diskCheck.status === 'critical') { - healthStatus.status = 'critical'; - } else if (diskCheck.status === 'warning' && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - // Check queue service - const queueCheck = await this.checkQueueService(); - healthStatus.services.queue = queueCheck.status; - healthStatus.details.queue = queueCheck; - if (queueCheck.status === 'down' && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - // Check cache service - const cacheCheck = await this.checkCacheService(redis); - healthStatus.services.cache = cacheCheck.status; - healthStatus.details.cache = cacheCheck; - if (cacheCheck.status === 'down' && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - // Check feature flags service - const featureFlagsCheck = await this.checkFeatureFlagsService(); - healthStatus.services.featureFlags = featureFlagsCheck.status; - healthStatus.details.featureFlags = featureFlagsCheck; - if (featureFlagsCheck.status === 'down' && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - // Check Bull queue service - const bullCheck = await this.checkBullService(); - healthStatus.services.bull = bullCheck.status; - healthStatus.details.bull = bullCheck; - if (bullCheck.status === 'down' && healthStatus.status === 'ok') { - healthStatus.status = 'degraded'; - } - - return healthStatus; - } - - async checkReadiness(dataSource: DataSource, redis: Redis): Promise { - // For readiness, we check if core dependencies are available - const healthStatus: HealthStatus = { - status: 'ok', - timestamp: new Date().toISOString(), - uptime: Date.now() - this.startTime, - services: { - database: 'unknown', - redis: 'unknown', - externalApis: {}, - disk: 'unknown', - }, - }; - - try { - const dbCheck = await this.checkDatabase(dataSource); - healthStatus.services.database = dbCheck.status; - if (dbCheck.status === 'down') { - healthStatus.status = 'critical'; - } - } catch { - healthStatus.services.database = 'down'; - healthStatus.status = 'critical'; + private readonly logger = new Logger(HealthService.name); + private readonly startTime = Date.now(); + // External API endpoints to check + private readonly externalApiEndpoints = [ + { name: 'stripe', url: process.env.STRIPE_HEALTH_URL, key: 'STRIPE_SECRET_KEY' }, + { name: 'sendgrid', url: process.env.SENDGRID_HEALTH_URL, key: 'SENDGRID_API_KEY' }, + { name: 'aws', url: process.env.AWS_HEALTH_URL, key: 'AWS_ACCESS_KEY_ID' }, + ]; + // Disk space thresholds + private readonly diskWarningThreshold = 85; // 85% + private readonly diskCriticalThreshold = 95; // 95% + async checkHealth(dataSource: DataSource, redis: Redis): Promise { + const healthStatus: HealthStatus = { + status: 'ok', + timestamp: new Date().toISOString(), + uptime: Date.now() - this.startTime, + version: process.env.npm_package_version || 'unknown', + environment: process.env.NODE_ENV || 'development', + services: { + database: 'unknown', + redis: 'unknown', + externalApis: {}, + disk: 'unknown', + queue: 'unknown', + cache: 'unknown', + featureFlags: 'unknown', + bull: 'unknown', + }, + details: {}, + }; + // Check database + const dbCheck = await this.checkDatabase(dataSource); + healthStatus.services.database = dbCheck.status; + healthStatus.details.database = { + responseTime: dbCheck.responseTime, + connectionStatus: dbCheck.status, + }; + if (dbCheck.status === 'down') { + healthStatus.status = 'degraded'; + } + // Check Redis + const redisCheck = await this.checkRedis(redis); + healthStatus.services.redis = redisCheck.status; + healthStatus.details.redis = { + responseTime: redisCheck.responseTime, + connectionStatus: redisCheck.status, + }; + if (redisCheck.status === 'down') { + healthStatus.status = 'degraded'; + } + // Check external APIs + const apiChecks = await this.checkExternalApis(); + healthStatus.services.externalApis = {}; + for (const [apiName, check] of Object.entries(apiChecks)) { + healthStatus.services.externalApis[apiName] = check.status; + } + healthStatus.details.externalApis = apiChecks; + // Check if any external API is down + const anyApiDown = Object.values(apiChecks).some((check) => check.status === 'down'); + if (anyApiDown && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + // Check disk space + const diskCheck = await this.checkDiskSpace(); + healthStatus.services.disk = diskCheck.status; + healthStatus.details.disk = { + used: diskCheck.used, + total: diskCheck.total, + percentage: diskCheck.percentage, + }; + if (diskCheck.status === 'critical') { + healthStatus.status = 'critical'; + } + else if (diskCheck.status === 'warning' && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + // Check queue service + const queueCheck = await this.checkQueueService(); + healthStatus.services.queue = queueCheck.status; + healthStatus.details.queue = queueCheck; + if (queueCheck.status === 'down' && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + // Check cache service + const cacheCheck = await this.checkCacheService(redis); + healthStatus.services.cache = cacheCheck.status; + healthStatus.details.cache = cacheCheck; + if (cacheCheck.status === 'down' && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + // Check feature flags service + const featureFlagsCheck = await this.checkFeatureFlagsService(); + healthStatus.services.featureFlags = featureFlagsCheck.status; + healthStatus.details.featureFlags = featureFlagsCheck; + if (featureFlagsCheck.status === 'down' && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + // Check Bull queue service + const bullCheck = await this.checkBullService(); + healthStatus.services.bull = bullCheck.status; + healthStatus.details.bull = bullCheck; + if (bullCheck.status === 'down' && healthStatus.status === 'ok') { + healthStatus.status = 'degraded'; + } + return healthStatus; } - - try { - const redisCheck = await this.checkRedis(redis); - healthStatus.services.redis = redisCheck.status; - if (redisCheck.status === 'down') { - healthStatus.status = 'critical'; - } - } catch { - healthStatus.services.redis = 'down'; - healthStatus.status = 'critical'; + async checkReadiness(dataSource: DataSource, redis: Redis): Promise { + // For readiness, we check if core dependencies are available + const healthStatus: HealthStatus = { + status: 'ok', + timestamp: new Date().toISOString(), + uptime: Date.now() - this.startTime, + services: { + database: 'unknown', + redis: 'unknown', + externalApis: {}, + disk: 'unknown', + }, + }; + try { + const dbCheck = await this.checkDatabase(dataSource); + healthStatus.services.database = dbCheck.status; + if (dbCheck.status === 'down') { + healthStatus.status = 'critical'; + } + } + catch { + healthStatus.services.database = 'down'; + healthStatus.status = 'critical'; + } + try { + const redisCheck = await this.checkRedis(redis); + healthStatus.services.redis = redisCheck.status; + if (redisCheck.status === 'down') { + healthStatus.status = 'critical'; + } + } + catch { + healthStatus.services.redis = 'down'; + healthStatus.status = 'critical'; + } + return healthStatus; } - - return healthStatus; - } - - private async checkDatabase(dataSource: DataSource): Promise<{ - status: string; - responseTime: number; - connectionCount?: number; - maxConnections?: number; - }> { - const startTime = Date.now(); - try { - await dataSource.query('SELECT 1'); - const responseTime = Date.now() - startTime; - - // Get connection pool stats - let connectionCount = 0; - let maxConnections = 0; - try { - const poolStats = await dataSource.query(` + private async checkDatabase(dataSource: DataSource): Promise<{ + status: string; + responseTime: number; + connectionCount?: number; + maxConnections?: number; + }> { + const startTime = Date.now(); + try { + await dataSource.query('SELECT 1'); + const responseTime = Date.now() - startTime; + // Get connection pool stats + let connectionCount = 0; + let maxConnections = 0; + try { + const poolStats = await dataSource.query(` SELECT count(*) as active_connections FROM pg_stat_activity WHERE state = 'active' `); - connectionCount = parseInt(poolStats[0]?.active_connections || '0'); - maxConnections = parseInt(process.env.DATABASE_POOL_MAX || '30'); - } catch (poolError) { - this.logger.warn(`Failed to get connection pool stats: ${poolError.message}`); - } - - return { - status: responseTime < 1000 ? 'up' : 'degraded', - responseTime, - connectionCount, - maxConnections, - }; - } catch (error) { - this.logger.error(`Database health check failed: ${error.message}`); - return { - status: 'down', - responseTime: Date.now() - startTime, - }; - } - } - - private async checkRedis(redis: Redis): Promise<{ - status: string; - responseTime: number; - memory?: { - used: number; - total: number; - percentage: number; - }; - }> { - const startTime = Date.now(); - try { - const pong = await (redis as any).ping(); - const responseTime = Date.now() - startTime; - - // Get Redis memory info - let memoryInfo = undefined; - try { - const info = await redis.info('memory'); - const lines = info.split('\r\n'); - const usedMemory = lines.find((line) => line.startsWith('used_memory:'))?.split(':')[1]; - const maxMemory = lines.find((line) => line.startsWith('maxmemory:'))?.split(':')[1]; - - if (usedMemory) { - const used = parseInt(usedMemory); - const total = maxMemory ? parseInt(maxMemory) : used * 2; // Estimate if max not set - memoryInfo = { - used, - total, - percentage: Math.round((used / total) * 100), - }; + connectionCount = parseInt(poolStats[0]?.active_connections || '0'); + maxConnections = parseInt(process.env.DATABASE_POOL_MAX || '30'); + } + catch (poolError) { + this.logger.warn(`Failed to get connection pool stats: ${poolError.message}`); + } + return { + status: responseTime < 1000 ? 'up' : 'degraded', + responseTime, + connectionCount, + maxConnections, + }; + } + catch (error) { + this.logger.error(`Database health check failed: ${error.message}`); + return { + status: 'down', + responseTime: Date.now() - startTime, + }; } - } catch (memError) { - this.logger.warn(`Failed to get Redis memory info: ${memError.message}`); - } - - return { - status: pong === 'PONG' && responseTime < 500 ? 'up' : 'degraded', - responseTime, - memory: memoryInfo, - }; - } catch (error) { - this.logger.error(`Redis health check failed: ${error.message}`); - return { - status: 'down', - responseTime: Date.now() - startTime, - }; } - } - - private async checkExternalApis(): Promise< - Record - > { - const results: Record = {}; - - for (const endpoint of this.externalApiEndpoints) { - // Skip if API key is not configured - if (!process.env[endpoint.key]) { - results[endpoint.name] = { - status: 'not_configured', - responseTime: 0, - }; - continue; - } - - // Skip if URL is not configured - if (!endpoint.url) { - results[endpoint.name] = { - status: 'not_configured', - responseTime: 0, - }; - continue; - } - - const startTime = Date.now(); - try { - const response = await axios.get(endpoint.url, { - timeout: 5000, - validateStatus: (status) => status < 500, - }); - const responseTime = Date.now() - startTime; - results[endpoint.name] = { - status: response.status < 400 ? 'up' : 'down', - responseTime, + private async checkRedis(redis: Redis): Promise<{ + status: string; + responseTime: number; + memory?: { + used: number; + total: number; + percentage: number; }; - } catch (error) { - const responseTime = Date.now() - startTime; - this.logger.warn(`External API health check failed for ${endpoint.name}: ${error.message}`); - results[endpoint.name] = { - status: 'down', - responseTime, - error: error.message, - }; - } + }> { + const startTime = Date.now(); + try { + const pong = await (redis as unknown).ping(); + const responseTime = Date.now() - startTime; + // Get Redis memory info + let memoryInfo = undefined; + try { + const info = await redis.info('memory'); + const lines = info.split('\r\n'); + const usedMemory = lines.find((line) => line.startsWith('used_memory:'))?.split(':')[1]; + const maxMemory = lines.find((line) => line.startsWith('maxmemory:'))?.split(':')[1]; + if (usedMemory) { + const used = parseInt(usedMemory); + const total = maxMemory ? parseInt(maxMemory) : used * 2; // Estimate if max not set + memoryInfo = { + used, + total, + percentage: Math.round((used / total) * 100), + }; + } + } + catch (memError) { + this.logger.warn(`Failed to get Redis memory info: ${memError.message}`); + } + return { + status: pong === 'PONG' && responseTime < 500 ? 'up' : 'degraded', + responseTime, + memory: memoryInfo, + }; + } + catch (error) { + this.logger.error(`Redis health check failed: ${error.message}`); + return { + status: 'down', + responseTime: Date.now() - startTime, + }; + } } - - return results; - } - - private async checkDiskSpace(): Promise<{ - status: string; - used: number; - total: number; - percentage: number; - }> { - try { - // Get disk space using Node.js fs module - const stats = await fs.promises.statfs(process.cwd()); - const total = stats.bsize * stats.blocks; - const free = stats.bsize * stats.bfree; - const used = total - free; - const percentage = Math.round((used / total) * 100); - - let status = 'up'; - if (percentage >= this.diskCriticalThreshold) { - status = 'critical'; - } else if (percentage >= this.diskWarningThreshold) { - status = 'warning'; - } - - return { - status, - used, - total, - percentage, - }; - } catch (error) { - this.logger.error(`Disk space check failed: ${error.message}`); - return { - status: 'unknown', - used: 0, - total: 0, - percentage: 0, - }; + private async checkExternalApis(): Promise> { + const results: Record = {}; + for (const endpoint of this.externalApiEndpoints) { + // Skip if API key is not configured + if (!process.env[endpoint.key]) { + results[endpoint.name] = { + status: 'not_configured', + responseTime: 0, + }; + continue; + } + // Skip if URL is not configured + if (!endpoint.url) { + results[endpoint.name] = { + status: 'not_configured', + responseTime: 0, + }; + continue; + } + const startTime = Date.now(); + try { + const response = await axios.get(endpoint.url, { + timeout: 5000, + validateStatus: (status) => status < 500, + }); + const responseTime = Date.now() - startTime; + results[endpoint.name] = { + status: response.status < 400 ? 'up' : 'down', + responseTime, + }; + } + catch (error) { + const responseTime = Date.now() - startTime; + this.logger.warn(`External API health check failed for ${endpoint.name}: ${error.message}`); + results[endpoint.name] = { + status: 'down', + responseTime, + error: error.message, + }; + } + } + return results; } - } - - private async checkQueueService(): Promise<{ - status: string; - activeJobs: number; - waitingJobs: number; - failedJobs: number; - responseTime: number; - }> { - const startTime = Date.now(); - try { - // Simulate queue service check - in real implementation, this would query actual queue service - const responseTime = Date.now() - startTime; - return { - status: responseTime < 1000 ? 'up' : 'degraded', - activeJobs: Math.floor(Math.random() * 10), - waitingJobs: Math.floor(Math.random() * 50), - failedJobs: Math.floor(Math.random() * 5), - responseTime, - }; - } catch (error) { - this.logger.error(`Queue service health check failed: ${error.message}`); - return { - status: 'down', - activeJobs: 0, - waitingJobs: 0, - failedJobs: 0, - responseTime: Date.now() - startTime, - }; + private async checkDiskSpace(): Promise<{ + status: string; + used: number; + total: number; + percentage: number; + }> { + try { + // Get disk space using Node.js fs module + const stats = await fs.promises.statfs(process.cwd()); + const total = stats.bsize * stats.blocks; + const free = stats.bsize * stats.bfree; + const used = total - free; + const percentage = Math.round((used / total) * 100); + let status = 'up'; + if (percentage >= this.diskCriticalThreshold) { + status = 'critical'; + } + else if (percentage >= this.diskWarningThreshold) { + status = 'warning'; + } + return { + status, + used, + total, + percentage, + }; + } + catch (error) { + this.logger.error(`Disk space check failed: ${error.message}`); + return { + status: 'unknown', + used: 0, + total: 0, + percentage: 0, + }; + } } - } - - private async checkCacheService(redis: Redis): Promise<{ - status: string; - hitRate: number; - missRate: number; - responseTime: number; - }> { - const startTime = Date.now(); - try { - // Test cache with a simple get/set operation - const testKey = 'health_check_test'; - await redis.set(testKey, 'test', 'EX', 10); - const value = await redis.get(testKey); - const responseTime = Date.now() - startTime; - - // Get cache stats - let hitRate = 0; - let missRate = 0; - try { - const info = await redis.info('stats'); - const lines = info.split('\r\n'); - const hits = lines.find((line) => line.startsWith('keyspace_hits:'))?.split(':')[1]; - const misses = lines.find((line) => line.startsWith('keyspace_misses:'))?.split(':')[1]; - - if (hits && misses) { - const totalHits = parseInt(hits); - const totalMisses = parseInt(misses); - const total = totalHits + totalMisses; - if (total > 0) { - hitRate = Math.round((totalHits / total) * 100); - missRate = Math.round((totalMisses / total) * 100); - } + private async checkQueueService(): Promise<{ + status: string; + activeJobs: number; + waitingJobs: number; + failedJobs: number; + responseTime: number; + }> { + const startTime = Date.now(); + try { + // Simulate queue service check - in real implementation, this would query actual queue service + const responseTime = Date.now() - startTime; + return { + status: responseTime < 1000 ? 'up' : 'degraded', + activeJobs: Math.floor(Math.random() * 10), + waitingJobs: Math.floor(Math.random() * 50), + failedJobs: Math.floor(Math.random() * 5), + responseTime, + }; + } + catch (error) { + this.logger.error(`Queue service health check failed: ${error.message}`); + return { + status: 'down', + activeJobs: 0, + waitingJobs: 0, + failedJobs: 0, + responseTime: Date.now() - startTime, + }; + } + } + private async checkCacheService(redis: Redis): Promise<{ + status: string; + hitRate: number; + missRate: number; + responseTime: number; + }> { + const startTime = Date.now(); + try { + // Test cache with a simple get/set operation + const testKey = 'health_check_test'; + await redis.set(testKey, 'test', 'EX', 10); + const value = await redis.get(testKey); + const responseTime = Date.now() - startTime; + // Get cache stats + let hitRate = 0; + let missRate = 0; + try { + const info = await redis.info('stats'); + const lines = info.split('\r\n'); + const hits = lines.find((line) => line.startsWith('keyspace_hits:'))?.split(':')[1]; + const misses = lines.find((line) => line.startsWith('keyspace_misses:'))?.split(':')[1]; + if (hits && misses) { + const totalHits = parseInt(hits); + const totalMisses = parseInt(misses); + const total = totalHits + totalMisses; + if (total > 0) { + hitRate = Math.round((totalHits / total) * 100); + missRate = Math.round((totalMisses / total) * 100); + } + } + } + catch (statsError) { + this.logger.warn(`Failed to get cache stats: ${statsError.message}`); + } + return { + status: value === 'test' && responseTime < 500 ? 'up' : 'degraded', + hitRate, + missRate, + responseTime, + }; + } + catch (error) { + this.logger.error(`Cache service health check failed: ${error.message}`); + return { + status: 'down', + hitRate: 0, + missRate: 0, + responseTime: Date.now() - startTime, + }; } - } catch (statsError) { - this.logger.warn(`Failed to get cache stats: ${statsError.message}`); - } - - return { - status: value === 'test' && responseTime < 500 ? 'up' : 'degraded', - hitRate, - missRate, - responseTime, - }; - } catch (error) { - this.logger.error(`Cache service health check failed: ${error.message}`); - return { - status: 'down', - hitRate: 0, - missRate: 0, - responseTime: Date.now() - startTime, - }; } - } - - private async checkFeatureFlagsService(): Promise<{ - status: string; - responseTime: number; - }> { - const startTime = Date.now(); - try { - // Simulate feature flags service check - // In real implementation, this would check the actual feature flags service - const responseTime = Date.now() - startTime; - return { - status: responseTime < 500 ? 'up' : 'degraded', - responseTime, - }; - } catch (error) { - this.logger.error(`Feature flags service health check failed: ${error.message}`); - return { - status: 'down', - responseTime: Date.now() - startTime, - }; + private async checkFeatureFlagsService(): Promise<{ + status: string; + responseTime: number; + }> { + const startTime = Date.now(); + try { + // Simulate feature flags service check + // In real implementation, this would check the actual feature flags service + const responseTime = Date.now() - startTime; + return { + status: responseTime < 500 ? 'up' : 'degraded', + responseTime, + }; + } + catch (error) { + this.logger.error(`Feature flags service health check failed: ${error.message}`); + return { + status: 'down', + responseTime: Date.now() - startTime, + }; + } } - } - - private async checkBullService(): Promise<{ - status: string; - activeQueues: number; - totalJobs: number; - responseTime: number; - }> { - const startTime = Date.now(); - try { - // Simulate Bull queue service check - // In real implementation, this would check actual Bull queues - const responseTime = Date.now() - startTime; - return { - status: responseTime < 1000 ? 'up' : 'degraded', - activeQueues: Math.floor(Math.random() * 5) + 1, - totalJobs: Math.floor(Math.random() * 1000), - responseTime, - }; - } catch (error) { - this.logger.error(`Bull service health check failed: ${error.message}`); - return { - status: 'down', - activeQueues: 0, - totalJobs: 0, - responseTime: Date.now() - startTime, - }; + private async checkBullService(): Promise<{ + status: string; + activeQueues: number; + totalJobs: number; + responseTime: number; + }> { + const startTime = Date.now(); + try { + // Simulate Bull queue service check + // In real implementation, this would check actual Bull queues + const responseTime = Date.now() - startTime; + return { + status: responseTime < 1000 ? 'up' : 'degraded', + activeQueues: Math.floor(Math.random() * 5) + 1, + totalJobs: Math.floor(Math.random() * 1000), + responseTime, + }; + } + catch (error) { + this.logger.error(`Bull service health check failed: ${error.message}`); + return { + status: 'down', + activeQueues: 0, + totalJobs: 0, + responseTime: Date.now() - startTime, + }; + } } - } } diff --git a/src/interfaces/api-error.interface.ts b/src/interfaces/api-error.interface.ts index bf3be158..606463c4 100644 --- a/src/interfaces/api-error.interface.ts +++ b/src/interfaces/api-error.interface.ts @@ -1,15 +1,14 @@ export interface ApiError { - statusCode: number; - message: string | string[]; - error?: string; - details?: ValidationErrorDetail[]; - timestamp?: string; - path?: string; - correlationId?: string; - stack?: string; + statusCode: number; + message: string | string[]; + error?: string; + details?: ValidationErrorDetail[]; + timestamp?: string; + path?: string; + correlationId?: string; + stack?: string; } - export interface ValidationErrorDetail { - property: string; - constraints: string[] | Record; + property: string; + constraints: string[] | Record; } diff --git a/src/learning-paths/learning-paths.controller.ts b/src/learning-paths/learning-paths.controller.ts index 4467c4f5..daaf019a 100644 --- a/src/learning-paths/learning-paths.controller.ts +++ b/src/learning-paths/learning-paths.controller.ts @@ -1,12 +1,12 @@ import { Controller, Post, Body } from '@nestjs/common'; import { LearningPathsService } from './learning-paths.service'; - @Controller('learning-paths') export class LearningPathsController { - constructor(private readonly learningPathsService: LearningPathsService) {} - - @Post('generate') - generateLearningPath(@Body() payload: any) { - return this.learningPathsService.generateLearningPath(payload); - } + constructor(private readonly learningPathsService: LearningPathsService) { } + @Post('generate') + generateLearningPath( + @Body() + payload: unknown) { + return this.learningPathsService.generateLearningPath(payload); + } } diff --git a/src/learning-paths/learning-paths.module.ts b/src/learning-paths/learning-paths.module.ts index e28653ba..91a686ec 100644 --- a/src/learning-paths/learning-paths.module.ts +++ b/src/learning-paths/learning-paths.module.ts @@ -4,15 +4,15 @@ import { LearningPathsService } from './learning-paths.service'; import { SkillAssessmentService } from './services/skill-assessment.service'; import { PathGenerationService } from './services/path-generation.service'; import { MilestoneTrackingService } from './services/milestone-tracking.service'; - @Module({ - controllers: [LearningPathsController], - providers: [ - LearningPathsService, - SkillAssessmentService, - PathGenerationService, - MilestoneTrackingService, - ], - exports: [LearningPathsService], + controllers: [LearningPathsController], + providers: [ + LearningPathsService, + SkillAssessmentService, + PathGenerationService, + MilestoneTrackingService, + ], + exports: [LearningPathsService], }) -export class LearningPathsModule {} +export class LearningPathsModule { +} diff --git a/src/learning-paths/learning-paths.service.ts b/src/learning-paths/learning-paths.service.ts index ffef2d72..76b67319 100644 --- a/src/learning-paths/learning-paths.service.ts +++ b/src/learning-paths/learning-paths.service.ts @@ -2,18 +2,12 @@ import { Injectable } from '@nestjs/common'; import { SkillAssessmentService } from './services/skill-assessment.service'; import { PathGenerationService } from './services/path-generation.service'; import { MilestoneTrackingService } from './services/milestone-tracking.service'; - @Injectable() export class LearningPathsService { - constructor( - private readonly skillAssessmentService: SkillAssessmentService, - private readonly pathGenerationService: PathGenerationService, - private readonly milestoneTrackingService: MilestoneTrackingService, - ) {} - - generateLearningPath(input: any) { - const assessment = this.skillAssessmentService.assess(input); - const path = this.pathGenerationService.generate(assessment); - return this.milestoneTrackingService.initialize(path); - } + constructor(private readonly skillAssessmentService: SkillAssessmentService, private readonly pathGenerationService: PathGenerationService, private readonly milestoneTrackingService: MilestoneTrackingService) { } + generateLearningPath(input: unknown) { + const assessment = this.skillAssessmentService.assess(input); + const path = this.pathGenerationService.generate(assessment); + return this.milestoneTrackingService.initialize(path); + } } diff --git a/src/learning-paths/services/milestone-tracking.service.ts b/src/learning-paths/services/milestone-tracking.service.ts index aa01432d..10fd27bf 100644 --- a/src/learning-paths/services/milestone-tracking.service.ts +++ b/src/learning-paths/services/milestone-tracking.service.ts @@ -1,15 +1,14 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class MilestoneTrackingService { - initialize(path: any) { - return { - ...path, - milestones: path.milestones.map((title: string) => ({ - title, - status: 'pending', - progress: 0, - })), - }; - } + initialize(path: unknown) { + return { + ...path, + milestones: path.milestones.map((title: string) => ({ + title, + status: 'pending', + progress: 0, + })), + }; + } } diff --git a/src/learning-paths/services/path-generation.service.ts b/src/learning-paths/services/path-generation.service.ts index 9384dca3..d56df191 100644 --- a/src/learning-paths/services/path-generation.service.ts +++ b/src/learning-paths/services/path-generation.service.ts @@ -1,22 +1,21 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class PathGenerationService { - generate(assessment: { level: string; goal: string }) { - const milestones = []; - - if (assessment.level === 'beginner') { - milestones.push('Fundamentals'); + generate(assessment: { + level: string; + goal: string; + }) { + const milestones = []; + if (assessment.level === 'beginner') { + milestones.push('Fundamentals'); + } + if (assessment.goal === 'frontend') { + milestones.push('HTML', 'CSS', 'JavaScript', 'React'); + } + return { + goal: assessment.goal, + level: assessment.level, + milestones, + }; } - - if (assessment.goal === 'frontend') { - milestones.push('HTML', 'CSS', 'JavaScript', 'React'); - } - - return { - goal: assessment.goal, - level: assessment.level, - milestones, - }; - } } diff --git a/src/learning-paths/services/skill-assessment.service.ts b/src/learning-paths/services/skill-assessment.service.ts index 93f79c29..fb3d342f 100644 --- a/src/learning-paths/services/skill-assessment.service.ts +++ b/src/learning-paths/services/skill-assessment.service.ts @@ -1,11 +1,10 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class SkillAssessmentService { - assess(input: any) { - return { - level: input?.level ?? 'beginner', - goal: input?.goal ?? 'general', - }; - } + assess(input: unknown) { + return { + level: input?.level ?? 'beginner', + goal: input?.goal ?? 'general', + }; + } } diff --git a/src/localization/dto/bundle-query.dto.ts b/src/localization/dto/bundle-query.dto.ts index c74cc259..0c79d602 100644 --- a/src/localization/dto/bundle-query.dto.ts +++ b/src/localization/dto/bundle-query.dto.ts @@ -1,19 +1,17 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { IsNotEmpty, IsOptional, IsString, MaxLength } from 'class-validator'; - export class BundleQueryDto { - @ApiProperty({ example: 'errors' }) - @IsString() - @IsNotEmpty() - @MaxLength(128) - namespace: string; - - @ApiPropertyOptional({ - example: 'en', - description: 'If omitted, uses detected language from middleware / Accept-Language', - }) - @IsOptional() - @IsString() - @MaxLength(32) - locale?: string; + @ApiProperty({ example: 'errors' }) + @IsString() + @IsNotEmpty() + @MaxLength(128) + namespace: string; + @ApiPropertyOptional({ + example: 'en', + description: 'If omitted, uses detected language from middleware / Accept-Language', + }) + @IsOptional() + @IsString() + @MaxLength(32) + locale?: string; } diff --git a/src/localization/dto/create-translation.dto.ts b/src/localization/dto/create-translation.dto.ts index 36cc28b0..cb7b6a81 100644 --- a/src/localization/dto/create-translation.dto.ts +++ b/src/localization/dto/create-translation.dto.ts @@ -1,27 +1,23 @@ import { ApiProperty } from '@nestjs/swagger'; import { IsNotEmpty, IsString, MaxLength } from 'class-validator'; - export class CreateTranslationDto { - @ApiProperty({ example: 'errors' }) - @IsString() - @IsNotEmpty() - @MaxLength(128) - namespace: string; - - @ApiProperty({ example: 'not_found' }) - @IsString() - @IsNotEmpty() - @MaxLength(512) - key: string; - - @ApiProperty({ example: 'en' }) - @IsString() - @IsNotEmpty() - @MaxLength(32) - locale: string; - - @ApiProperty({ example: 'Resource was not found.' }) - @IsString() - @IsNotEmpty() - value: string; + @ApiProperty({ example: 'errors' }) + @IsString() + @IsNotEmpty() + @MaxLength(128) + namespace: string; + @ApiProperty({ example: 'not_found' }) + @IsString() + @IsNotEmpty() + @MaxLength(512) + key: string; + @ApiProperty({ example: 'en' }) + @IsString() + @IsNotEmpty() + @MaxLength(32) + locale: string; + @ApiProperty({ example: 'Resource was not found.' }) + @IsString() + @IsNotEmpty() + value: string; } diff --git a/src/localization/dto/export-query.dto.ts b/src/localization/dto/export-query.dto.ts index 6067cf4f..0f4071ce 100644 --- a/src/localization/dto/export-query.dto.ts +++ b/src/localization/dto/export-query.dto.ts @@ -1,22 +1,19 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { IsIn, IsNotEmpty, IsOptional, IsString, MaxLength } from 'class-validator'; - export class ExportQueryDto { - @ApiProperty({ example: 'errors' }) - @IsString() - @IsNotEmpty() - @MaxLength(128) - namespace: string; - - @ApiPropertyOptional({ example: 'en' }) - @IsOptional() - @IsString() - @MaxLength(32) - locale?: string; - - @ApiPropertyOptional({ enum: ['json', 'csv'], default: 'json' }) - @IsOptional() - @IsString() - @IsIn(['json', 'csv']) - format?: 'json' | 'csv' = 'json'; + @ApiProperty({ example: 'errors' }) + @IsString() + @IsNotEmpty() + @MaxLength(128) + namespace: string; + @ApiPropertyOptional({ example: 'en' }) + @IsOptional() + @IsString() + @MaxLength(32) + locale?: string; + @ApiPropertyOptional({ enum: ['json', 'csv'], default: 'json' }) + @IsOptional() + @IsString() + @IsIn(['json', 'csv']) + format?: 'json' | 'csv' = 'json'; } diff --git a/src/localization/dto/import-translations.dto.ts b/src/localization/dto/import-translations.dto.ts index ca39c269..7103cfa2 100644 --- a/src/localization/dto/import-translations.dto.ts +++ b/src/localization/dto/import-translations.dto.ts @@ -1,52 +1,39 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { Type } from 'class-transformer'; -import { - IsArray, - IsNotEmpty, - IsOptional, - IsString, - MaxLength, - ValidateNested, -} from 'class-validator'; - +import { IsArray, IsNotEmpty, IsOptional, IsString, MaxLength, ValidateNested, } from 'class-validator'; export class TranslationImportRowDto { - @ApiProperty({ example: 'errors' }) - @IsString() - @IsNotEmpty() - @MaxLength(128) - namespace: string; - - @ApiProperty({ example: 'not_found' }) - @IsString() - @IsNotEmpty() - @MaxLength(512) - key: string; - - @ApiProperty({ example: 'en' }) - @IsString() - @IsNotEmpty() - @MaxLength(32) - locale: string; - - @ApiProperty({ example: 'Not found.' }) - @IsString() - @IsNotEmpty() - value: string; + @ApiProperty({ example: 'errors' }) + @IsString() + @IsNotEmpty() + @MaxLength(128) + namespace: string; + @ApiProperty({ example: 'not_found' }) + @IsString() + @IsNotEmpty() + @MaxLength(512) + key: string; + @ApiProperty({ example: 'en' }) + @IsString() + @IsNotEmpty() + @MaxLength(32) + locale: string; + @ApiProperty({ example: 'Not found.' }) + @IsString() + @IsNotEmpty() + value: string; } - /** Body: `{ "translations": [...] }` or `{ "rows": [...] }` (alias). */ export class ImportTranslationsDto { - @ApiPropertyOptional({ type: [TranslationImportRowDto] }) - @IsOptional() - @IsArray() - @ValidateNested({ each: true }) - @Type(() => TranslationImportRowDto) - translations?: TranslationImportRowDto[]; - - @ApiPropertyOptional({ type: [TranslationImportRowDto], description: 'Alias of translations' }) - @IsOptional() - @IsArray() - @ValidateNested({ each: true }) - @Type(() => TranslationImportRowDto) - rows?: TranslationImportRowDto[]; + @ApiPropertyOptional({ type: [TranslationImportRowDto] }) + @IsOptional() + @IsArray() + @ValidateNested({ each: true }) + @Type(() => TranslationImportRowDto) + translations?: TranslationImportRowDto[]; + @ApiPropertyOptional({ type: [TranslationImportRowDto], description: 'Alias of translations' }) + @IsOptional() + @IsArray() + @ValidateNested({ each: true }) + @Type(() => TranslationImportRowDto) + rows?: TranslationImportRowDto[]; } diff --git a/src/localization/dto/list-translations-query.dto.ts b/src/localization/dto/list-translations-query.dto.ts index d22205d9..9ac61aa0 100644 --- a/src/localization/dto/list-translations-query.dto.ts +++ b/src/localization/dto/list-translations-query.dto.ts @@ -1,38 +1,33 @@ import { ApiPropertyOptional } from '@nestjs/swagger'; import { Type } from 'class-transformer'; import { IsInt, IsOptional, IsString, Max, MaxLength, Min } from 'class-validator'; - export class ListTranslationsQueryDto { - @ApiPropertyOptional({ example: 'errors' }) - @IsOptional() - @IsString() - @MaxLength(128) - namespace?: string; - - @ApiPropertyOptional({ example: 'en' }) - @IsOptional() - @IsString() - @MaxLength(32) - locale?: string; - - @ApiPropertyOptional({ description: 'Search substring in key or value' }) - @IsOptional() - @IsString() - @MaxLength(256) - search?: string; - - @ApiPropertyOptional({ default: 1 }) - @IsOptional() - @Type(() => Number) - @IsInt() - @Min(1) - page?: number = 1; - - @ApiPropertyOptional({ default: 20 }) - @IsOptional() - @Type(() => Number) - @IsInt() - @Min(1) - @Max(100) - limit?: number = 20; + @ApiPropertyOptional({ example: 'errors' }) + @IsOptional() + @IsString() + @MaxLength(128) + namespace?: string; + @ApiPropertyOptional({ example: 'en' }) + @IsOptional() + @IsString() + @MaxLength(32) + locale?: string; + @ApiPropertyOptional({ description: 'Search substring in key or value' }) + @IsOptional() + @IsString() + @MaxLength(256) + search?: string; + @ApiPropertyOptional({ default: 1 }) + @IsOptional() + @Type(() => Number) + @IsInt() + @Min(1) + page?: number = 1; + @ApiPropertyOptional({ default: 20 }) + @IsOptional() + @Type(() => Number) + @IsInt() + @Min(1) + @Max(100) + limit?: number = 20; } diff --git a/src/localization/dto/update-translation.dto.ts b/src/localization/dto/update-translation.dto.ts index c78caa78..72f091c1 100644 --- a/src/localization/dto/update-translation.dto.ts +++ b/src/localization/dto/update-translation.dto.ts @@ -1,27 +1,23 @@ import { ApiPropertyOptional } from '@nestjs/swagger'; import { IsOptional, IsString, MaxLength } from 'class-validator'; - export class UpdateTranslationDto { - @ApiPropertyOptional({ example: 'errors' }) - @IsOptional() - @IsString() - @MaxLength(128) - namespace?: string; - - @ApiPropertyOptional({ example: 'not_found' }) - @IsOptional() - @IsString() - @MaxLength(512) - key?: string; - - @ApiPropertyOptional({ example: 'en' }) - @IsOptional() - @IsString() - @MaxLength(32) - locale?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsString() - value?: string; + @ApiPropertyOptional({ example: 'errors' }) + @IsOptional() + @IsString() + @MaxLength(128) + namespace?: string; + @ApiPropertyOptional({ example: 'not_found' }) + @IsOptional() + @IsString() + @MaxLength(512) + key?: string; + @ApiPropertyOptional({ example: 'en' }) + @IsOptional() + @IsString() + @MaxLength(32) + locale?: string; + @ApiPropertyOptional() + @IsOptional() + @IsString() + value?: string; } diff --git a/src/localization/entities/translation.entity.ts b/src/localization/entities/translation.entity.ts index d9929398..c28805ec 100644 --- a/src/localization/entities/translation.entity.ts +++ b/src/localization/entities/translation.entity.ts @@ -1,39 +1,22 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - Index, - Unique, -} from 'typeorm'; - +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, Index, Unique, } from 'typeorm'; @Entity('translations') @Unique(['namespace', 'translationKey', 'locale']) @Index('IDX_translations_namespace_locale', ['namespace', 'locale']) export class Translation { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - namespace: string; - - @Column({ name: 'key' }) - translationKey: string; - - @Column() - locale: string; - - @Column({ type: 'text' }) - value: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + namespace: string; + @Column({ name: 'key' }) + translationKey: string; + @Column() + locale: string; + @Column({ type: 'text' }) + value: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/localization/language-detection.service.spec.ts b/src/localization/language-detection.service.spec.ts index cd369074..3c91fe06 100644 --- a/src/localization/language-detection.service.spec.ts +++ b/src/localization/language-detection.service.spec.ts @@ -1,82 +1,70 @@ import { Test, TestingModule } from '@nestjs/testing'; import { ConfigService } from '@nestjs/config'; import { LanguageDetectionService } from './language-detection.service'; - function mockReq(headers: Record = {}, query?: Record) { - return { - headers, - query: query ?? {}, - } as import('express').Request; + return { + headers, + query: query ?? {}, + } as import('express').Request; } - describe('LanguageDetectionService', () => { - let service: LanguageDetectionService; - - async function createModule(config: Record) { - const module: TestingModule = await Test.createTestingModule({ - providers: [ - LanguageDetectionService, - { - provide: ConfigService, - useValue: { - get: (key: string) => config[key], - }, - }, - ], - }).compile(); - return module.get(LanguageDetectionService); - } - - beforeEach(async () => { - service = await createModule({ - I18N_DEFAULT_LOCALE: 'en', - I18N_SUPPORTED_LOCALES: 'en,fr,de', + let service: LanguageDetectionService; + async function createModule(config: Record) { + const module: TestingModule = await Test.createTestingModule({ + providers: [ + LanguageDetectionService, + { + provide: ConfigService, + useValue: { + get: (key: string) => config[key], + }, + }, + ], + }).compile(); + return module.get(LanguageDetectionService); + } + beforeEach(async () => { + service = await createModule({ + I18N_DEFAULT_LOCALE: 'en', + I18N_SUPPORTED_LOCALES: 'en,fr,de', + }); }); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('pickSupported', () => { - it('returns exact tag when listed', () => { - expect(service.pickSupported('fr')).toBe('fr'); - expect(service.pickSupported('FR')).toBe('fr'); + it('should be defined', () => { + expect(service).toBeDefined(); }); - - it('maps region subtag to primary when primary is supported', () => { - expect(service.pickSupported('fr-CA')).toBe('fr'); + describe('pickSupported', () => { + it('returns exact tag when listed', () => { + expect(service.pickSupported('fr')).toBe('fr'); + expect(service.pickSupported('FR')).toBe('fr'); + }); + it('maps region subtag to primary when primary is supported', () => { + expect(service.pickSupported('fr-CA')).toBe('fr'); + }); + it('returns null when nothing matches', async () => { + const s = await createModule({ I18N_DEFAULT_LOCALE: 'en', I18N_SUPPORTED_LOCALES: 'en' }); + expect(s.pickSupported('zz')).toBeNull(); + }); }); - - it('returns null when nothing matches', async () => { - const s = await createModule({ I18N_DEFAULT_LOCALE: 'en', I18N_SUPPORTED_LOCALES: 'en' }); - expect(s.pickSupported('zz')).toBeNull(); + describe('resolveWithSource', () => { + it('uses query lang when supported', () => { + const req = mockReq({ 'accept-language': 'en-US,en;q=0.9' }); + expect(service.resolveWithSource(req, 'de')).toEqual({ locale: 'de', source: 'query' }); + }); + it('ignores unsupported query lang and falls through to header', () => { + const req = mockReq({ 'accept-language': 'fr-FR,fr;q=0.9,en;q=0.8' }); + expect(service.resolveWithSource(req, 'xx')).toEqual({ locale: 'fr', source: 'header' }); + }); + it('respects q-values on Accept-Language', () => { + const req = mockReq({ 'accept-language': 'de;q=0.8,fr;q=0.9' }); + expect(service.resolveWithSource(req)).toEqual({ locale: 'fr', source: 'header' }); + }); + it('uses default when header missing or unmatched', async () => { + const s = await createModule({ I18N_DEFAULT_LOCALE: 'en', I18N_SUPPORTED_LOCALES: 'en' }); + expect(s.resolveWithSource(mockReq({}))).toEqual({ locale: 'en', source: 'default' }); + expect(s.resolveWithSource(mockReq({ 'accept-language': 'ja,zh-CN;q=0.9' }))).toEqual({ + locale: 'en', + source: 'default', + }); + }); }); - }); - - describe('resolveWithSource', () => { - it('uses query lang when supported', () => { - const req = mockReq({ 'accept-language': 'en-US,en;q=0.9' }); - expect(service.resolveWithSource(req, 'de')).toEqual({ locale: 'de', source: 'query' }); - }); - - it('ignores unsupported query lang and falls through to header', () => { - const req = mockReq({ 'accept-language': 'fr-FR,fr;q=0.9,en;q=0.8' }); - expect(service.resolveWithSource(req, 'xx')).toEqual({ locale: 'fr', source: 'header' }); - }); - - it('respects q-values on Accept-Language', () => { - const req = mockReq({ 'accept-language': 'de;q=0.8,fr;q=0.9' }); - expect(service.resolveWithSource(req)).toEqual({ locale: 'fr', source: 'header' }); - }); - - it('uses default when header missing or unmatched', async () => { - const s = await createModule({ I18N_DEFAULT_LOCALE: 'en', I18N_SUPPORTED_LOCALES: 'en' }); - expect(s.resolveWithSource(mockReq({}))).toEqual({ locale: 'en', source: 'default' }); - expect(s.resolveWithSource(mockReq({ 'accept-language': 'ja,zh-CN;q=0.9' }))).toEqual({ - locale: 'en', - source: 'default', - }); - }); - }); }); diff --git a/src/localization/language-detection.service.ts b/src/localization/language-detection.service.ts index 92fccba3..67821a48 100644 --- a/src/localization/language-detection.service.ts +++ b/src/localization/language-detection.service.ts @@ -1,104 +1,100 @@ import { Injectable } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import { Request } from 'express'; - export type LocaleResolutionSource = 'query' | 'header' | 'default'; - export interface ResolvedLocale { - locale: string; - source: LocaleResolutionSource; + locale: string; + source: LocaleResolutionSource; } - @Injectable() export class LanguageDetectionService { - constructor(private readonly configService: ConfigService) {} - - getDefaultLocale(): string { - return this.configService.get('I18N_DEFAULT_LOCALE')?.trim().toLowerCase() || 'en'; - } - - getSupportedLocales(): string[] { - const raw = - this.configService.get('I18N_SUPPORTED_LOCALES') || - this.configService.get('I18N_DEFAULT_LOCALE') || - 'en'; - return raw - .split(',') - .map((s) => this.normalizeLocaleTag(s)) - .filter(Boolean); - } - - normalizeLocaleTag(tag: string): string { - if (!tag) return ''; - return tag.trim().toLowerCase().replace(/_/g, '-'); - } - - /** First supported locale that matches the tag (exact or primary subtag). */ - pickSupported(tag: string): string | null { - const normalized = this.normalizeLocaleTag(tag); - if (!normalized) return null; - const supported = this.getSupportedLocales(); - if (supported.includes(normalized)) return normalized; - const primary = normalized.split('-')[0]; - if (supported.includes(primary)) return primary; - for (const s of supported) { - if (s.startsWith(`${primary}-`) || primary === s.split('-')[0]) { - return s; - } + constructor(private readonly configService: ConfigService) { } + getDefaultLocale(): string { + return this.configService.get('I18N_DEFAULT_LOCALE')?.trim().toLowerCase() || 'en'; } - return null; - } - - resolveWithSource(req: Request, queryLang?: string): ResolvedLocale { - const defaultLocale = this.getDefaultLocale(); - - if (queryLang) { - const picked = this.pickSupported(queryLang); - if (picked) { - return { locale: picked, source: 'query' }; - } + getSupportedLocales(): string[] { + const raw = this.configService.get('I18N_SUPPORTED_LOCALES') || + this.configService.get('I18N_DEFAULT_LOCALE') || + 'en'; + return raw + .split(',') + .map((s) => this.normalizeLocaleTag(s)) + .filter(Boolean); } - - const header = req.headers['accept-language']; - if (header && typeof header === 'string') { - const fromHeader = this.parseAcceptLanguage(header); - if (fromHeader) { - return { locale: fromHeader, source: 'header' }; - } + normalizeLocaleTag(tag: string): string { + if (!tag) + return ''; + return tag.trim().toLowerCase().replace(/_/g, '-'); } - - return { locale: defaultLocale, source: 'default' }; - } - - resolveLocale(req: Request, queryLang?: string): string { - return this.resolveWithSource(req, queryLang).locale; - } - - /** - * RFC 7231-style Accept-Language: pick highest-q language that we support. - */ - private parseAcceptLanguage(header: string): string | null { - const parts = header.split(','); - const candidates: Array<{ tag: string; q: number }> = []; - for (const part of parts) { - const [langPart, ...params] = part.trim().split(';'); - const tag = this.normalizeLocaleTag(langPart); - if (!tag || tag === '*') continue; - let q = 1; - for (const p of params) { - const [k, v] = p.trim().split('='); - if (k?.toLowerCase() === 'q' && v) { - const n = parseFloat(v); - if (!Number.isNaN(n)) q = n; + /** First supported locale that matches the tag (exact or primary subtag). */ + pickSupported(tag: string): string | null { + const normalized = this.normalizeLocaleTag(tag); + if (!normalized) + return null; + const supported = this.getSupportedLocales(); + if (supported.includes(normalized)) + return normalized; + const primary = normalized.split('-')[0]; + if (supported.includes(primary)) + return primary; + for (const s of supported) { + if (s.startsWith(`${primary}-`) || primary === s.split('-')[0]) { + return s; + } } - } - candidates.push({ tag, q }); + return null; } - candidates.sort((a, b) => b.q - a.q); - for (const { tag } of candidates) { - const picked = this.pickSupported(tag); - if (picked) return picked; + resolveWithSource(req: Request, queryLang?: string): ResolvedLocale { + const defaultLocale = this.getDefaultLocale(); + if (queryLang) { + const picked = this.pickSupported(queryLang); + if (picked) { + return { locale: picked, source: 'query' }; + } + } + const header = req.headers['accept-language']; + if (header && typeof header === 'string') { + const fromHeader = this.parseAcceptLanguage(header); + if (fromHeader) { + return { locale: fromHeader, source: 'header' }; + } + } + return { locale: defaultLocale, source: 'default' }; + } + resolveLocale(req: Request, queryLang?: string): string { + return this.resolveWithSource(req, queryLang).locale; + } + /** + * RFC 7231-style Accept-Language: pick highest-q language that we support. + */ + private parseAcceptLanguage(header: string): string | null { + const parts = header.split(','); + const candidates: Array<{ + tag: string; + q: number; + }> = []; + for (const part of parts) { + const [langPart, ...params] = part.trim().split(';'); + const tag = this.normalizeLocaleTag(langPart); + if (!tag || tag === '*') + continue; + let q = 1; + for (const p of params) { + const [k, v] = p.trim().split('='); + if (k?.toLowerCase() === 'q' && v) { + const n = parseFloat(v); + if (!Number.isNaN(n)) + q = n; + } + } + candidates.push({ tag, q }); + } + candidates.sort((a, b) => b.q - a.q); + for (const { tag } of candidates) { + const picked = this.pickSupported(tag); + if (picked) + return picked; + } + return null; } - return null; - } } diff --git a/src/localization/language.middleware.ts b/src/localization/language.middleware.ts index cefa5972..cefa0fac 100644 --- a/src/localization/language.middleware.ts +++ b/src/localization/language.middleware.ts @@ -2,20 +2,17 @@ import { Injectable, NestMiddleware } from '@nestjs/common'; import { NextFunction, Response } from 'express'; import { RequestWithLocale } from '../common/types/request-with-locale'; import { LanguageDetectionService } from './language-detection.service'; - @Injectable() export class LanguageMiddleware implements NestMiddleware { - constructor(private readonly languageDetection: LanguageDetectionService) {} - - use(req: RequestWithLocale, _res: Response, next: NextFunction): void { - const raw = req.query?.lang; - const queryLang = - typeof raw === 'string' - ? raw - : Array.isArray(raw) && typeof raw[0] === 'string' - ? raw[0] - : undefined; - req.resolvedLocale = this.languageDetection.resolveLocale(req, queryLang); - next(); - } + constructor(private readonly languageDetection: LanguageDetectionService) { } + use(req: RequestWithLocale, _res: Response, next: NextFunction): void { + const raw = req.query?.lang; + const queryLang = typeof raw === 'string' + ? raw + : Array.isArray(raw) && typeof raw[0] === 'string' + ? raw[0] + : undefined; + req.resolvedLocale = this.languageDetection.resolveLocale(req, queryLang); + next(); + } } diff --git a/src/localization/localization.constants.ts b/src/localization/localization.constants.ts index 8bc56c40..19f08a91 100644 --- a/src/localization/localization.constants.ts +++ b/src/localization/localization.constants.ts @@ -1,5 +1,4 @@ export const I18N_CACHE_KEY_PREFIX = 'i18n:ns:'; - export function bundleCacheKey(namespace: string, locale: string): string { - return `${I18N_CACHE_KEY_PREFIX}${namespace}:${locale}`; + return `${I18N_CACHE_KEY_PREFIX}${namespace}:${locale}`; } diff --git a/src/localization/localization.controller.ts b/src/localization/localization.controller.ts index 7455ce06..e62acf0c 100644 --- a/src/localization/localization.controller.ts +++ b/src/localization/localization.controller.ts @@ -1,17 +1,4 @@ -import { - BadRequestException, - Body, - Controller, - Delete, - Get, - Param, - Patch, - Post, - Query, - Req, - Res, - UseGuards, -} from '@nestjs/common'; +import { BadRequestException, Body, Controller, Delete, Get, Param, Patch, Post, Query, Req, Res, UseGuards, } from '@nestjs/common'; import { ApiBearerAuth, ApiOperation, ApiProduces, ApiQuery, ApiTags } from '@nestjs/swagger'; import { Response } from 'express'; import { RequestWithLocale } from '../common/types/request-with-locale'; @@ -27,107 +14,116 @@ import { ListTranslationsQueryDto } from './dto/list-translations-query.dto'; import { UpdateTranslationDto } from './dto/update-translation.dto'; import { LanguageDetectionService } from './language-detection.service'; import { LocalizationService } from './localization.service'; - @ApiTags('localization') @Controller('localization') export class LocalizationController { - constructor( - private readonly localizationService: LocalizationService, - private readonly languageDetection: LanguageDetectionService, - ) {} - - @Get('bundle') - @ApiOperation({ summary: 'Get merged translation bundle for a namespace' }) - async getBundle(@Query() query: BundleQueryDto, @Req() req: RequestWithLocale) { - const locale = - query.locale?.trim() || req.resolvedLocale || this.languageDetection.getDefaultLocale(); - return this.localizationService.getBundleForApi(query.namespace, locale); - } - - @Get('detect') - @ApiOperation({ summary: 'Show how the request locale was resolved' }) - @ApiQuery({ name: 'lang', required: false }) - detect(@Req() req: RequestWithLocale, @Query('lang') lang?: string) { - return this.languageDetection.resolveWithSource(req, lang); - } - - @Get('admin/translations') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'List translations (admin)' }) - listAdmin(@Query() query: ListTranslationsQueryDto) { - return this.localizationService.findAll(query); - } - - @Get('admin/translations/:id') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Get one translation (admin)' }) - findOne(@Param('id') id: string) { - return this.localizationService.findOne(id); - } - - @Post('admin/translations') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Create translation (admin)' }) - create(@Body() dto: CreateTranslationDto) { - return this.localizationService.create(dto); - } - - @Patch('admin/translations/:id') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Update translation (admin)' }) - update(@Param('id') id: string, @Body() dto: UpdateTranslationDto) { - return this.localizationService.update(id, dto); - } - - @Delete('admin/translations/:id') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Delete translation (admin)' }) - async remove(@Param('id') id: string) { - await this.localizationService.remove(id); - return { deleted: true }; - } - - @Post('admin/import') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Upsert translations from JSON (admin)' }) - import(@Body() body: ImportTranslationsDto) { - const rows = body.translations ?? body.rows; - if (!rows?.length) { - throw new BadRequestException('Provide translations or rows array with at least one item'); + constructor(private readonly localizationService: LocalizationService, private readonly languageDetection: LanguageDetectionService) { } + @Get('bundle') + @ApiOperation({ summary: 'Get merged translation bundle for a namespace' }) + async getBundle( + @Query() + query: BundleQueryDto, + @Req() + req: RequestWithLocale) { + const locale = query.locale?.trim() || req.resolvedLocale || this.languageDetection.getDefaultLocale(); + return this.localizationService.getBundleForApi(query.namespace, locale); } - return this.localizationService.importRows(rows); - } - - @Get('admin/export') - @UseGuards(JwtAuthGuard, RolesGuard) - @Roles(UserRole.ADMIN) - @ApiBearerAuth() - @ApiOperation({ summary: 'Export translations as JSON or CSV (admin)' }) - @ApiProduces('application/json', 'text/csv') - async export(@Query() query: ExportQueryDto, @Res({ passthrough: true }) res: Response) { - const rows = await this.localizationService.exportRows(query.namespace, query.locale); - const format = query.format ?? 'json'; - if (format === 'csv') { - const csv = LocalizationService.toCsv(rows); - res.setHeader('Content-Type', 'text/csv; charset=utf-8'); - res.setHeader( - 'Content-Disposition', - `attachment; filename="translations-${query.namespace}.csv"`, - ); - return csv; + @Get('detect') + @ApiOperation({ summary: 'Show how the request locale was resolved' }) + @ApiQuery({ name: 'lang', required: false }) + detect( + @Req() + req: RequestWithLocale, + @Query('lang') + lang?: string) { + return this.languageDetection.resolveWithSource(req, lang); + } + @Get('admin/translations') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'List translations (admin)' }) + listAdmin( + @Query() + query: ListTranslationsQueryDto) { + return this.localizationService.findAll(query); + } + @Get('admin/translations/:id') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Get one translation (admin)' }) + findOne( + @Param('id') + id: string) { + return this.localizationService.findOne(id); + } + @Post('admin/translations') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Create translation (admin)' }) + create( + @Body() + dto: CreateTranslationDto) { + return this.localizationService.create(dto); + } + @Patch('admin/translations/:id') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Update translation (admin)' }) + update( + @Param('id') + id: string, + @Body() + dto: UpdateTranslationDto) { + return this.localizationService.update(id, dto); + } + @Delete('admin/translations/:id') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Delete translation (admin)' }) + async remove( + @Param('id') + id: string) { + await this.localizationService.remove(id); + return { deleted: true }; + } + @Post('admin/import') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Upsert translations from JSON (admin)' }) + import( + @Body() + body: ImportTranslationsDto) { + const rows = body.translations ?? body.rows; + if (!rows?.length) { + throw new BadRequestException('Provide translations or rows array with at least one item'); + } + return this.localizationService.importRows(rows); + } + @Get('admin/export') + @UseGuards(JwtAuthGuard, RolesGuard) + @Roles(UserRole.ADMIN) + @ApiBearerAuth() + @ApiOperation({ summary: 'Export translations as JSON or CSV (admin)' }) + @ApiProduces('application/json', 'text/csv') + async export( + @Query() + query: ExportQueryDto, + @Res({ passthrough: true }) + res: Response) { + const rows = await this.localizationService.exportRows(query.namespace, query.locale); + const format = query.format ?? 'json'; + if (format === 'csv') { + const csv = LocalizationService.toCsv(rows); + res.setHeader('Content-Type', 'text/csv; charset=utf-8'); + res.setHeader('Content-Disposition', `attachment; filename="translations-${query.namespace}.csv"`); + return csv; + } + return rows; } - return rows; - } } diff --git a/src/localization/localization.module.ts b/src/localization/localization.module.ts index bc27f392..f80a76c9 100644 --- a/src/localization/localization.module.ts +++ b/src/localization/localization.module.ts @@ -7,21 +7,20 @@ import { LanguageDetectionService } from './language-detection.service'; import { LanguageMiddleware } from './language.middleware'; import { LocalizationController } from './localization.controller'; import { LocalizationService } from './localization.service'; - @Module({ - imports: [TypeOrmModule.forFeature([Translation])], - controllers: [LocalizationController], - providers: [ - LocalizationService, - LanguageDetectionService, - LanguageMiddleware, - JwtAuthGuard, - RolesGuard, - ], - exports: [LocalizationService, LanguageDetectionService], + imports: [TypeOrmModule.forFeature([Translation])], + controllers: [LocalizationController], + providers: [ + LocalizationService, + LanguageDetectionService, + LanguageMiddleware, + JwtAuthGuard, + RolesGuard, + ], + exports: [LocalizationService, LanguageDetectionService], }) export class LocalizationModule implements NestModule { - configure(consumer: MiddlewareConsumer) { - consumer.apply(LanguageMiddleware).forRoutes(LocalizationController); - } + configure(consumer: MiddlewareConsumer) { + consumer.apply(LanguageMiddleware).forRoutes(LocalizationController); + } } diff --git a/src/localization/localization.service.spec.ts b/src/localization/localization.service.spec.ts index fd2d26bf..25a2d627 100644 --- a/src/localization/localization.service.spec.ts +++ b/src/localization/localization.service.spec.ts @@ -6,156 +6,150 @@ import { LocalizationService } from './localization.service'; import { LanguageDetectionService } from './language-detection.service'; import { Translation } from './entities/translation.entity'; import { bundleCacheKey } from './localization.constants'; - describe('LocalizationService', () => { - let service: LocalizationService; - let cacheManager: { get: jest.Mock; set: jest.Mock; del: jest.Mock }; - let translationRepo: { - find: jest.Mock; - findOne: jest.Mock; - save: jest.Mock; - remove: jest.Mock; - create: jest.Mock; - upsert: jest.Mock; - createQueryBuilder: jest.Mock; - }; - - beforeEach(async () => { - cacheManager = { - get: jest.fn().mockResolvedValue(undefined), - set: jest.fn().mockResolvedValue(undefined), - del: jest.fn().mockResolvedValue(undefined), + let service: LocalizationService; + let cacheManager: { + get: jest.Mock; + set: jest.Mock; + del: jest.Mock; }; - - translationRepo = { - find: jest.fn(), - findOne: jest.fn(), - save: jest.fn(), - remove: jest.fn(), - create: jest.fn((x) => ({ ...x, id: 'new-id' })), - upsert: jest.fn().mockResolvedValue(undefined), - createQueryBuilder: jest.fn(), + let translationRepo: { + find: jest.Mock; + findOne: jest.Mock; + save: jest.Mock; + remove: jest.Mock; + create: jest.Mock; + upsert: jest.Mock; + createQueryBuilder: jest.Mock; }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - LocalizationService, - LanguageDetectionService, - { - provide: ConfigService, - useValue: { - get: (key: string) => { - if (key === 'I18N_DEFAULT_LOCALE') return 'en'; - if (key === 'I18N_SUPPORTED_LOCALES') return 'en,fr'; - if (key === 'I18N_CACHE_TTL_SECONDS') return '300'; - return undefined; - }, - }, - }, - { provide: CACHE_MANAGER, useValue: cacheManager }, - { provide: getRepositoryToken(Translation), useValue: translationRepo }, - ], - }).compile(); - - service = module.get(LocalizationService); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('interpolate', () => { - it('replaces {{name}} placeholders', () => { - expect(service.interpolate('Hello {{name}}', { name: 'Ada' })).toBe('Hello Ada'); + beforeEach(async () => { + cacheManager = { + get: jest.fn().mockResolvedValue(undefined), + set: jest.fn().mockResolvedValue(undefined), + del: jest.fn().mockResolvedValue(undefined), + }; + translationRepo = { + find: jest.fn(), + findOne: jest.fn(), + save: jest.fn(), + remove: jest.fn(), + create: jest.fn((x) => ({ ...x, id: 'new-id' })), + upsert: jest.fn().mockResolvedValue(undefined), + createQueryBuilder: jest.fn(), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + LocalizationService, + LanguageDetectionService, + { + provide: ConfigService, + useValue: { + get: (key: string) => { + if (key === 'I18N_DEFAULT_LOCALE') + return 'en'; + if (key === 'I18N_SUPPORTED_LOCALES') + return 'en,fr'; + if (key === 'I18N_CACHE_TTL_SECONDS') + return '300'; + return undefined; + }, + }, + }, + { provide: CACHE_MANAGER, useValue: cacheManager }, + { provide: getRepositoryToken(Translation), useValue: translationRepo }, + ], + }).compile(); + service = module.get(LocalizationService); }); - - it('leaves unknown tokens', () => { - expect(service.interpolate('{{a}}', {})).toBe('{{a}}'); + it('should be defined', () => { + expect(service).toBeDefined(); }); - }); - - describe('translate', () => { - beforeEach(() => { - translationRepo.find.mockImplementation( - (opts: { where: { namespace: string; locale: string } }) => { - const { namespace, locale } = opts.where; - if (namespace === 'app' && locale === 'fr') { - return Promise.resolve([{ translationKey: 'b', value: 'deux' }]); - } - if (namespace === 'app' && locale === 'en') { - return Promise.resolve([ - { translationKey: 'a', value: 'one' }, - { translationKey: 'b', value: 'two' }, - { translationKey: 'greet', value: 'Hi {{name}}' }, - ]); - } - return Promise.resolve([]); - }, - ); - }); - - it('interpolates variables', async () => { - const t = await service.translate('app', 'greet', 'en', { name: 'Bo' }); - expect(t).toBe('Hi Bo'); - }); - - it('falls back to default locale for missing key in requested locale', async () => { - const t = await service.translate('app', 'a', 'fr'); - expect(t).toBe('one'); + describe('interpolate', () => { + it('replaces {{name}} placeholders', () => { + expect(service.interpolate('Hello {{name}}', { name: 'Ada' })).toBe('Hello Ada'); + }); + it('leaves unknown tokens', () => { + expect(service.interpolate('{{a}}', {})).toBe('{{a}}'); + }); }); - - it('uses primary locale when key exists there', async () => { - const t = await service.translate('app', 'b', 'fr'); - expect(t).toBe('deux'); + describe('translate', () => { + beforeEach(() => { + translationRepo.find.mockImplementation((opts: { + where: { + namespace: string; + locale: string; + }; + }) => { + const { namespace, locale } = opts.where; + if (namespace === 'app' && locale === 'fr') { + return Promise.resolve([{ translationKey: 'b', value: 'deux' }]); + } + if (namespace === 'app' && locale === 'en') { + return Promise.resolve([ + { translationKey: 'a', value: 'one' }, + { translationKey: 'b', value: 'two' }, + { translationKey: 'greet', value: 'Hi {{name}}' }, + ]); + } + return Promise.resolve([]); + }); + }); + it('interpolates variables', async () => { + const t = await service.translate('app', 'greet', 'en', { name: 'Bo' }); + expect(t).toBe('Hi Bo'); + }); + it('falls back to default locale for missing key in requested locale', async () => { + const t = await service.translate('app', 'a', 'fr'); + expect(t).toBe('one'); + }); + it('uses primary locale when key exists there', async () => { + const t = await service.translate('app', 'b', 'fr'); + expect(t).toBe('deux'); + }); + it('returns namespace.key when missing everywhere', async () => { + const t = await service.translate('app', 'missing', 'en'); + expect(t).toBe('app.missing'); + }); }); - - it('returns namespace.key when missing everywhere', async () => { - const t = await service.translate('app', 'missing', 'en'); - expect(t).toBe('app.missing'); - }); - }); - - describe('invalidateBundles', () => { - it('calls cache del for unique namespace/locale pairs', async () => { - await service.invalidateBundles([ - { namespace: 'n', locale: 'en' }, - { namespace: 'n', locale: 'en' }, - { namespace: 'n', locale: 'fr' }, - ]); - expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('n', 'en')); - expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('n', 'fr')); - expect(cacheManager.del).toHaveBeenCalledTimes(2); + describe('invalidateBundles', () => { + it('calls cache del for unique namespace/locale pairs', async () => { + await service.invalidateBundles([ + { namespace: 'n', locale: 'en' }, + { namespace: 'n', locale: 'en' }, + { namespace: 'n', locale: 'fr' }, + ]); + expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('n', 'en')); + expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('n', 'fr')); + expect(cacheManager.del).toHaveBeenCalledTimes(2); + }); }); - }); - - describe('create', () => { - it('invalidates bundle cache after save', async () => { - const saved = { - id: '1', - namespace: 'errors', - translationKey: 'x', - locale: 'en', - value: 'v', - createdAt: new Date(), - updatedAt: new Date(), - }; - translationRepo.save.mockResolvedValue(saved); - await service.create({ - namespace: 'errors', - key: 'x', - locale: 'en', - value: 'v', - }); - expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('errors', 'en')); + describe('create', () => { + it('invalidates bundle cache after save', async () => { + const saved = { + id: '1', + namespace: 'errors', + translationKey: 'x', + locale: 'en', + value: 'v', + createdAt: new Date(), + updatedAt: new Date(), + }; + translationRepo.save.mockResolvedValue(saved); + await service.create({ + namespace: 'errors', + key: 'x', + locale: 'en', + value: 'v', + }); + expect(cacheManager.del).toHaveBeenCalledWith(bundleCacheKey('errors', 'en')); + }); }); - }); - - describe('toCsv', () => { - it('escapes commas and quotes', () => { - const csv = LocalizationService.toCsv([ - { namespace: 'n', key: 'k', locale: 'en', value: 'say "hi", ok' }, - ]); - expect(csv).toContain('"say ""hi"", ok"'); + describe('toCsv', () => { + it('escapes commas and quotes', () => { + const csv = LocalizationService.toCsv([ + { namespace: 'n', key: 'k', locale: 'en', value: 'say "hi", ok' }, + ]); + expect(csv).toContain('"say ""hi"", ok"'); + }); }); - }); }); diff --git a/src/localization/localization.service.ts b/src/localization/localization.service.ts index 80870ac3..cbb1f565 100644 --- a/src/localization/localization.service.ts +++ b/src/localization/localization.service.ts @@ -1,11 +1,5 @@ import { CACHE_MANAGER } from '@nestjs/cache-manager'; -import { - BadRequestException, - ConflictException, - Inject, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { BadRequestException, ConflictException, Inject, Injectable, NotFoundException, } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import { InjectRepository } from '@nestjs/typeorm'; import { Cache } from 'cache-manager'; @@ -17,328 +11,300 @@ import { UpdateTranslationDto } from './dto/update-translation.dto'; import { Translation } from './entities/translation.entity'; import { bundleCacheKey } from './localization.constants'; import { LanguageDetectionService } from './language-detection.service'; - export interface TranslationListItemDto { - id: string; - namespace: string; - key: string; - locale: string; - value: string; - createdAt: Date; - updatedAt: Date; + id: string; + namespace: string; + key: string; + locale: string; + value: string; + createdAt: Date; + updatedAt: Date; } - export interface PaginatedTranslations { - items: TranslationListItemDto[]; - total: number; - page: number; - limit: number; + items: TranslationListItemDto[]; + total: number; + page: number; + limit: number; } - @Injectable() export class LocalizationService { - constructor( + constructor( @InjectRepository(Translation) - private readonly translationRepo: Repository, - @Inject(CACHE_MANAGER) private readonly cacheManager: Cache, - private readonly configService: ConfigService, - private readonly languageDetection: LanguageDetectionService, - ) {} - - private getCacheTtlMs(): number { - const sec = parseInt(this.configService.get('I18N_CACHE_TTL_SECONDS') || '300', 10); - return Math.max(0, sec) * 1000; - } - - private getDefaultLocale(): string { - return this.languageDetection.getDefaultLocale(); - } - - private toItem(entity: Translation): TranslationListItemDto { - return { - id: entity.id, - namespace: entity.namespace, - key: entity.translationKey, - locale: entity.locale, - value: entity.value, - createdAt: entity.createdAt, - updatedAt: entity.updatedAt, - }; - } - - async invalidateBundles(pairs: Array<{ namespace: string; locale: string }>): Promise { - const seen = new Set(); - for (const { namespace, locale } of pairs) { - const k = `${namespace}\0${locale}`; - if (seen.has(k)) continue; - seen.add(k); - await this.cacheManager.del(bundleCacheKey(namespace, locale)); + private readonly translationRepo: Repository, + @Inject(CACHE_MANAGER) + private readonly cacheManager: Cache, private readonly configService: ConfigService, private readonly languageDetection: LanguageDetectionService) { } + private getCacheTtlMs(): number { + const sec = parseInt(this.configService.get('I18N_CACHE_TTL_SECONDS') || '300', 10); + return Math.max(0, sec) * 1000; } - } - - private async loadRawBundleFromDb( - namespace: string, - locale: string, - ): Promise> { - const rows = await this.translationRepo.find({ - where: { namespace, locale: languageDetectionNormalize(locale) }, - select: ['translationKey', 'value'], - }); - const map: Record = {}; - for (const r of rows) { - map[r.translationKey] = r.value; + private getDefaultLocale(): string { + return this.languageDetection.getDefaultLocale(); } - return map; - } - - async getRawBundleCached(namespace: string, locale: string): Promise> { - const loc = languageDetectionNormalize(locale); - const key = bundleCacheKey(namespace, loc); - const ttl = this.getCacheTtlMs(); - const cached = await this.cacheManager.get>(key); - if (cached) return cached; - const fresh = await this.loadRawBundleFromDb(namespace, loc); - if (ttl > 0) { - await this.cacheManager.set(key, fresh, ttl); - } else { - await this.cacheManager.set(key, fresh); + private toItem(entity: Translation): TranslationListItemDto { + return { + id: entity.id, + namespace: entity.namespace, + key: entity.translationKey, + locale: entity.locale, + value: entity.value, + createdAt: entity.createdAt, + updatedAt: entity.updatedAt, + }; } - return fresh; - } - - async getMessagesMerged(namespace: string, locale: string): Promise> { - const loc = this.languageDetection.pickSupported(locale) || this.getDefaultLocale(); - const primary = await this.getRawBundleCached(namespace, loc); - const def = this.getDefaultLocale(); - if (loc === def) { - return { ...primary }; + async invalidateBundles(pairs: Array<{ + namespace: string; + locale: string; + }>): Promise { + const seen = new Set(); + for (const { namespace, locale } of pairs) { + const k = `${namespace}\0${locale}`; + if (seen.has(k)) + continue; + seen.add(k); + await this.cacheManager.del(bundleCacheKey(namespace, locale)); + } } - const fallback = await this.getRawBundleCached(namespace, def); - return { ...fallback, ...primary }; - } - - interpolate(template: string, vars?: Record): string { - if (!vars) return template; - return template.replace(/\{\{\s*(\w+)\s*\}\}/g, (_, k) => - Object.prototype.hasOwnProperty.call(vars, k) ? String(vars[k]) : `{{${k}}}`, - ); - } - - async translate( - namespace: string, - key: string, - locale?: string, - vars?: Record, - ): Promise { - const loc = locale - ? this.languageDetection.pickSupported(locale) || this.getDefaultLocale() - : this.getDefaultLocale(); - const merged = await this.getMessagesMerged(namespace, loc); - const raw = merged[key]; - const text = raw !== undefined && raw !== null && raw !== '' ? raw : `${namespace}.${key}`; - return this.interpolate(text, vars); - } - - async getBundleForApi( - namespace: string, - locale: string, - ): Promise<{ locale: string; namespace: string; messages: Record }> { - const loc = this.languageDetection.pickSupported(locale) || this.getDefaultLocale(); - const messages = await this.getMessagesMerged(namespace, loc); - return { namespace, locale: loc, messages }; - } - - async create(dto: CreateTranslationDto): Promise { - const namespace = dto.namespace.trim(); - const key = dto.key.trim(); - const locale = languageDetectionNormalize(dto.locale); - const existing = await this.translationRepo.findOne({ - where: { namespace, translationKey: key, locale }, - withDeleted: true, - }); - - if (existing) { - if (!existing.deletedAt) { - throw new ConflictException( - 'Translation already exists for this namespace, key, and locale', - ); - } - - existing.value = dto.value; - existing.deletedAt = null; - - const restored = await this.translationRepo.save(existing); - await this.invalidateBundles([{ namespace: restored.namespace, locale: restored.locale }]); - return this.toItem(restored); + private async loadRawBundleFromDb(namespace: string, locale: string): Promise> { + const rows = await this.translationRepo.find({ + where: { namespace, locale: languageDetectionNormalize(locale) }, + select: ['translationKey', 'value'], + }); + const map: Record = {}; + for (const r of rows) { + map[r.translationKey] = r.value; + } + return map; } - - const entity = this.translationRepo.create({ - namespace, - translationKey: key, - locale, - value: dto.value, - }); - try { - const saved = await this.translationRepo.save(entity); - await this.invalidateBundles([{ namespace: saved.namespace, locale: saved.locale }]); - return this.toItem(saved); - } catch (e) { - if (isUniqueViolation(e)) { - throw new ConflictException( - 'Translation already exists for this namespace, key, and locale', - ); - } - throw e; + async getRawBundleCached(namespace: string, locale: string): Promise> { + const loc = languageDetectionNormalize(locale); + const key = bundleCacheKey(namespace, loc); + const ttl = this.getCacheTtlMs(); + const cached = await this.cacheManager.get>(key); + if (cached) + return cached; + const fresh = await this.loadRawBundleFromDb(namespace, loc); + if (ttl > 0) { + await this.cacheManager.set(key, fresh, ttl); + } + else { + await this.cacheManager.set(key, fresh); + } + return fresh; } - } - - async findAll(query: ListTranslationsQueryDto): Promise { - const page = query.page ?? 1; - const limit = query.limit ?? 20; - const qb = this.translationRepo.createQueryBuilder('t'); - if (query.namespace) { - qb.andWhere('t.namespace = :ns', { ns: query.namespace.trim() }); + async getMessagesMerged(namespace: string, locale: string): Promise> { + const loc = this.languageDetection.pickSupported(locale) || this.getDefaultLocale(); + const primary = await this.getRawBundleCached(namespace, loc); + const def = this.getDefaultLocale(); + if (loc === def) { + return { ...primary }; + } + const fallback = await this.getRawBundleCached(namespace, def); + return { ...fallback, ...primary }; } - if (query.locale) { - qb.andWhere('t.locale = :loc', { loc: languageDetectionNormalize(query.locale) }); + interpolate(template: string, vars?: Record): string { + if (!vars) + return template; + return template.replace(/\{\{\s*(\w+)\s*\}\}/g, (_, k) => Object.prototype.hasOwnProperty.call(vars, k) ? String(vars[k]) : `{{${k}}}`); } - if (query.search?.trim()) { - const term = `%${query.search.trim()}%`; - qb.andWhere('(t.translationKey ILIKE :term OR t.value ILIKE :term)', { term }); + async translate(namespace: string, key: string, locale?: string, vars?: Record): Promise { + const loc = locale + ? this.languageDetection.pickSupported(locale) || this.getDefaultLocale() + : this.getDefaultLocale(); + const merged = await this.getMessagesMerged(namespace, loc); + const raw = merged[key]; + const text = raw !== undefined && raw !== null && raw !== '' ? raw : `${namespace}.${key}`; + return this.interpolate(text, vars); } - qb.orderBy('t.namespace', 'ASC') - .addOrderBy('t.locale', 'ASC') - .addOrderBy('t.translationKey', 'ASC'); - const total = await qb.getCount(); - qb.skip((page - 1) * limit).take(limit); - const rows = await qb.getMany(); - return { - items: rows.map((r) => this.toItem(r)), - total, - page, - limit, - }; - } - - async findOne(id: string): Promise { - const row = await this.translationRepo.findOne({ where: { id } }); - if (!row) throw new NotFoundException('Translation not found'); - return this.toItem(row); - } - - async update(id: string, dto: UpdateTranslationDto): Promise { - const row = await this.translationRepo.findOne({ where: { id } }); - if (!row) throw new NotFoundException('Translation not found'); - const before = { namespace: row.namespace, locale: row.locale }; - if (dto.namespace !== undefined) row.namespace = dto.namespace.trim(); - if (dto.key !== undefined) row.translationKey = dto.key.trim(); - if (dto.locale !== undefined) row.locale = languageDetectionNormalize(dto.locale); - if (dto.value !== undefined) row.value = dto.value; - try { - const saved = await this.translationRepo.save(row); - await this.invalidateBundles([before, { namespace: saved.namespace, locale: saved.locale }]); - return this.toItem(saved); - } catch (e) { - if (isUniqueViolation(e)) { - throw new ConflictException( - 'Translation already exists for this namespace, key, and locale', - ); - } - throw e; + async getBundleForApi(namespace: string, locale: string): Promise<{ + locale: string; + namespace: string; + messages: Record; + }> { + const loc = this.languageDetection.pickSupported(locale) || this.getDefaultLocale(); + const messages = await this.getMessagesMerged(namespace, loc); + return { namespace, locale: loc, messages }; } - } - - async remove(id: string): Promise { - const row = await this.translationRepo.findOne({ where: { id } }); - if (!row) throw new NotFoundException('Translation not found'); - await this.translationRepo.softRemove(row); - await this.invalidateBundles([{ namespace: row.namespace, locale: row.locale }]); - } - - async importRows(rows: TranslationImportRowDto[]): Promise<{ upserted: number }> { - if (!rows?.length) { - throw new BadRequestException('Import payload must contain at least one row'); + async create(dto: CreateTranslationDto): Promise { + const namespace = dto.namespace.trim(); + const key = dto.key.trim(); + const locale = languageDetectionNormalize(dto.locale); + const existing = await this.translationRepo.findOne({ + where: { namespace, translationKey: key, locale }, + withDeleted: true, + }); + if (existing) { + if (!existing.deletedAt) { + throw new ConflictException('Translation already exists for this namespace, key, and locale'); + } + existing.value = dto.value; + existing.deletedAt = null; + const restored = await this.translationRepo.save(existing); + await this.invalidateBundles([{ namespace: restored.namespace, locale: restored.locale }]); + return this.toItem(restored); + } + const entity = this.translationRepo.create({ + namespace, + translationKey: key, + locale, + value: dto.value, + }); + try { + const saved = await this.translationRepo.save(entity); + await this.invalidateBundles([{ namespace: saved.namespace, locale: saved.locale }]); + return this.toItem(saved); + } + catch (e) { + if (isUniqueViolation(e)) { + throw new ConflictException('Translation already exists for this namespace, key, and locale'); + } + throw e; + } } - - for (const row of rows) { - const namespace = row.namespace.trim(); - const translationKey = row.key.trim(); - const locale = languageDetectionNormalize(row.locale); - const existing = await this.translationRepo.findOne({ - where: { namespace, translationKey, locale }, - withDeleted: true, - }); - - if (existing) { - existing.value = row.value; - existing.deletedAt = null; - await this.translationRepo.save(existing); - continue; - } - - await this.translationRepo.save( - this.translationRepo.create({ - namespace, - translationKey, - locale, - value: row.value, - }), - ); + async findAll(query: ListTranslationsQueryDto): Promise { + const page = query.page ?? 1; + const limit = query.limit ?? 20; + const qb = this.translationRepo.createQueryBuilder('t'); + if (query.namespace) { + qb.andWhere('t.namespace = :ns', { ns: query.namespace.trim() }); + } + if (query.locale) { + qb.andWhere('t.locale = :loc', { loc: languageDetectionNormalize(query.locale) }); + } + if (query.search?.trim()) { + const term = `%${query.search.trim()}%`; + qb.andWhere('(t.translationKey ILIKE :term OR t.value ILIKE :term)', { term }); + } + qb.orderBy('t.namespace', 'ASC') + .addOrderBy('t.locale', 'ASC') + .addOrderBy('t.translationKey', 'ASC'); + const total = await qb.getCount(); + qb.skip((page - 1) * limit).take(limit); + const rows = await qb.getMany(); + return { + items: rows.map((r) => this.toItem(r)), + total, + page, + limit, + }; } - - const pairs = rows.map((r) => ({ - namespace: r.namespace.trim(), - locale: languageDetectionNormalize(r.locale), - })); - await this.invalidateBundles(pairs); - return { upserted: rows.length }; - } - - async exportRows( - namespace: string, - locale?: string, - ): Promise> { - const qb = this.translationRepo - .createQueryBuilder('t') - .where('t.namespace = :ns', { ns: namespace.trim() }); - if (locale) { - qb.andWhere('t.locale = :loc', { loc: languageDetectionNormalize(locale) }); + async findOne(id: string): Promise { + const row = await this.translationRepo.findOne({ where: { id } }); + if (!row) + throw new NotFoundException('Translation not found'); + return this.toItem(row); + } + async update(id: string, dto: UpdateTranslationDto): Promise { + const row = await this.translationRepo.findOne({ where: { id } }); + if (!row) + throw new NotFoundException('Translation not found'); + const before = { namespace: row.namespace, locale: row.locale }; + if (dto.namespace !== undefined) + row.namespace = dto.namespace.trim(); + if (dto.key !== undefined) + row.translationKey = dto.key.trim(); + if (dto.locale !== undefined) + row.locale = languageDetectionNormalize(dto.locale); + if (dto.value !== undefined) + row.value = dto.value; + try { + const saved = await this.translationRepo.save(row); + await this.invalidateBundles([before, { namespace: saved.namespace, locale: saved.locale }]); + return this.toItem(saved); + } + catch (e) { + if (isUniqueViolation(e)) { + throw new ConflictException('Translation already exists for this namespace, key, and locale'); + } + throw e; + } + } + async remove(id: string): Promise { + const row = await this.translationRepo.findOne({ where: { id } }); + if (!row) + throw new NotFoundException('Translation not found'); + await this.translationRepo.softRemove(row); + await this.invalidateBundles([{ namespace: row.namespace, locale: row.locale }]); + } + async importRows(rows: TranslationImportRowDto[]): Promise<{ + upserted: number; + }> { + if (!rows?.length) { + throw new BadRequestException('Import payload must contain at least one row'); + } + for (const row of rows) { + const namespace = row.namespace.trim(); + const translationKey = row.key.trim(); + const locale = languageDetectionNormalize(row.locale); + const existing = await this.translationRepo.findOne({ + where: { namespace, translationKey, locale }, + withDeleted: true, + }); + if (existing) { + existing.value = row.value; + existing.deletedAt = null; + await this.translationRepo.save(existing); + continue; + } + await this.translationRepo.save(this.translationRepo.create({ + namespace, + translationKey, + locale, + value: row.value, + })); + } + const pairs = rows.map((r) => ({ + namespace: r.namespace.trim(), + locale: languageDetectionNormalize(r.locale), + })); + await this.invalidateBundles(pairs); + return { upserted: rows.length }; + } + async exportRows(namespace: string, locale?: string): Promise> { + const qb = this.translationRepo + .createQueryBuilder('t') + .where('t.namespace = :ns', { ns: namespace.trim() }); + if (locale) { + qb.andWhere('t.locale = :loc', { loc: languageDetectionNormalize(locale) }); + } + qb.orderBy('t.locale', 'ASC').addOrderBy('t.translationKey', 'ASC'); + const rows = await qb.getMany(); + return rows.map((r) => ({ + namespace: r.namespace, + key: r.translationKey, + locale: r.locale, + value: r.value, + })); + } + static toCsv(rows: Array<{ + namespace: string; + key: string; + locale: string; + value: string; + }>): string { + const header = 'namespace,key,locale,value'; + const lines = rows.map((r) => [csvEscape(r.namespace), csvEscape(r.key), csvEscape(r.locale), csvEscape(r.value)].join(',')); + return [header, ...lines].join('\n'); } - qb.orderBy('t.locale', 'ASC').addOrderBy('t.translationKey', 'ASC'); - const rows = await qb.getMany(); - return rows.map((r) => ({ - namespace: r.namespace, - key: r.translationKey, - locale: r.locale, - value: r.value, - })); - } - - static toCsv( - rows: Array<{ namespace: string; key: string; locale: string; value: string }>, - ): string { - const header = 'namespace,key,locale,value'; - const lines = rows.map((r) => - [csvEscape(r.namespace), csvEscape(r.key), csvEscape(r.locale), csvEscape(r.value)].join(','), - ); - return [header, ...lines].join('\n'); - } } - function languageDetectionNormalize(locale: string): string { - return locale.trim().toLowerCase().replace(/_/g, '-'); + return locale.trim().toLowerCase().replace(/_/g, '-'); } - function csvEscape(field: string): string { - if (/[",\n\r]/.test(field)) { - return `"${field.replace(/"/g, '""')}"`; - } - return field; + if (/[",\n\r]/.test(field)) { + return `"${field.replace(/"/g, '""')}"`; + } + return field; } - function isUniqueViolation(err: unknown): boolean { - return ( - err instanceof QueryFailedError && - (err as { driverError?: { code?: string } }).driverError?.code === '23505' - ); + return (err instanceof QueryFailedError && + (err as { + driverError?: { + code?: string; + }; + }).driverError?.code === '23505'); } diff --git a/src/main.ts b/src/main.ts index 8f57ed97..7216b443 100644 --- a/src/main.ts +++ b/src/main.ts @@ -11,283 +11,218 @@ import { GlobalExceptionFilter } from './common/interceptors/global-exception.fi import { ResponseTransformInterceptor } from './common/interceptors/response-transform.interceptor'; import { LoggingInterceptor } from './common/interceptors/logging.interceptor'; import { correlationMiddleware } from './common/utils/correlation.utils'; -import { - API_VERSION_HEADER, - DEFAULT_API_VERSION, - SUPPORTED_API_VERSIONS, -} from './common/interceptors/api-version.interceptor'; +import { API_VERSION_HEADER, DEFAULT_API_VERSION, SUPPORTED_API_VERSIONS, } from './common/interceptors/api-version.interceptor'; import { API_VERSIONING_DOCUMENTATION } from './common/modules/api-versioning.module'; import { sessionConfig } from './config/cache.config'; import { SESSION_REDIS_CLIENT } from './session/session.constants'; import helmet from 'helmet'; import { corsConfig } from './config/cors.config'; import { ShutdownStateService } from './common/services/shutdown-state.service'; - async function bootstrapWorker(): Promise { - const logger = new Logger('Bootstrap'); - const bootstrapStartTime = Date.now(); - - // Create the application with dynamic module loading - const app = await NestFactory.create(await AppModule.forRoot(), { rawBody: true }); - const shutdownState = app.get(ShutdownStateService); - - app.enableVersioning({ - type: VersioningType.HEADER, - header: API_VERSION_HEADER, - defaultVersion: DEFAULT_API_VERSION, - }); - - // ─── Security Headers ───────────────────────────────────────────────────── - app.use( - helmet({ - hsts: { - maxAge: 31536000, - includeSubDomains: true, - preload: true, - }, - crossOriginEmbedderPolicy: false, - contentSecurityPolicy: { - directives: { - defaultSrc: ["'self'"], - styleSrc: ["'self'", "'unsafe-inline'"], - scriptSrc: ["'self'"], - imgSrc: ["'self'", 'data:', 'https:'], + const logger = new Logger('Bootstrap'); + const bootstrapStartTime = Date.now(); + // Create the application with dynamic module loading + const app = await NestFactory.create(await AppModule.forRoot(), { rawBody: true }); + const shutdownState = app.get(ShutdownStateService); + app.enableVersioning({ + type: VersioningType.HEADER, + header: API_VERSION_HEADER, + defaultVersion: DEFAULT_API_VERSION, + }); + // ─── Security Headers ───────────────────────────────────────────────────── + app.use(helmet({ + hsts: { + maxAge: 31536000, + includeSubDomains: true, + preload: true, + }, + crossOriginEmbedderPolicy: false, + contentSecurityPolicy: { + directives: { + defaultSrc: ["'self'"], + styleSrc: ["'self'", "'unsafe-inline'"], + scriptSrc: ["'self'"], + imgSrc: ["'self'", 'data:', 'https:'], + }, }, - }, - }), - ); - - const redisClient = app.get(SESSION_REDIS_CLIENT); - - if (sessionConfig.trustProxy) { - const expressApp = app.getHttpAdapter().getInstance(); - expressApp.set('trust proxy', 1); - } - - app.use(correlationMiddleware); - - app.use( - session({ - store: new RedisStore({ - client: redisClient, - prefix: sessionConfig.prefix, - ttl: sessionConfig.ttlSeconds, - }), - name: sessionConfig.name, - secret: sessionConfig.secret, - resave: false, - saveUninitialized: false, - rolling: true, - cookie: { - maxAge: sessionConfig.cookieMaxAgeMs, - httpOnly: true, - sameSite: 'strict', - secure: true, - }, - }), - ); - - // Session fixation protection: bind session to User-Agent - app.use((req: any, res: any, next: any) => { - if (!req.session) { - return next(); + })); + const redisClient = app.get(SESSION_REDIS_CLIENT); + if (sessionConfig.trustProxy) { + const expressApp = app.getHttpAdapter().getInstance(); + expressApp.set('trust proxy', 1); } - - const userAgent = req.headers['user-agent'] || 'unknown'; - if (!req.session.userAgent) { - req.session.userAgent = userAgent; - } else if (req.session.userAgent !== userAgent) { - return req.session.destroy((err: any) => { - if (err) { - logger.error('Error destroying session', err); + app.use(correlationMiddleware); + app.use(session({ + store: new RedisStore({ + client: redisClient, + prefix: sessionConfig.prefix, + ttl: sessionConfig.ttlSeconds, + }), + name: sessionConfig.name, + secret: sessionConfig.secret, + resave: false, + saveUninitialized: false, + rolling: true, + cookie: { + maxAge: sessionConfig.cookieMaxAgeMs, + httpOnly: true, + sameSite: 'strict', + secure: true, + }, + })); + // Session fixation protection: bind session to User-Agent + app.use((req: unknown, res: unknown, next: unknown) => { + if (!req.session) { + return next(); } - res.status(401).json({ message: 'Session invalidation due to fixation protection' }); - }); - } - next(); - }); - - // ─── Global Exception Filter ────────────────────────────────────────────── - app.useGlobalFilters(new GlobalExceptionFilter()); - - // ─── Global Logging Interceptor ─────────────────────────────────────────── - app.useGlobalInterceptors(new LoggingInterceptor()); - - // ─── Global Response Transform Interceptor ─────────────────────────────── - app.useGlobalInterceptors(new ResponseTransformInterceptor()); - - // ─── Global Timeout Interceptor ───────────────────────────────────────── - // TimeoutInterceptor is now provided globally via APP_INTERCEPTOR in AppModule - - // ─── CORS ───────────────────────────────────────────────────────────────── - app.enableCors(corsConfig); - - // ─── Validation ────────────────────────────────────────────────────────── - app.useGlobalPipes( - new ValidationPipe({ - whitelist: true, - transform: true, - forbidNonWhitelisted: true, - }), - ); - - // ─── Swagger ────────────────────────────────────────────────────────────── - const config = new DocumentBuilder() - .setTitle('TeachLink API') - .setDescription( - `The TeachLink API documentation - Unified System. ${API_VERSIONING_DOCUMENTATION}`, - ) - .setVersion('1.0') - .addBearerAuth() - .addTag('gamification', 'Gamification and user rewards') - .addTag('Email Marketing - Campaigns', 'Create and manage email campaigns') - .addTag('Email Marketing - Templates', 'Email template management') - .addTag('Email Marketing - Automation', 'Automation workflows') - .addTag('Email Marketing - Segments', 'Audience segmentation') - .addTag('Email Marketing - A/B Testing', 'A/B testing for campaigns') - .addTag('Email Marketing - Analytics', 'Campaign analytics and reporting') - .build(); - - const document = SwaggerModule.createDocument(app, config); - SwaggerModule.setup('api', app, document); - - const port = process.env.PORT || 3000; - app.enableShutdownHooks(); - await app.listen(port); - - const startupTime = Date.now() - bootstrapStartTime; - - if (sessionConfig.stickySessionsRequired) { - logger.log( - 'Sticky sessions are enabled by policy. Configure LB cookie affinity on teachlink.sid.', - ); - } - - logger.log(`🚀 TeachLink API running on http://localhost:${port}`); - logger.log(`📚 Swagger docs available at http://localhost:${port}/api`); - logger.log( - `🧭 API versioning enabled via ${API_VERSION_HEADER}. Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}; default route version: ${DEFAULT_API_VERSION}.`, - ); - logger.log(`⏱️ Application startup completed in ${startupTime}ms`); - - const shutdownTimeoutMs = parseInt(process.env.SHUTDOWN_TIMEOUT_MS || '30000', 10); - let isShuttingDown = false; - - const shutdown = async (signal: string): Promise => { - if (isShuttingDown) { - return; - } - - isShuttingDown = true; - shutdownState.markShuttingDown(); - logger.log(`Received ${signal}. Starting graceful shutdown...`); - - const forceExitTimer = setTimeout(() => { - logger.error(`Graceful shutdown timed out after ${shutdownTimeoutMs}ms. Forcing exit.`); - process.exit(1); - }, shutdownTimeoutMs); - forceExitTimer.unref(); - - try { - await app.close(); - logger.log('Graceful shutdown completed.'); - process.exit(0); - } catch (error) { - logger.error( - 'Error during graceful shutdown', - error instanceof Error ? error.stack : String(error), - ); - process.exit(1); + const userAgent = req.headers['user-agent'] || 'unknown'; + if (!req.session.userAgent) { + req.session.userAgent = userAgent; + } + else if (req.session.userAgent !== userAgent) { + return req.session.destroy((err: unknown) => { + if (err) { + logger.error('Error destroying session', err); + } + res.status(401).json({ message: 'Session invalidation due to fixation protection' }); + }); + } + next(); + }); + // ─── Global Exception Filter ────────────────────────────────────────────── + app.useGlobalFilters(new GlobalExceptionFilter()); + // ─── Global Logging Interceptor ─────────────────────────────────────────── + app.useGlobalInterceptors(new LoggingInterceptor()); + // ─── Global Response Transform Interceptor ─────────────────────────────── + app.useGlobalInterceptors(new ResponseTransformInterceptor()); + // ─── Global Timeout Interceptor ───────────────────────────────────────── + // TimeoutInterceptor is now provided globally via APP_INTERCEPTOR in AppModule + // ─── CORS ───────────────────────────────────────────────────────────────── + app.enableCors(corsConfig); + // ─── Validation ────────────────────────────────────────────────────────── + app.useGlobalPipes(new ValidationPipe({ + whitelist: true, + transform: true, + forbidNonWhitelisted: true, + })); + // ─── Swagger ────────────────────────────────────────────────────────────── + const config = new DocumentBuilder() + .setTitle('TeachLink API') + .setDescription(`The TeachLink API documentation - Unified System. ${API_VERSIONING_DOCUMENTATION}`) + .setVersion('1.0') + .addBearerAuth() + .addTag('gamification', 'Gamification and user rewards') + .addTag('Email Marketing - Campaigns', 'Create and manage email campaigns') + .addTag('Email Marketing - Templates', 'Email template management') + .addTag('Email Marketing - Automation', 'Automation workflows') + .addTag('Email Marketing - Segments', 'Audience segmentation') + .addTag('Email Marketing - A/B Testing', 'A/B testing for campaigns') + .addTag('Email Marketing - Analytics', 'Campaign analytics and reporting') + .build(); + const document = SwaggerModule.createDocument(app, config); + SwaggerModule.setup('api', app, document); + const port = process.env.PORT || 3000; + app.enableShutdownHooks(); + await app.listen(port); + const startupTime = Date.now() - bootstrapStartTime; + if (sessionConfig.stickySessionsRequired) { + logger.log('Sticky sessions are enabled by policy. Configure LB cookie affinity on teachlink.sid.'); } - }; - - process.on('SIGTERM', () => { - void shutdown('SIGTERM'); - }); - process.on('SIGINT', () => { - void shutdown('SIGINT'); - }); -} - -async function bootstrap(): Promise { - const logger = new Logger('Cluster'); - const clusterModeEnabled = (process.env.CLUSTER_MODE || 'false') === 'true'; - - if (clusterModeEnabled && cluster.isPrimary) { - const workerCount = parseInt(process.env.CLUSTER_WORKERS || `${cpus().length}`, 10); + logger.log(`🚀 TeachLink API running on http://localhost:${port}`); + logger.log(`📚 Swagger docs available at http://localhost:${port}/api`); + logger.log(`🧭 API versioning enabled via ${API_VERSION_HEADER}. Supported versions: ${SUPPORTED_API_VERSIONS.join(', ')}; default route version: ${DEFAULT_API_VERSION}.`); + logger.log(`⏱️ Application startup completed in ${startupTime}ms`); const shutdownTimeoutMs = parseInt(process.env.SHUTDOWN_TIMEOUT_MS || '30000', 10); let isShuttingDown = false; - let forceExitTimer: NodeJS.Timeout | undefined; - - logger.log(`Primary process started in cluster mode with ${workerCount} workers.`); - - for (let i = 0; i < workerCount; i += 1) { - cluster.fork(); - } - - cluster.on('exit', (worker, code, signal) => { - if (isShuttingDown) { - logger.log( - `Worker ${worker.id} (${worker.process.pid}) exited during shutdown (code: ${code}, signal: ${signal}).`, - ); - const remainingWorkers = Object.keys(cluster.workers || {}).length; - if (remainingWorkers === 0) { - if (forceExitTimer) { - clearTimeout(forceExitTimer); - } - logger.log('All workers have exited. Primary shutting down.'); - process.exit(0); + const shutdown = async (signal: string): Promise => { + if (isShuttingDown) { + return; } - return; - } - - logger.warn( - `Worker ${worker.id} (${worker.process.pid}) died (code: ${code}, signal: ${signal}). Restarting...`, - ); - cluster.fork(); - }); - - const shutdownCluster = (signal: string): void => { - if (isShuttingDown) { - return; - } - - isShuttingDown = true; - logger.log( - `Primary received ${signal}. Shutting down ${Object.keys(cluster.workers || {}).length} workers...`, - ); - - forceExitTimer = setTimeout(() => { - logger.error(`Cluster shutdown timed out after ${shutdownTimeoutMs}ms. Forcing exit.`); - for (const id in cluster.workers) { - const worker = cluster.workers[id]; - if (worker && !worker.isDead()) { - worker.process.kill('SIGKILL'); - } + isShuttingDown = true; + shutdownState.markShuttingDown(); + logger.log(`Received ${signal}. Starting graceful shutdown...`); + const forceExitTimer = setTimeout(() => { + logger.error(`Graceful shutdown timed out after ${shutdownTimeoutMs}ms. Forcing exit.`); + process.exit(1); + }, shutdownTimeoutMs); + forceExitTimer.unref(); + try { + await app.close(); + logger.log('Graceful shutdown completed.'); + process.exit(0); } - process.exit(1); - }, shutdownTimeoutMs); - forceExitTimer.unref(); - - for (const id in cluster.workers) { - const worker = cluster.workers[id]; - if (worker) { - worker.process.kill(signal as NodeJS.Signals); + catch (error) { + logger.error('Error during graceful shutdown', error instanceof Error ? error.stack : String(error)); + process.exit(1); } - } }; - process.on('SIGTERM', () => { - shutdownCluster('SIGTERM'); + void shutdown('SIGTERM'); }); process.on('SIGINT', () => { - shutdownCluster('SIGINT'); + void shutdown('SIGINT'); }); - - return; - } - - await bootstrapWorker(); } - +async function bootstrap(): Promise { + const logger = new Logger('Cluster'); + const clusterModeEnabled = (process.env.CLUSTER_MODE || 'false') === 'true'; + if (clusterModeEnabled && cluster.isPrimary) { + const workerCount = parseInt(process.env.CLUSTER_WORKERS || `${cpus().length}`, 10); + const shutdownTimeoutMs = parseInt(process.env.SHUTDOWN_TIMEOUT_MS || '30000', 10); + let isShuttingDown = false; + let forceExitTimer: NodeJS.Timeout | undefined; + logger.log(`Primary process started in cluster mode with ${workerCount} workers.`); + for (let i = 0; i < workerCount; i += 1) { + cluster.fork(); + } + cluster.on('exit', (worker, code, signal) => { + if (isShuttingDown) { + logger.log(`Worker ${worker.id} (${worker.process.pid}) exited during shutdown (code: ${code}, signal: ${signal}).`); + const remainingWorkers = Object.keys(cluster.workers || {}).length; + if (remainingWorkers === 0) { + if (forceExitTimer) { + clearTimeout(forceExitTimer); + } + logger.log('All workers have exited. Primary shutting down.'); + process.exit(0); + } + return; + } + logger.warn(`Worker ${worker.id} (${worker.process.pid}) died (code: ${code}, signal: ${signal}). Restarting...`); + cluster.fork(); + }); + const shutdownCluster = (signal: string): void => { + if (isShuttingDown) { + return; + } + isShuttingDown = true; + logger.log(`Primary received ${signal}. Shutting down ${Object.keys(cluster.workers || {}).length} workers...`); + forceExitTimer = setTimeout(() => { + logger.error(`Cluster shutdown timed out after ${shutdownTimeoutMs}ms. Forcing exit.`); + for (const id in cluster.workers) { + const worker = cluster.workers[id]; + if (worker && !worker.isDead()) { + worker.process.kill('SIGKILL'); + } + } + process.exit(1); + }, shutdownTimeoutMs); + forceExitTimer.unref(); + for (const id in cluster.workers) { + const worker = cluster.workers[id]; + if (worker) { + worker.process.kill(signal as NodeJS.Signals); + } + } + }; + process.on('SIGTERM', () => { + shutdownCluster('SIGTERM'); + }); + process.on('SIGINT', () => { + shutdownCluster('SIGINT'); + }); + return; + } + await bootstrapWorker(); +} bootstrap(); diff --git a/src/media/media.controller.spec.ts b/src/media/media.controller.spec.ts index 7bb64812..686b0abc 100644 --- a/src/media/media.controller.spec.ts +++ b/src/media/media.controller.spec.ts @@ -1,59 +1,46 @@ import { UnsupportedMediaTypeException } from '@nestjs/common'; import { MediaController } from './media.controller'; import { MediaService } from './media.service'; - describe('MediaController', () => { - let controller: MediaController; - let mediaService: { createFromUpload: jest.Mock }; - - beforeEach(() => { - mediaService = { - createFromUpload: jest.fn(), + let controller: MediaController; + let mediaService: { + createFromUpload: jest.Mock; }; - - controller = new MediaController(mediaService as unknown as MediaService); - }); - - it('rejects uploads blocked by the MIME whitelist', async () => { - await expect( - controller.upload( - undefined as any, - { - uploadValidationError: { - message: 'File type "application/x-msdownload" is not allowed', - }, - } as any, - ), - ).rejects.toBeInstanceOf(UnsupportedMediaTypeException); - }); - - it('passes validated files to the media service with enforced scanning', async () => { - const file = { - originalname: 'avatar.png', - mimetype: 'image/png', - size: 1024, - buffer: Buffer.from('png'), - }; - const req = { - user: { - id: 'user-1', - tenantId: 'tenant-1', - }, - }; - - mediaService.createFromUpload.mockResolvedValue({ content: { contentId: 'content-1' } }); - - await expect( - controller.upload(file as any, req, { - compress: 'false', - generateThumbnails: 'true', - }), - ).resolves.toEqual({ content: { contentId: 'content-1' } }); - - expect(mediaService.createFromUpload).toHaveBeenCalledWith('user-1', 'tenant-1', file, { - compress: false, - generateThumbnails: true, - trackProgress: true, + beforeEach(() => { + mediaService = { + createFromUpload: jest.fn(), + }; + controller = new MediaController(mediaService as unknown as MediaService); + }); + it('rejects uploads blocked by the MIME whitelist', async () => { + await expect(controller.upload(undefined as unknown, { + uploadValidationError: { + message: 'File type "application/x-msdownload" is not allowed', + }, + } as unknown)).rejects.toBeInstanceOf(UnsupportedMediaTypeException); + }); + it('passes validated files to the media service with enforced scanning', async () => { + const file = { + originalname: 'avatar.png', + mimetype: 'image/png', + size: 1024, + buffer: Buffer.from('png'), + }; + const req = { + user: { + id: 'user-1', + tenantId: 'tenant-1', + }, + }; + mediaService.createFromUpload.mockResolvedValue({ content: { contentId: 'content-1' } }); + await expect(controller.upload(file as unknown, req, { + compress: 'false', + generateThumbnails: 'true', + })).resolves.toEqual({ content: { contentId: 'content-1' } }); + expect(mediaService.createFromUpload).toHaveBeenCalledWith('user-1', 'tenant-1', file, { + compress: false, + generateThumbnails: true, + trackProgress: true, + }); }); - }); }); diff --git a/src/media/media.controller.ts b/src/media/media.controller.ts index 4962f79a..d6c9af61 100644 --- a/src/media/media.controller.ts +++ b/src/media/media.controller.ts @@ -1,167 +1,133 @@ -import { - Controller, - Post, - UseInterceptors, - UploadedFile, - UseGuards, - Get, - Param, - Req, - HttpException, - HttpStatus, - Logger, - Body, - UnsupportedMediaTypeException, -} from '@nestjs/common'; +import { Controller, Post, UseInterceptors, UploadedFile, UseGuards, Get, Param, Req, HttpException, HttpStatus, Logger, Body, UnsupportedMediaTypeException, } from '@nestjs/common'; import { FileInterceptor } from '@nestjs/platform-express'; import { Throttle } from '@nestjs/throttler'; -import { - ApiTags, - ApiOperation, - ApiResponse, - ApiConsumes, - ApiBody, - ApiParam, - ApiBearerAuth, -} from '@nestjs/swagger'; +import { ApiTags, ApiOperation, ApiResponse, ApiConsumes, ApiBody, ApiParam, ApiBearerAuth, } from '@nestjs/swagger'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { MediaService } from './media.service'; import { UploadedFile as FileUpload } from '../common/types/file.types'; -import { - buildUploadValidationDetails, - MEDIA_UPLOAD_INTERCEPTOR_OPTIONS, -} from './validation/upload-validation.util'; - +import { buildUploadValidationDetails, MEDIA_UPLOAD_INTERCEPTOR_OPTIONS, } from './validation/upload-validation.util'; @ApiTags('Media') @ApiBearerAuth() @Controller('media') export class MediaController { - private readonly logger = new Logger(MediaController.name); - - constructor(private readonly mediaService: MediaService) {} - - @Post('upload') - @Throttle({ default: { limit: 10, ttl: 3600000 } }) - @UseGuards(JwtAuthGuard) - @UseInterceptors(FileInterceptor('file', MEDIA_UPLOAD_INTERCEPTOR_OPTIONS)) - @ApiOperation({ summary: 'Upload media file with full validation' }) - @ApiConsumes('multipart/form-data') - @ApiBody({ - description: 'Media file upload', - schema: { - type: 'object', - properties: { - file: { - type: 'string', - format: 'binary', - description: 'File to upload', + private readonly logger = new Logger(MediaController.name); + constructor(private readonly mediaService: MediaService) { } + @Post('upload') + @Throttle({ default: { limit: 10, ttl: 3600000 } }) + @UseGuards(JwtAuthGuard) + @UseInterceptors(FileInterceptor('file', MEDIA_UPLOAD_INTERCEPTOR_OPTIONS)) + @ApiOperation({ summary: 'Upload media file with full validation' }) + @ApiConsumes('multipart/form-data') + @ApiBody({ + description: 'Media file upload', + schema: { + type: 'object', + properties: { + file: { + type: 'string', + format: 'binary', + description: 'File to upload', + }, + compress: { + type: 'boolean', + description: 'Compress images automatically', + default: true, + }, + generateThumbnails: { + type: 'boolean', + description: 'Generate thumbnails for images', + default: true, + }, + }, }, - compress: { - type: 'boolean', - description: 'Compress images automatically', - default: true, - }, - generateThumbnails: { - type: 'boolean', - description: 'Generate thumbnails for images', - default: true, - }, - }, - }, - }) - @ApiResponse({ status: 201, description: 'File uploaded successfully' }) - @ApiResponse({ status: 400, description: 'Validation failed' }) - @ApiResponse({ status: 403, description: 'Malware detected' }) - @ApiResponse({ status: 413, description: 'File too large' }) - @ApiResponse({ status: 415, description: 'Unsupported file type' }) - @ApiResponse({ status: 503, description: 'Malware scanning unavailable' }) - async upload( - @UploadedFile() file: FileUpload, - @Req() req: any, - @Body() body?: { compress?: string; generateThumbnails?: string }, - ) { - if (req.uploadValidationError) { - throw new UnsupportedMediaTypeException({ - message: req.uploadValidationError.message, - ...buildUploadValidationDetails(), - }); + }) + @ApiResponse({ status: 201, description: 'File uploaded successfully' }) + @ApiResponse({ status: 400, description: 'Validation failed' }) + @ApiResponse({ status: 403, description: 'Malware detected' }) + @ApiResponse({ status: 413, description: 'File too large' }) + @ApiResponse({ status: 415, description: 'Unsupported file type' }) + @ApiResponse({ status: 503, description: 'Malware scanning unavailable' }) + async upload( + @UploadedFile() + file: FileUpload, + @Req() + req: unknown, + @Body() + body?: { + compress?: string; + generateThumbnails?: string; + }) { + if (req.uploadValidationError) { + throw new UnsupportedMediaTypeException({ + message: req.uploadValidationError.message, + ...buildUploadValidationDetails(), + }); + } + if (!file) { + throw new HttpException('No file provided', HttpStatus.BAD_REQUEST); + } + const user = req.user; + this.logger.log(`User ${user?.id} uploading file ${file.originalname}`); + const options = { + compress: body?.compress !== 'false', + generateThumbnails: body?.generateThumbnails !== 'false', + trackProgress: true, + }; + const result = await this.mediaService.createFromUpload(user?.id, user?.tenantId, file, options); + return result; + } + @Get('uploads/progress/:uploadId') + @UseGuards(JwtAuthGuard) + @ApiOperation({ summary: 'Get upload progress by ID' }) + @ApiParam({ name: 'uploadId', description: 'Upload tracking ID' }) + @ApiResponse({ status: 200, description: 'Upload progress', type: Object }) + @ApiResponse({ status: 404, description: 'Upload not found' }) + async getUploadProgress( + @Param('uploadId') + uploadId: string) { + const progress = await this.mediaService.getUploadProgress(uploadId); + if (!progress) { + throw new HttpException('Upload not found', HttpStatus.NOT_FOUND); + } + return progress; } - - if (!file) { - throw new HttpException('No file provided', HttpStatus.BAD_REQUEST); + @Get('uploads/active') + @UseGuards(JwtAuthGuard) + @ApiOperation({ summary: 'List active uploads' }) + @ApiResponse({ status: 200, description: 'List of active uploads' }) + async listActiveUploads() { + return this.mediaService.listActiveUploads(); } - - const user = req.user; - this.logger.log(`User ${user?.id} uploading file ${file.originalname}`); - - const options = { - compress: body?.compress !== 'false', - generateThumbnails: body?.generateThumbnails !== 'false', - trackProgress: true, - }; - - const result = await this.mediaService.createFromUpload( - user?.id, - user?.tenantId, - file, - options, - ); - - return result; - } - - @Get('uploads/progress/:uploadId') - @UseGuards(JwtAuthGuard) - @ApiOperation({ summary: 'Get upload progress by ID' }) - @ApiParam({ name: 'uploadId', description: 'Upload tracking ID' }) - @ApiResponse({ status: 200, description: 'Upload progress', type: Object }) - @ApiResponse({ status: 404, description: 'Upload not found' }) - async getUploadProgress(@Param('uploadId') uploadId: string) { - const progress = await this.mediaService.getUploadProgress(uploadId); - if (!progress) { - throw new HttpException('Upload not found', HttpStatus.NOT_FOUND); + @Get('uploads/statistics') + @UseGuards(JwtAuthGuard) + @ApiOperation({ summary: 'Get upload statistics' }) + @ApiResponse({ status: 200, description: 'Upload statistics' }) + async getUploadStatistics() { + return this.mediaService.getUploadStatistics(); } - return progress; - } - - @Get('uploads/active') - @UseGuards(JwtAuthGuard) - @ApiOperation({ summary: 'List active uploads' }) - @ApiResponse({ status: 200, description: 'List of active uploads' }) - async listActiveUploads() { - return this.mediaService.listActiveUploads(); - } - - @Get('uploads/statistics') - @UseGuards(JwtAuthGuard) - @ApiOperation({ summary: 'Get upload statistics' }) - @ApiResponse({ status: 200, description: 'Upload statistics' }) - async getUploadStatistics() { - return this.mediaService.getUploadStatistics(); - } - - @Get(':contentId') - @UseGuards(JwtAuthGuard) - @ApiOperation({ summary: 'Get media metadata by content ID' }) - @ApiParam({ name: 'contentId', description: 'Content identifier' }) - @ApiResponse({ status: 200, description: 'Media metadata' }) - @ApiResponse({ status: 404, description: 'Not found' }) - @ApiResponse({ status: 403, description: 'Forbidden' }) - async getMetadata(@Param('contentId') contentId: string, @Req() req: any) { - const user = req.user; - const meta = await this.mediaService.findByContentId(contentId); - if (!meta) throw new HttpException('Not found', HttpStatus.NOT_FOUND); - - // Access control: owner or same tenant or admin - if ( - meta.ownerId && - meta.ownerId !== user?.id && - user?.role !== 'admin' && - meta.tenantId !== user?.tenantId - ) { - throw new HttpException('Forbidden', HttpStatus.FORBIDDEN); + @Get(':contentId') + @UseGuards(JwtAuthGuard) + @ApiOperation({ summary: 'Get media metadata by content ID' }) + @ApiParam({ name: 'contentId', description: 'Content identifier' }) + @ApiResponse({ status: 200, description: 'Media metadata' }) + @ApiResponse({ status: 404, description: 'Not found' }) + @ApiResponse({ status: 403, description: 'Forbidden' }) + async getMetadata( + @Param('contentId') + contentId: string, + @Req() + req: unknown) { + const user = req.user; + const meta = await this.mediaService.findByContentId(contentId); + if (!meta) + throw new HttpException('Not found', HttpStatus.NOT_FOUND); + // Access control: owner or same tenant or admin + if (meta.ownerId && + meta.ownerId !== user?.id && + user?.role !== 'admin' && + meta.tenantId !== user?.tenantId) { + throw new HttpException('Forbidden', HttpStatus.FORBIDDEN); + } + return meta; } - - return meta; - } } diff --git a/src/media/media.module.ts b/src/media/media.module.ts index ba34cbe1..9facf846 100644 --- a/src/media/media.module.ts +++ b/src/media/media.module.ts @@ -13,31 +13,31 @@ import { MalwareScanningService } from './validation/malware-scanning.service'; import { UploadProgressService } from './validation/upload-progress.service'; import { ContentMetadata } from '../cdn/entities/content-metadata.entity'; import { VideoProcessor } from './processing/video.processor'; - @Module({ - imports: [ - TypeOrmModule.forFeature([ContentMetadata]), - BullModule.registerQueue({ name: QUEUE_NAMES.MEDIA_PROCESSING }), - ], - controllers: [MediaController], - providers: [ - MediaService, - FileStorageService, - VideoProcessingService, - DocumentProcessingService, - ImageProcessingService, - FileValidationService, - MalwareScanningService, - UploadProgressService, - // processors - VideoProcessor, - ], - exports: [ - MediaService, - FileStorageService, - VideoProcessingService, - FileValidationService, - ImageProcessingService, - ], + imports: [ + TypeOrmModule.forFeature([ContentMetadata]), + BullModule.registerQueue({ name: QUEUE_NAMES.MEDIA_PROCESSING }), + ], + controllers: [MediaController], + providers: [ + MediaService, + FileStorageService, + VideoProcessingService, + DocumentProcessingService, + ImageProcessingService, + FileValidationService, + MalwareScanningService, + UploadProgressService, + // processors + VideoProcessor, + ], + exports: [ + MediaService, + FileStorageService, + VideoProcessingService, + FileValidationService, + ImageProcessingService, + ], }) -export class MediaModule {} +export class MediaModule { +} diff --git a/src/media/media.service.spec.ts b/src/media/media.service.spec.ts index c0b8f257..7eeb9f10 100644 --- a/src/media/media.service.spec.ts +++ b/src/media/media.service.spec.ts @@ -1,116 +1,84 @@ import { ForbiddenException, ServiceUnavailableException } from '@nestjs/common'; import { ContentStatus } from '../cdn/entities/content-metadata.entity'; import { MediaService } from './media.service'; - describe('MediaService', () => { - // ─── Declarations ────────────────────────────────────────────────────────── - let service: MediaService; - let mockContentRepo: jest.Mocked; - let mockStorage: jest.Mocked; - let mockVideoProcessing: jest.Mocked; - let mockFileValidation: jest.Mocked; - let mockMalwareScanning: jest.Mocked; - let mockImageProcessing: jest.Mocked; - let mockUploadProgress: jest.Mocked; - - const file = { - originalname: 'avatar.png', - mimetype: 'image/png', - size: 1024, - buffer: Buffer.from('png'), - }; - - // ─── Setup ───────────────────────────────────────────────────────────────── - beforeEach(() => { - // Initialize all dependency mocks with proper typing - mockContentRepo = { - create: jest.fn().mockImplementation((value) => value), - save: jest.fn().mockImplementation(async (value) => ({ - status: ContentStatus.READY, - ...value, - })), - findOne: jest.fn(), - } as jest.Mocked; - - mockStorage = { - uploadFile: jest.fn(), - uploadProcessedFile: jest.fn(), - } as jest.Mocked; - - mockVideoProcessing = { - enqueueTranscode: jest.fn(), - } as jest.Mocked; - - mockFileValidation = { - validateFile: jest.fn().mockResolvedValue({ - valid: true, - errors: [], - warnings: [], - metadata: {}, - }), - } as jest.Mocked; - - mockMalwareScanning = { - isScanningAvailable: jest.fn(), - scanFile: jest.fn(), - } as jest.Mocked; - - mockImageProcessing = { - compressImage: jest.fn(), - generateThumbnails: jest.fn(), - } as jest.Mocked; - - mockUploadProgress = { - initializeUpload: jest.fn(), - updateProgress: jest.fn(), - markFailed: jest.fn(), - markCompleted: jest.fn(), - getProgress: jest.fn(), - listActiveUploads: jest.fn(), - getStatistics: jest.fn(), - } as jest.Mocked; - - service = new MediaService( - mockContentRepo, - mockStorage, - mockVideoProcessing, - mockFileValidation, - mockMalwareScanning, - mockImageProcessing, - mockUploadProgress, - ); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - it('fails closed when malware scanning is required but unavailable', async () => { - mockMalwareScanning.isScanningAvailable.mockReturnValue(false); - - await expect( - service.createFromUpload('user-1', 'tenant-1', file as any), - ).rejects.toBeInstanceOf(ServiceUnavailableException); - - expect(mockMalwareScanning.scanFile).not.toHaveBeenCalled(); - expect(mockUploadProgress.markFailed).toHaveBeenCalled(); - }); - - it('blocks uploads when malware is detected', async () => { - mockMalwareScanning.isScanningAvailable.mockReturnValue(true); - mockMalwareScanning.scanFile.mockResolvedValue({ - clean: false, - threats: ['EICAR-Test-File'], - scanTime: 42, + // ─── Declarations ────────────────────────────────────────────────────────── + let service: MediaService; + let mockContentRepo: jest.Mocked; + let mockStorage: jest.Mocked; + let mockVideoProcessing: jest.Mocked; + let mockFileValidation: jest.Mocked; + let mockMalwareScanning: jest.Mocked; + let mockImageProcessing: jest.Mocked; + let mockUploadProgress: jest.Mocked; + const file = { + originalname: 'avatar.png', + mimetype: 'image/png', + size: 1024, + buffer: Buffer.from('png'), + }; + // ─── Setup ───────────────────────────────────────────────────────────────── + beforeEach(() => { + // Initialize all dependency mocks with proper typing + mockContentRepo = { + create: jest.fn().mockImplementation((value) => value), + save: jest.fn().mockImplementation(async (value) => ({ + status: ContentStatus.READY, + ...value, + })), + findOne: jest.fn(), + } as jest.Mocked; + mockStorage = { + uploadFile: jest.fn(), + uploadProcessedFile: jest.fn(), + } as jest.Mocked; + mockVideoProcessing = { + enqueueTranscode: jest.fn(), + } as jest.Mocked; + mockFileValidation = { + validateFile: jest.fn().mockResolvedValue({ + valid: true, + errors: [], + warnings: [], + metadata: {}, + }), + } as jest.Mocked; + mockMalwareScanning = { + isScanningAvailable: jest.fn(), + scanFile: jest.fn(), + } as jest.Mocked; + mockImageProcessing = { + compressImage: jest.fn(), + generateThumbnails: jest.fn(), + } as jest.Mocked; + mockUploadProgress = { + initializeUpload: jest.fn(), + updateProgress: jest.fn(), + markFailed: jest.fn(), + markCompleted: jest.fn(), + getProgress: jest.fn(), + listActiveUploads: jest.fn(), + getStatistics: jest.fn(), + } as jest.Mocked; + service = new MediaService(mockContentRepo, mockStorage, mockVideoProcessing, mockFileValidation, mockMalwareScanning, mockImageProcessing, mockUploadProgress); + }); + afterEach(() => { + jest.clearAllMocks(); + }); + it('fails closed when malware scanning is required but unavailable', async () => { + mockMalwareScanning.isScanningAvailable.mockReturnValue(false); + await expect(service.createFromUpload('user-1', 'tenant-1', file as unknown)).rejects.toBeInstanceOf(ServiceUnavailableException); + expect(mockMalwareScanning.scanFile).not.toHaveBeenCalled(); + expect(mockUploadProgress.markFailed).toHaveBeenCalled(); + }); + it('blocks uploads when malware is detected', async () => { + mockMalwareScanning.isScanningAvailable.mockReturnValue(true); + mockMalwareScanning.scanFile.mockResolvedValue({ + clean: false, + threats: ['EICAR-Test-File'], + scanTime: 42, + }); + await expect(service.createFromUpload('user-1', 'tenant-1', file as unknown)).rejects.toBeInstanceOf(ForbiddenException); + expect(mockUploadProgress.markFailed).toHaveBeenCalledWith(expect.any(String), 'Malware detected: EICAR-Test-File'); }); - - await expect( - service.createFromUpload('user-1', 'tenant-1', file as any), - ).rejects.toBeInstanceOf(ForbiddenException); - - expect(mockUploadProgress.markFailed).toHaveBeenCalledWith( - expect.any(String), - 'Malware detected: EICAR-Test-File', - ); - }); }); diff --git a/src/media/media.service.ts b/src/media/media.service.ts index a61c1bcf..4a297b92 100644 --- a/src/media/media.service.ts +++ b/src/media/media.service.ts @@ -1,17 +1,7 @@ -import { - Injectable, - Logger, - BadRequestException, - ForbiddenException, - ServiceUnavailableException, -} from '@nestjs/common'; +import { Injectable, Logger, BadRequestException, ForbiddenException, ServiceUnavailableException, } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; -import { - ContentMetadata, - ContentStatus, - ContentType, -} from '../cdn/entities/content-metadata.entity'; +import { ContentMetadata, ContentStatus, ContentType, } from '../cdn/entities/content-metadata.entity'; import { FileStorageService } from './storage/file-storage.service'; import { VideoProcessingService } from './processing/video-processing.service'; import { UploadedFile } from '../common/types/file.types'; @@ -20,285 +10,240 @@ import { MalwareScanningService } from './validation/malware-scanning.service'; import { ImageProcessingService } from './processing/image-processing.service'; import { UploadProgressService } from './validation/upload-progress.service'; import { v4 as uuidv4 } from 'uuid'; - export interface UploadOptions { - compress?: boolean; - generateThumbnails?: boolean; - scanForMalware?: boolean; - trackProgress?: boolean; + compress?: boolean; + generateThumbnails?: boolean; + scanForMalware?: boolean; + trackProgress?: boolean; } - export interface UploadResult { - content: ContentMetadata; - thumbnails?: Array<{ name: string; url: string }>; - compressionRatio?: number; - scanResult?: { clean: boolean; threats: string[] }; + content: ContentMetadata; + thumbnails?: Array<{ + name: string; + url: string; + }>; + compressionRatio?: number; + scanResult?: { + clean: boolean; + threats: string[]; + }; } - @Injectable() export class MediaService { - private readonly logger = new Logger(MediaService.name); - - constructor( + private readonly logger = new Logger(MediaService.name); + constructor( @InjectRepository(ContentMetadata) - private readonly contentRepo: Repository, - private readonly storage: FileStorageService, - private readonly videoProcessing: VideoProcessingService, - private readonly fileValidation: FileValidationService, - private readonly malwareScanning: MalwareScanningService, - private readonly imageProcessing: ImageProcessingService, - private readonly uploadProgress: UploadProgressService, - ) {} - - async createFromUpload( - ownerId: string, - tenantId: string | undefined, - file: UploadedFile, - options: UploadOptions = {}, - ): Promise { - const uploadId = uuidv4(); - const result: UploadResult = { content: null as unknown as ContentMetadata }; - - try { - // Initialize progress tracking if enabled - if (options.trackProgress !== false) { - await this.uploadProgress.initializeUpload(uploadId, file.originalname, file.size); - } - - // Step 1: Validate file - if (options.trackProgress !== false) { - await this.uploadProgress.updateProgress(uploadId, { - status: 'validating', - progress: 10, - stage: 'validation', - message: 'Validating file...', - }); - } - - const validationResult = await this.fileValidation.validateFile(file); - if (!validationResult.valid) { - if (options.trackProgress !== false) { - await this.uploadProgress.markFailed(uploadId, validationResult.errors.join(', ')); - } - throw new BadRequestException({ - message: 'File validation failed', - errors: validationResult.errors, - warnings: validationResult.warnings, - }); - } - - // Step 2: Malware scanning - if (options.scanForMalware !== false) { - if (!this.malwareScanning.isScanningAvailable()) { - const errorMsg = - 'Malware scanning is required for uploads but no scanning service is available'; - - if (options.trackProgress !== false) { - await this.uploadProgress.markFailed(uploadId, errorMsg); - } - - throw new ServiceUnavailableException(errorMsg); - } - - if (options.trackProgress !== false) { - await this.uploadProgress.updateProgress(uploadId, { - status: 'scanning', - progress: 25, - stage: 'malware-scan', - message: 'Scanning for malware...', - }); - } - - const scanResult = await this.malwareScanning.scanFile(file); - result.scanResult = { - clean: scanResult.clean, - threats: scanResult.threats, - }; - - if (!scanResult.clean) { - const detectedThreats = scanResult.threats.filter(Boolean); - const errorMsg = - detectedThreats.length > 0 - ? `Malware detected: ${detectedThreats.join(', ')}` - : scanResult.error || 'File failed security scan'; - - if (options.trackProgress !== false) { - await this.uploadProgress.markFailed(uploadId, errorMsg); - } - - if (detectedThreats.length > 0) { - throw new ForbiddenException(errorMsg); - } - - throw new ServiceUnavailableException(errorMsg); - } - } - - // Create metadata entity - const content = this.contentRepo.create({ - contentId: `media_${Date.now()}_${Math.random().toString(36).substr(2, 8)}`, - originalUrl: '', - cdnUrl: '', - contentType: this.mapContentType(file.mimetype), - fileName: file.originalname, - mimeType: file.mimetype, - fileSize: file.size, - status: ContentStatus.UPLOADING, - provider: 'media', - metadata: validationResult.metadata || {}, - ownerId, - tenantId, - } as Partial); - - await this.contentRepo.save(content); - result.content = content; - - // Step 3: Process image (compression & thumbnails) - let processedFile = file; - if (content.contentType === ContentType.IMAGE && options.compress !== false) { - if (options.trackProgress !== false) { - await this.uploadProgress.updateProgress(uploadId, { - status: 'processing', - progress: 40, - stage: 'image-processing', - message: 'Optimizing image...', - }); - } - + private readonly contentRepo: Repository, private readonly storage: FileStorageService, private readonly videoProcessing: VideoProcessingService, private readonly fileValidation: FileValidationService, private readonly malwareScanning: MalwareScanningService, private readonly imageProcessing: ImageProcessingService, private readonly uploadProgress: UploadProgressService) { } + async createFromUpload(ownerId: string, tenantId: string | undefined, file: UploadedFile, options: UploadOptions = {}): Promise { + const uploadId = uuidv4(); + const result: UploadResult = { content: null as unknown as ContentMetadata }; try { - const compressed = await this.imageProcessing.compressImage(file.buffer); - processedFile = { - ...file, - buffer: compressed.buffer, - size: compressed.size, - }; - result.compressionRatio = compressed.compressionRatio; - content.optimizedSize = compressed.size; - - // Update metadata with dimensions - if (compressed.width && compressed.height) { - content.metadata = { - ...content.metadata, - width: compressed.width, - height: compressed.height, - }; - } - - // Generate thumbnails - if (options.generateThumbnails !== false) { + // Initialize progress tracking if enabled + if (options.trackProgress !== false) { + await this.uploadProgress.initializeUpload(uploadId, file.originalname, file.size); + } + // Step 1: Validate file + if (options.trackProgress !== false) { + await this.uploadProgress.updateProgress(uploadId, { + status: 'validating', + progress: 10, + stage: 'validation', + message: 'Validating file...', + }); + } + const validationResult = await this.fileValidation.validateFile(file); + if (!validationResult.valid) { + if (options.trackProgress !== false) { + await this.uploadProgress.markFailed(uploadId, validationResult.errors.join(', ')); + } + throw new BadRequestException({ + message: 'File validation failed', + errors: validationResult.errors, + warnings: validationResult.warnings, + }); + } + // Step 2: Malware scanning + if (options.scanForMalware !== false) { + if (!this.malwareScanning.isScanningAvailable()) { + const errorMsg = 'Malware scanning is required for uploads but no scanning service is available'; + if (options.trackProgress !== false) { + await this.uploadProgress.markFailed(uploadId, errorMsg); + } + throw new ServiceUnavailableException(errorMsg); + } + if (options.trackProgress !== false) { + await this.uploadProgress.updateProgress(uploadId, { + status: 'scanning', + progress: 25, + stage: 'malware-scan', + message: 'Scanning for malware...', + }); + } + const scanResult = await this.malwareScanning.scanFile(file); + result.scanResult = { + clean: scanResult.clean, + threats: scanResult.threats, + }; + if (!scanResult.clean) { + const detectedThreats = scanResult.threats.filter(Boolean); + const errorMsg = detectedThreats.length > 0 + ? `Malware detected: ${detectedThreats.join(', ')}` + : scanResult.error || 'File failed security scan'; + if (options.trackProgress !== false) { + await this.uploadProgress.markFailed(uploadId, errorMsg); + } + if (detectedThreats.length > 0) { + throw new ForbiddenException(errorMsg); + } + throw new ServiceUnavailableException(errorMsg); + } + } + // Create metadata entity + const content = this.contentRepo.create({ + contentId: `media_${Date.now()}_${Math.random().toString(36).substr(2, 8)}`, + originalUrl: '', + cdnUrl: '', + contentType: this.mapContentType(file.mimetype), + fileName: file.originalname, + mimeType: file.mimetype, + fileSize: file.size, + status: ContentStatus.UPLOADING, + provider: 'media', + metadata: validationResult.metadata || {}, + ownerId, + tenantId, + } as Partial); + await this.contentRepo.save(content); + result.content = content; + // Step 3: Process image (compression & thumbnails) + let processedFile = file; + if (content.contentType === ContentType.IMAGE && options.compress !== false) { + if (options.trackProgress !== false) { + await this.uploadProgress.updateProgress(uploadId, { + status: 'processing', + progress: 40, + stage: 'image-processing', + message: 'Optimizing image...', + }); + } + try { + const compressed = await this.imageProcessing.compressImage(file.buffer); + processedFile = { + ...file, + buffer: compressed.buffer, + size: compressed.size, + }; + result.compressionRatio = compressed.compressionRatio; + content.optimizedSize = compressed.size; + // Update metadata with dimensions + if (compressed.width && compressed.height) { + content.metadata = { + ...content.metadata, + width: compressed.width, + height: compressed.height, + }; + } + // Generate thumbnails + if (options.generateThumbnails !== false) { + if (options.trackProgress !== false) { + await this.uploadProgress.updateProgress(uploadId, { + progress: 60, + stage: 'thumbnail-generation', + message: 'Generating thumbnails...', + }); + } + const thumbnails = await this.imageProcessing.generateThumbnails(file.buffer); + result.thumbnails = []; + for (const thumb of thumbnails) { + const thumbKey = `${content.contentId}/thumbnails/${thumb.name}.webp`; + await this.storage.uploadProcessedFile(thumb.buffer, thumbKey, 'image/webp'); + result.thumbnails.push({ + name: thumb.name, + url: `https://${process.env.AWS_S3_BUCKET}.s3.amazonaws.com/${thumbKey}`, + }); + } + // Store thumbnail URLs in variants + content.variants = result.thumbnails.map((t) => ({ + name: t.name, + url: t.url, + width: 0, // Will be populated from thumbnail data + height: 0, + size: 0, + })); + } + } + catch (error) { + this.logger.warn('Image processing failed, using original:', error); + } + } + // Step 4: Upload to storage + if (options.trackProgress !== false) { + await this.uploadProgress.updateProgress(uploadId, { + status: 'uploading', + progress: 80, + stage: 'storage-upload', + message: 'Uploading to storage...', + }); + } + const upload = await this.storage.uploadFile(processedFile, content); + content.originalUrl = upload.url; + content.cdnUrl = upload.url; + content.etag = upload.etag; + content.status = ContentStatus.READY; + await this.contentRepo.save(content); + // Step 5: Video processing (if applicable) + if (content.contentType === ContentType.VIDEO) { + this.logger.log(`Enqueue video processing for ${content.contentId}`); + await this.videoProcessing.enqueueTranscode(content); + content.status = ContentStatus.PROCESSING; + await this.contentRepo.save(content); + } + // Mark as completed if (options.trackProgress !== false) { - await this.uploadProgress.updateProgress(uploadId, { - progress: 60, - stage: 'thumbnail-generation', - message: 'Generating thumbnails...', - }); + await this.uploadProgress.markCompleted(uploadId, { + contentId: content.contentId, + url: content.cdnUrl, + thumbnails: result.thumbnails?.map((t) => t.url), + }); } - - const thumbnails = await this.imageProcessing.generateThumbnails(file.buffer); - result.thumbnails = []; - - for (const thumb of thumbnails) { - const thumbKey = `${content.contentId}/thumbnails/${thumb.name}.webp`; - await this.storage.uploadProcessedFile(thumb.buffer, thumbKey, 'image/webp'); - result.thumbnails.push({ - name: thumb.name, - url: `https://${process.env.AWS_S3_BUCKET}.s3.amazonaws.com/${thumbKey}`, - }); + return result; + } + catch (error) { + this.logger.error('Upload failed:', error); + if (options.trackProgress !== false) { + await this.uploadProgress.markFailed(uploadId, error instanceof Error ? error.message : 'Unknown error'); } - - // Store thumbnail URLs in variants - content.variants = result.thumbnails.map((t) => ({ - name: t.name, - url: t.url, - width: 0, // Will be populated from thumbnail data - height: 0, - size: 0, - })); - } - } catch (error) { - this.logger.warn('Image processing failed, using original:', error); + throw error; } - } - - // Step 4: Upload to storage - if (options.trackProgress !== false) { - await this.uploadProgress.updateProgress(uploadId, { - status: 'uploading', - progress: 80, - stage: 'storage-upload', - message: 'Uploading to storage...', - }); - } - - const upload = await this.storage.uploadFile(processedFile, content); - - content.originalUrl = upload.url; - content.cdnUrl = upload.url; - content.etag = upload.etag; - content.status = ContentStatus.READY; - await this.contentRepo.save(content); - - // Step 5: Video processing (if applicable) - if (content.contentType === ContentType.VIDEO) { - this.logger.log(`Enqueue video processing for ${content.contentId}`); - await this.videoProcessing.enqueueTranscode(content); - content.status = ContentStatus.PROCESSING; - await this.contentRepo.save(content); - } - - // Mark as completed - if (options.trackProgress !== false) { - await this.uploadProgress.markCompleted(uploadId, { - contentId: content.contentId, - url: content.cdnUrl, - thumbnails: result.thumbnails?.map((t) => t.url), - }); - } - - return result; - } catch (error) { - this.logger.error('Upload failed:', error); - - if (options.trackProgress !== false) { - await this.uploadProgress.markFailed( - uploadId, - error instanceof Error ? error.message : 'Unknown error', - ); - } - - throw error; } - } - - async findByContentId(contentId: string) { - return this.contentRepo.findOne({ where: { contentId } }); - } - - /** - * Get upload progress - */ - async getUploadProgress(uploadId: string) { - return this.uploadProgress.getProgress(uploadId); - } - - /** - * List active uploads - */ - async listActiveUploads() { - return this.uploadProgress.listActiveUploads(); - } - - /** - * Get upload statistics - */ - async getUploadStatistics() { - return this.uploadProgress.getStatistics(); - } - - private mapContentType(mime: string) { - if (mime.startsWith('image/')) return ContentType.IMAGE; - if (mime.startsWith('video/')) return ContentType.VIDEO; - if (mime.startsWith('audio/')) return ContentType.AUDIO; - return ContentType.DOCUMENT; - } + async findByContentId(contentId: string) { + return this.contentRepo.findOne({ where: { contentId } }); + } + /** + * Get upload progress + */ + async getUploadProgress(uploadId: string) { + return this.uploadProgress.getProgress(uploadId); + } + /** + * List active uploads + */ + async listActiveUploads() { + return this.uploadProgress.listActiveUploads(); + } + /** + * Get upload statistics + */ + async getUploadStatistics() { + return this.uploadProgress.getStatistics(); + } + private mapContentType(mime: string) { + if (mime.startsWith('image/')) + return ContentType.IMAGE; + if (mime.startsWith('video/')) + return ContentType.VIDEO; + if (mime.startsWith('audio/')) + return ContentType.AUDIO; + return ContentType.DOCUMENT; + } } diff --git a/src/media/processing/document-processing.service.ts b/src/media/processing/document-processing.service.ts index 2e884a36..d0c8eed1 100644 --- a/src/media/processing/document-processing.service.ts +++ b/src/media/processing/document-processing.service.ts @@ -4,47 +4,42 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { ContentMetadata } from '../../cdn/entities/content-metadata.entity'; import { FileStorageService } from '../storage/file-storage.service'; - @Injectable() export class DocumentProcessingService { - private readonly logger = new Logger(DocumentProcessingService.name); - - constructor( - private readonly storage: FileStorageService, + private readonly logger = new Logger(DocumentProcessingService.name); + constructor(private readonly storage: FileStorageService, @InjectRepository(ContentMetadata) - private readonly contentRepo: Repository, - ) {} - - async parsePdfFromContent(contentId: string) { - const meta = await this.contentRepo.findOne({ where: { contentId } }); - if (!meta) return null; - - const signed = await this.storage.getSignedUrl(meta.cdnUrl, 60); - const buffer = await downloadToBuffer(signed); - - try { - const parsed = await pdfParse(buffer); - meta.metadata = meta.metadata || {}; - // Extend metadata type to include text for documents - (meta.metadata as any).text = parsed.text; - await this.contentRepo.save(meta); - return parsed.text; - } catch (err) { - this.logger.error('PDF parsing failed', err); - throw err; + private readonly contentRepo: Repository) { } + async parsePdfFromContent(contentId: string) { + const meta = await this.contentRepo.findOne({ where: { contentId } }); + if (!meta) + return null; + const signed = await this.storage.getSignedUrl(meta.cdnUrl, 60); + const buffer = await downloadToBuffer(signed); + try { + const parsed = await pdfParse(buffer); + meta.metadata = meta.metadata || {}; + // Extend metadata type to include text for documents + (meta.metadata as unknown).text = parsed.text; + await this.contentRepo.save(meta); + return parsed.text; + } + catch (err) { + this.logger.error('PDF parsing failed', err); + throw err; + } } - } } - async function downloadToBuffer(url: string): Promise { - const https = url.startsWith('https') ? await import('https') : await import('http'); - return new Promise((resolve, reject) => { - const req = https.get(url, (res: any) => { - if (res.statusCode >= 400) return reject(new Error(`Failed to download: ${res.statusCode}`)); - const data: Buffer[] = []; - res.on('data', (chunk: Buffer) => data.push(chunk)); - res.on('end', () => resolve(Buffer.concat(data))); + const https = url.startsWith('https') ? await import('https') : await import('http'); + return new Promise((resolve, reject) => { + const req = https.get(url, (res: unknown) => { + if (res.statusCode >= 400) + return reject(new Error(`Failed to download: ${res.statusCode}`)); + const data: Buffer[] = []; + res.on('data', (chunk: Buffer) => data.push(chunk)); + res.on('end', () => resolve(Buffer.concat(data))); + }); + req.on('error', (err: Error) => reject(err)); }); - req.on('error', (err: Error) => reject(err)); - }); } diff --git a/src/media/processing/image-processing.service.ts b/src/media/processing/image-processing.service.ts index d5852f95..5c1203f9 100644 --- a/src/media/processing/image-processing.service.ts +++ b/src/media/processing/image-processing.service.ts @@ -1,373 +1,309 @@ import { Injectable, Logger } from '@nestjs/common'; import sharp from 'sharp'; -import { - COMPRESSION_CONFIG, - THUMBNAIL_CONFIG, - IMAGE_DIMENSION_LIMITS, -} from '../validation/file-validation.constants'; - +import { COMPRESSION_CONFIG, THUMBNAIL_CONFIG, IMAGE_DIMENSION_LIMITS, } from '../validation/file-validation.constants'; export interface ProcessedImage { - buffer: Buffer; - format: string; - width: number; - height: number; - size: number; - originalSize: number; - compressionRatio: number; + buffer: Buffer; + format: string; + width: number; + height: number; + size: number; + originalSize: number; + compressionRatio: number; } - export interface ThumbnailResult { - name: string; - buffer: Buffer; - width: number; - height: number; - size: number; - url?: string; + name: string; + buffer: Buffer; + width: number; + height: number; + size: number; + url?: string; } - export interface ImageMetadata { - width: number; - height: number; - format: string; - hasAlpha: boolean; - size: number; - density?: number; - space?: string; - channels?: number; + width: number; + height: number; + format: string; + hasAlpha: boolean; + size: number; + density?: number; + space?: string; + channels?: number; } - @Injectable() export class ImageProcessingService { - private readonly logger = new Logger(ImageProcessingService.name); - - /** - * Get image metadata - */ - async getMetadata(buffer: Buffer): Promise { - const metadata = await sharp(buffer).metadata(); - const stats = await sharp(buffer).stats(); - - return { - width: metadata.width || 0, - height: metadata.height || 0, - format: metadata.format || 'unknown', - hasAlpha: metadata.hasAlpha || false, - size: buffer.length, - density: metadata.density, - space: metadata.space, - channels: stats.channels.length, - }; - } - - /** - * Compress and optimize image - */ - async compressImage( - buffer: Buffer, - options?: { - maxWidth?: number; - maxHeight?: number; - quality?: number; - format?: 'jpeg' | 'png' | 'webp' | 'avif'; - preserveAspectRatio?: boolean; - }, - ): Promise { - const originalSize = buffer.length; - const metadata = await sharp(buffer).metadata(); - - let pipeline = sharp(buffer); - - // Determine output format - const outputFormat = options?.format || this.getOptimalFormat(metadata.format); - - // Resize if dimensions exceed limits - const maxWidth = options?.maxWidth || COMPRESSION_CONFIG.MAX_DIMENSION; - const maxHeight = options?.maxHeight || COMPRESSION_CONFIG.MAX_DIMENSION; - - if (metadata.width && metadata.height) { - if (metadata.width > maxWidth || metadata.height > maxHeight) { - pipeline = pipeline.resize(maxWidth, maxHeight, { - fit: 'inside', - withoutEnlargement: true, - }); - } - } - - // Apply format-specific compression - const quality = options?.quality || this.getDefaultQuality(outputFormat); - - switch (outputFormat) { - case 'jpeg': - case 'jpg': - pipeline = pipeline.jpeg({ - quality, - progressive: true, - mozjpeg: true, - }); - break; - - case 'png': - pipeline = pipeline.png({ - compressionLevel: COMPRESSION_CONFIG.PNG_COMPRESSION_LEVEL, - progressive: true, - adaptiveFiltering: true, - }); - break; - - case 'webp': - pipeline = pipeline.webp({ - quality, - effort: 6, - }); - break; - - case 'avif': - pipeline = pipeline.avif({ - quality, - effort: 4, - }); - break; - - default: - // Keep original format with default compression - pipeline = pipeline.jpeg({ quality: COMPRESSION_CONFIG.JPEG_QUALITY }); + private readonly logger = new Logger(ImageProcessingService.name); + /** + * Get image metadata + */ + async getMetadata(buffer: Buffer): Promise { + const metadata = await sharp(buffer).metadata(); + const stats = await sharp(buffer).stats(); + return { + width: metadata.width || 0, + height: metadata.height || 0, + format: metadata.format || 'unknown', + hasAlpha: metadata.hasAlpha || false, + size: buffer.length, + density: metadata.density, + space: metadata.space, + channels: stats.channels.length, + }; } - - // Process image - const processedBuffer = await pipeline.toBuffer(); - const processedMetadata = await sharp(processedBuffer).metadata(); - - const compressionRatio = - originalSize > 0 ? ((originalSize - processedBuffer.length) / originalSize) * 100 : 0; - - this.logger.log( - `Image compressed: ${originalSize} -> ${processedBuffer.length} bytes (${compressionRatio.toFixed(1)}% reduction)`, - ); - - return { - buffer: processedBuffer, - format: outputFormat, - width: processedMetadata.width || 0, - height: processedMetadata.height || 0, - size: processedBuffer.length, - originalSize, - compressionRatio, - }; - } - - /** - * Generate thumbnails in multiple sizes - */ - async generateThumbnails( - buffer: Buffer, - options?: { - sizes?: Array<{ name: string; width: number; height: number }>; - format?: 'jpeg' | 'png' | 'webp'; - quality?: number; - }, - ): Promise { - const sizes = options?.sizes || THUMBNAIL_CONFIG.SIZES; - const format = options?.format || THUMBNAIL_CONFIG.DEFAULT_FORMAT; - const quality = options?.quality || THUMBNAIL_CONFIG.DEFAULT_QUALITY; - - const thumbnails: ThumbnailResult[] = []; - - for (const size of sizes) { - try { + /** + * Compress and optimize image + */ + async compressImage(buffer: Buffer, options?: { + maxWidth?: number; + maxHeight?: number; + quality?: number; + format?: 'jpeg' | 'png' | 'webp' | 'avif'; + preserveAspectRatio?: boolean; + }): Promise { + const originalSize = buffer.length; + const metadata = await sharp(buffer).metadata(); let pipeline = sharp(buffer); - - // Resize to thumbnail dimensions - pipeline = pipeline.resize(size.width, size.height, { - fit: 'cover', - position: 'center', - }); - - // Apply format-specific settings - switch (format) { - case 'jpeg': - pipeline = pipeline.jpeg({ quality, progressive: true }); - break; - case 'png': - pipeline = pipeline.png({ compressionLevel: 9 }); - break; - case 'webp': - default: - pipeline = pipeline.webp({ quality, effort: 6 }); + // Determine output format + const outputFormat = options?.format || this.getOptimalFormat(metadata.format); + // Resize if dimensions exceed limits + const maxWidth = options?.maxWidth || COMPRESSION_CONFIG.MAX_DIMENSION; + const maxHeight = options?.maxHeight || COMPRESSION_CONFIG.MAX_DIMENSION; + if (metadata.width && metadata.height) { + if (metadata.width > maxWidth || metadata.height > maxHeight) { + pipeline = pipeline.resize(maxWidth, maxHeight, { + fit: 'inside', + withoutEnlargement: true, + }); + } } - - const thumbnailBuffer = await pipeline.toBuffer(); - const metadata = await sharp(thumbnailBuffer).metadata(); - - thumbnails.push({ - name: size.name, - buffer: thumbnailBuffer, - width: metadata.width || size.width, - height: metadata.height || size.height, - size: thumbnailBuffer.length, - }); - - this.logger.log(`Generated ${size.name} thumbnail: ${thumbnailBuffer.length} bytes`); - } catch (error) { - this.logger.error(`Failed to generate ${size.name} thumbnail:`, error); - } - } - - return thumbnails; - } - - /** - * Generate a single thumbnail - */ - async generateThumbnail( - buffer: Buffer, - width: number, - height: number, - options?: { - format?: 'jpeg' | 'png' | 'webp'; - quality?: number; - fit?: 'cover' | 'contain' | 'fill' | 'inside' | 'outside'; - }, - ): Promise { - const format = options?.format || THUMBNAIL_CONFIG.DEFAULT_FORMAT; - const quality = options?.quality || THUMBNAIL_CONFIG.DEFAULT_QUALITY; - const fit = options?.fit || 'cover'; - - let pipeline = sharp(buffer).resize(width, height, { - fit, - position: 'center', - }); - - switch (format) { - case 'jpeg': - pipeline = pipeline.jpeg({ quality, progressive: true }); - break; - case 'png': - pipeline = pipeline.png({ compressionLevel: 9 }); - break; - case 'webp': - default: - pipeline = pipeline.webp({ quality, effort: 6 }); + // Apply format-specific compression + const quality = options?.quality || this.getDefaultQuality(outputFormat); + switch (outputFormat) { + case 'jpeg': + case 'jpg': + pipeline = pipeline.jpeg({ + quality, + progressive: true, + mozjpeg: true, + }); + break; + case 'png': + pipeline = pipeline.png({ + compressionLevel: COMPRESSION_CONFIG.PNG_COMPRESSION_LEVEL, + progressive: true, + adaptiveFiltering: true, + }); + break; + case 'webp': + pipeline = pipeline.webp({ + quality, + effort: 6, + }); + break; + case 'avif': + pipeline = pipeline.avif({ + quality, + effort: 4, + }); + break; + default: + // Keep original format with default compression + pipeline = pipeline.jpeg({ quality: COMPRESSION_CONFIG.JPEG_QUALITY }); + } + // Process image + const processedBuffer = await pipeline.toBuffer(); + const processedMetadata = await sharp(processedBuffer).metadata(); + const compressionRatio = originalSize > 0 ? ((originalSize - processedBuffer.length) / originalSize) * 100 : 0; + this.logger.log(`Image compressed: ${originalSize} -> ${processedBuffer.length} bytes (${compressionRatio.toFixed(1)}% reduction)`); + return { + buffer: processedBuffer, + format: outputFormat, + width: processedMetadata.width || 0, + height: processedMetadata.height || 0, + size: processedBuffer.length, + originalSize, + compressionRatio, + }; } - - return pipeline.toBuffer(); - } - - /** - * Validate image dimensions - */ - async validateDimensions(buffer: Buffer): Promise<{ - valid: boolean; - width?: number; - height?: number; - errors: string[]; - }> { - const metadata = await sharp(buffer).metadata(); - const errors: string[] = []; - - if (!metadata.width || !metadata.height) { - return { - valid: false, - errors: ['Could not determine image dimensions'], - }; + /** + * Generate thumbnails in multiple sizes + */ + async generateThumbnails(buffer: Buffer, options?: { + sizes?: Array<{ + name: string; + width: number; + height: number; + }>; + format?: 'jpeg' | 'png' | 'webp'; + quality?: number; + }): Promise { + const sizes = options?.sizes || THUMBNAIL_CONFIG.SIZES; + const format = options?.format || THUMBNAIL_CONFIG.DEFAULT_FORMAT; + const quality = options?.quality || THUMBNAIL_CONFIG.DEFAULT_QUALITY; + const thumbnails: ThumbnailResult[] = []; + for (const size of sizes) { + try { + let pipeline = sharp(buffer); + // Resize to thumbnail dimensions + pipeline = pipeline.resize(size.width, size.height, { + fit: 'cover', + position: 'center', + }); + // Apply format-specific settings + switch (format) { + case 'jpeg': + pipeline = pipeline.jpeg({ quality, progressive: true }); + break; + case 'png': + pipeline = pipeline.png({ compressionLevel: 9 }); + break; + case 'webp': + default: + pipeline = pipeline.webp({ quality, effort: 6 }); + } + const thumbnailBuffer = await pipeline.toBuffer(); + const metadata = await sharp(thumbnailBuffer).metadata(); + thumbnails.push({ + name: size.name, + buffer: thumbnailBuffer, + width: metadata.width || size.width, + height: metadata.height || size.height, + size: thumbnailBuffer.length, + }); + this.logger.log(`Generated ${size.name} thumbnail: ${thumbnailBuffer.length} bytes`); + } + catch (error) { + this.logger.error(`Failed to generate ${size.name} thumbnail:`, error); + } + } + return thumbnails; } - - const { MIN_WIDTH, MIN_HEIGHT, MAX_WIDTH, MAX_HEIGHT, MAX_PIXELS } = IMAGE_DIMENSION_LIMITS; - - if (metadata.width < MIN_WIDTH || metadata.height < MIN_HEIGHT) { - errors.push(`Image dimensions too small (minimum: ${MIN_WIDTH}x${MIN_HEIGHT})`); + /** + * Generate a single thumbnail + */ + async generateThumbnail(buffer: Buffer, width: number, height: number, options?: { + format?: 'jpeg' | 'png' | 'webp'; + quality?: number; + fit?: 'cover' | 'contain' | 'fill' | 'inside' | 'outside'; + }): Promise { + const format = options?.format || THUMBNAIL_CONFIG.DEFAULT_FORMAT; + const quality = options?.quality || THUMBNAIL_CONFIG.DEFAULT_QUALITY; + const fit = options?.fit || 'cover'; + let pipeline = sharp(buffer).resize(width, height, { + fit, + position: 'center', + }); + switch (format) { + case 'jpeg': + pipeline = pipeline.jpeg({ quality, progressive: true }); + break; + case 'png': + pipeline = pipeline.png({ compressionLevel: 9 }); + break; + case 'webp': + default: + pipeline = pipeline.webp({ quality, effort: 6 }); + } + return pipeline.toBuffer(); } - - if (metadata.width > MAX_WIDTH || metadata.height > MAX_HEIGHT) { - errors.push(`Image dimensions too large (maximum: ${MAX_WIDTH}x${MAX_HEIGHT})`); + /** + * Validate image dimensions + */ + async validateDimensions(buffer: Buffer): Promise<{ + valid: boolean; + width?: number; + height?: number; + errors: string[]; + }> { + const metadata = await sharp(buffer).metadata(); + const errors: string[] = []; + if (!metadata.width || !metadata.height) { + return { + valid: false, + errors: ['Could not determine image dimensions'], + }; + } + const { MIN_WIDTH, MIN_HEIGHT, MAX_WIDTH, MAX_HEIGHT, MAX_PIXELS } = IMAGE_DIMENSION_LIMITS; + if (metadata.width < MIN_WIDTH || metadata.height < MIN_HEIGHT) { + errors.push(`Image dimensions too small (minimum: ${MIN_WIDTH}x${MIN_HEIGHT})`); + } + if (metadata.width > MAX_WIDTH || metadata.height > MAX_HEIGHT) { + errors.push(`Image dimensions too large (maximum: ${MAX_WIDTH}x${MAX_HEIGHT})`); + } + const totalPixels = metadata.width * metadata.height; + if (totalPixels > MAX_PIXELS) { + errors.push(`Image has too many pixels (maximum: ${MAX_PIXELS.toLocaleString()})`); + } + return { + valid: errors.length === 0, + width: metadata.width, + height: metadata.height, + errors, + }; } - - const totalPixels = metadata.width * metadata.height; - if (totalPixels > MAX_PIXELS) { - errors.push(`Image has too many pixels (maximum: ${MAX_PIXELS.toLocaleString()})`); + /** + * Convert image to different format + */ + async convertFormat(buffer: Buffer, targetFormat: 'jpeg' | 'png' | 'webp' | 'avif', quality?: number): Promise { + let pipeline = sharp(buffer); + const q = quality || this.getDefaultQuality(targetFormat); + switch (targetFormat) { + case 'jpeg': + pipeline = pipeline.jpeg({ quality: q, progressive: true }); + break; + case 'png': + pipeline = pipeline.png({ compressionLevel: COMPRESSION_CONFIG.PNG_COMPRESSION_LEVEL }); + break; + case 'webp': + pipeline = pipeline.webp({ quality: q, effort: 6 }); + break; + case 'avif': + pipeline = pipeline.avif({ quality: q, effort: 4 }); + break; + } + return pipeline.toBuffer(); } - - return { - valid: errors.length === 0, - width: metadata.width, - height: metadata.height, - errors, - }; - } - - /** - * Convert image to different format - */ - async convertFormat( - buffer: Buffer, - targetFormat: 'jpeg' | 'png' | 'webp' | 'avif', - quality?: number, - ): Promise { - let pipeline = sharp(buffer); - const q = quality || this.getDefaultQuality(targetFormat); - - switch (targetFormat) { - case 'jpeg': - pipeline = pipeline.jpeg({ quality: q, progressive: true }); - break; - case 'png': - pipeline = pipeline.png({ compressionLevel: COMPRESSION_CONFIG.PNG_COMPRESSION_LEVEL }); - break; - case 'webp': - pipeline = pipeline.webp({ quality: q, effort: 6 }); - break; - case 'avif': - pipeline = pipeline.avif({ quality: q, effort: 4 }); - break; + /** + * Strip metadata from image (privacy/security) + */ + async stripMetadata(buffer: Buffer): Promise { + return sharp(buffer).withMetadata().toBuffer(); } - - return pipeline.toBuffer(); - } - - /** - * Strip metadata from image (privacy/security) - */ - async stripMetadata(buffer: Buffer): Promise { - return sharp(buffer).withMetadata().toBuffer(); - } - - /** - * Get optimal format based on content type - */ - private getOptimalFormat(currentFormat?: string): string { - switch (currentFormat) { - case 'png': - return 'png'; // Keep PNG for transparency - case 'gif': - return 'webp'; // Convert GIF to WebP - case 'svg': - return 'png'; // Rasterize SVG - case 'webp': - return 'webp'; - case 'avif': - return 'avif'; - case 'jpeg': - case 'jpg': - default: - return 'webp'; // Default to WebP for best compression + /** + * Get optimal format based on content type + */ + private getOptimalFormat(currentFormat?: string): string { + switch (currentFormat) { + case 'png': + return 'png'; // Keep PNG for transparency + case 'gif': + return 'webp'; // Convert GIF to WebP + case 'svg': + return 'png'; // Rasterize SVG + case 'webp': + return 'webp'; + case 'avif': + return 'avif'; + case 'jpeg': + case 'jpg': + default: + return 'webp'; // Default to WebP for best compression + } } - } - - /** - * Get default quality for format - */ - private getDefaultQuality(format: string): number { - switch (format) { - case 'jpeg': - case 'jpg': - return COMPRESSION_CONFIG.JPEG_QUALITY; - case 'webp': - return COMPRESSION_CONFIG.WEBP_QUALITY; - case 'avif': - return COMPRESSION_CONFIG.AVIF_QUALITY; - case 'png': - return 100; // PNG uses compression level, not quality - default: - return 85; + /** + * Get default quality for format + */ + private getDefaultQuality(format: string): number { + switch (format) { + case 'jpeg': + case 'jpg': + return COMPRESSION_CONFIG.JPEG_QUALITY; + case 'webp': + return COMPRESSION_CONFIG.WEBP_QUALITY; + case 'avif': + return COMPRESSION_CONFIG.AVIF_QUALITY; + case 'png': + return 100; // PNG uses compression level, not quality + default: + return 85; + } } - } } diff --git a/src/media/processing/video-processing.service.ts b/src/media/processing/video-processing.service.ts index ef3cf2eb..37efdcf1 100644 --- a/src/media/processing/video-processing.service.ts +++ b/src/media/processing/video-processing.service.ts @@ -3,27 +3,22 @@ import { InjectQueue } from '@nestjs/bull'; import { Queue } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; import { ContentMetadata } from '../../cdn/entities/content-metadata.entity'; - @Injectable() export class VideoProcessingService { - private readonly logger = new Logger(VideoProcessingService.name); - - constructor(@InjectQueue(QUEUE_NAMES.MEDIA_PROCESSING) private readonly queue: Queue) {} - - async enqueueTranscode(content: ContentMetadata) { - await this.queue.add( - JOB_NAMES.TRANSCODE_VIDEO, - { - contentId: content.contentId, - url: content.cdnUrl, - fileName: content.fileName, - mimeType: content.mimeType, - }, - { - attempts: 3, - backoff: { type: 'exponential', delay: 5000 }, - }, - ); - this.logger.log(`Job enqueued for content ${content.contentId}`); - } + private readonly logger = new Logger(VideoProcessingService.name); + constructor( + @InjectQueue(QUEUE_NAMES.MEDIA_PROCESSING) + private readonly queue: Queue) { } + async enqueueTranscode(content: ContentMetadata) { + await this.queue.add(JOB_NAMES.TRANSCODE_VIDEO, { + contentId: content.contentId, + url: content.cdnUrl, + fileName: content.fileName, + mimeType: content.mimeType, + }, { + attempts: 3, + backoff: { type: 'exponential', delay: 5000 }, + }); + this.logger.log(`Job enqueued for content ${content.contentId}`); + } } diff --git a/src/media/processing/video.processor.ts b/src/media/processing/video.processor.ts index f3d570f6..075e2289 100644 --- a/src/media/processing/video.processor.ts +++ b/src/media/processing/video.processor.ts @@ -11,125 +11,113 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { UploadedFile } from '../../common/types/file.types'; import { ContentMetadata } from '../../cdn/entities/content-metadata.entity'; - @Processor(QUEUE_NAMES.MEDIA_PROCESSING) export class VideoProcessor { - private readonly logger = new Logger(VideoProcessor.name); - - constructor( - private readonly storage: FileStorageService, + private readonly logger = new Logger(VideoProcessor.name); + constructor(private readonly storage: FileStorageService, @InjectRepository(ContentMetadata) - private readonly contentRepo: Repository, - ) {} - - @Process(JOB_NAMES.TRANSCODE_VIDEO) - async handleTranscode(job: Job) { - const { contentId, url, fileName } = job.data as { - contentId: string; - url: string; - fileName: string; - }; - this.logger.log(`Transcoding job for ${contentId} - ${url}`); - - const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), `media-${contentId}-`)); - const inputPath = path.join(tmpDir, fileName); - - // Download the source - const signed = await this.storage.getSignedUrl(url, 60); - await downloadToFile(signed, inputPath); - - // Produce HLS with 3 variants (1080p,720p,480p) - const hlsDir = path.join(tmpDir, 'hls'); - fs.mkdirSync(hlsDir); - - await new Promise((resolve, reject) => { - ffmpeg(inputPath) - .addOption('-preset', 'fast') - .addOption('-g', '48') - .addOption('-sc_threshold', '0') - .outputOptions(['-map 0:v', '-map 0:a?', '-c:a aac', '-c:v h264', '-profile:v main']) - .output(path.join(hlsDir, 'index.m3u8')) - .on('end', () => resolve()) - .on('error', (err) => reject(err)) - .run(); - }); - - // Upload HLS directory contents - const files = fs.readdirSync(hlsDir); - const uploaded: string[] = []; - for (const f of files) { - const p = path.join(hlsDir, f); - const buffer = fs.readFileSync(p); - // store each file under contentId/hls/ - const fakeFile: UploadedFile = { - buffer, - originalname: f, - mimetype: 'application/octet-stream', - size: buffer.length, - fieldname: 'file', - encoding: '7bit', - destination: '', - filename: f, - stream: null as any, - path: p, - }; - const keyRes = await this.storage.uploadFile(fakeFile as any, { contentId } as any); - uploaded.push(keyRes.url); + private readonly contentRepo: Repository) { } + @Process(JOB_NAMES.TRANSCODE_VIDEO) + async handleTranscode(job: Job) { + const { contentId, url, fileName } = job.data as { + contentId: string; + url: string; + fileName: string; + }; + this.logger.log(`Transcoding job for ${contentId} - ${url}`); + const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), `media-${contentId}-`)); + const inputPath = path.join(tmpDir, fileName); + // Download the source + const signed = await this.storage.getSignedUrl(url, 60); + await downloadToFile(signed, inputPath); + // Produce HLS with 3 variants (1080p,720p,480p) + const hlsDir = path.join(tmpDir, 'hls'); + fs.mkdirSync(hlsDir); + await new Promise((resolve, reject) => { + ffmpeg(inputPath) + .addOption('-preset', 'fast') + .addOption('-g', '48') + .addOption('-sc_threshold', '0') + .outputOptions(['-map 0:v', '-map 0:a?', '-c:a aac', '-c:v h264', '-profile:v main']) + .output(path.join(hlsDir, 'index.m3u8')) + .on('end', () => resolve()) + .on('error', (err) => reject(err)) + .run(); + }); + // Upload HLS directory contents + const files = fs.readdirSync(hlsDir); + const uploaded: string[] = []; + for (const f of files) { + const p = path.join(hlsDir, f); + const buffer = fs.readFileSync(p); + // store each file under contentId/hls/ + const fakeFile: UploadedFile = { + buffer, + originalname: f, + mimetype: 'application/octet-stream', + size: buffer.length, + fieldname: 'file', + encoding: '7bit', + destination: '', + filename: f, + stream: null as unknown, + path: p, + }; + const keyRes = await this.storage.uploadFile(fakeFile as unknown, { contentId } as unknown); + uploaded.push(keyRes.url); + } + // Update metadata + const meta = await this.contentRepo.findOne({ where: { contentId } }); + if (meta) { + meta.metadata = meta.metadata || {}; + // Extend metadata type to include hlsManifest for videos + (meta.metadata as unknown).hlsManifest = + uploaded.find((u) => u.endsWith('index.m3u8')) || uploaded[0]; + meta.variants = uploaded.map((u) => ({ + name: u.split('/').pop(), + url: u, + width: 0, + height: 0, + size: 0, + })); + meta.status = 'ready' as unknown; + await this.contentRepo.save(meta); + } + // Cleanup + try { + fs.rmSync(tmpDir, { recursive: true, force: true }); + } + catch (e) { + this.logger.warn('Failed to clean tmpdir', e); + } + return { uploaded }; } - - // Update metadata - const meta = await this.contentRepo.findOne({ where: { contentId } }); - if (meta) { - meta.metadata = meta.metadata || {}; - // Extend metadata type to include hlsManifest for videos - (meta.metadata as any).hlsManifest = - uploaded.find((u) => u.endsWith('index.m3u8')) || uploaded[0]; - meta.variants = uploaded.map((u) => ({ - name: u.split('/').pop(), - url: u, - width: 0, - height: 0, - size: 0, - })); - meta.status = 'ready' as any; - await this.contentRepo.save(meta); + @OnQueueFailed() + async onFailed(job: Job, err: Error) { + this.logger.error(`Job ${job.id} failed: ${err.message}`); } - - // Cleanup - try { - fs.rmSync(tmpDir, { recursive: true, force: true }); - } catch (e) { - this.logger.warn('Failed to clean tmpdir', e); + @OnQueueCompleted() + async onComplete(job: Job, _result: unknown) { + this.logger.log(`Job ${job.id} completed`); } - - return { uploaded }; - } - - @OnQueueFailed() - async onFailed(job: Job, err: Error) { - this.logger.error(`Job ${job.id} failed: ${err.message}`); - } - - @OnQueueCompleted() - async onComplete(job: Job, _result: any) { - this.logger.log(`Job ${job.id} completed`); - } } - async function downloadToFile(url: string, dest: string): Promise { - const https = url.startsWith('https') ? await import('https') : await import('http'); - return new Promise((resolve, reject) => { - const file = fs.createWriteStream(dest); - const req = https.get(url, (res: any) => { - if (res.statusCode >= 400) return reject(new Error(`Failed to download: ${res.statusCode}`)); - res.pipe(file); - file.on('finish', () => { - file.close((err?: NodeJS.ErrnoException | null) => { - if (err) reject(err); - else resolve(); + const https = url.startsWith('https') ? await import('https') : await import('http'); + return new Promise((resolve, reject) => { + const file = fs.createWriteStream(dest); + const req = https.get(url, (res: unknown) => { + if (res.statusCode >= 400) + return reject(new Error(`Failed to download: ${res.statusCode}`)); + res.pipe(file); + file.on('finish', () => { + file.close((err?: NodeJS.ErrnoException | null) => { + if (err) + reject(err); + else + resolve(); + }); + }); }); - }); + req.on('error', (err: Error) => reject(err)); }); - req.on('error', (err: Error) => reject(err)); - }); } diff --git a/src/media/storage/file-storage.service.ts b/src/media/storage/file-storage.service.ts index bda50f09..9ccea4d0 100644 --- a/src/media/storage/file-storage.service.ts +++ b/src/media/storage/file-storage.service.ts @@ -1,96 +1,77 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; -import { - S3Client, - GetObjectCommand, - PutObjectCommand, - DeleteObjectCommand, -} from '@aws-sdk/client-s3'; +import { S3Client, GetObjectCommand, PutObjectCommand, DeleteObjectCommand, } from '@aws-sdk/client-s3'; import { Readable } from 'stream'; import { ContentMetadata } from '../../cdn/entities/content-metadata.entity'; import { UploadedFile } from '../../common/types/file.types'; - @Injectable() export class FileStorageService { - private readonly logger = new Logger(FileStorageService.name); - private readonly s3Client: S3Client; - private readonly bucketName: string; - - constructor(private configService: ConfigService) { - this.bucketName = this.configService.get('AWS_S3_BUCKET', ''); - - this.s3Client = new S3Client({ - region: this.configService.get('AWS_REGION', 'us-east-1'), - credentials: { - accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), - secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), - }, - }); - } - - // Legacy method for backward compatibility - async uploadFile( - file: UploadedFile, - metadata: ContentMetadata, - ): Promise<{ url: string; etag?: string }> { - const key = `${metadata.contentId}/${Date.now()}_${file.originalname}`; - await this.uploadProcessedFile(file.buffer, key, file.mimetype); - return { - url: `https://${this.bucketName}.s3.amazonaws.com/${key}`, - etag: undefined, - }; - } - - // Legacy method for backward compatibility - async getSignedUrl(keyOrUrl: string, _expiresSec = 300): Promise { - // If a full URL is provided, return as-is - if (keyOrUrl.startsWith('http')) return keyOrUrl; - - const _command = new GetObjectCommand({ - Bucket: this.bucketName, - Key: keyOrUrl, - }); - - // For simplicity, return the key as URL (in production, generate proper signed URL) - return `https://${this.bucketName}.s3.amazonaws.com/${keyOrUrl}`; - } - - async uploadProcessedFile(buffer: Buffer, key: string, contentType: string): Promise { - const command = new PutObjectCommand({ - Bucket: this.bucketName, - Key: key, - Body: buffer, - ContentType: contentType, - }); - - await this.s3Client.send(command); - this.logger.log(`Uploaded file to ${key}`); - } - - async downloadFile(storageKey: string): Promise { - const command = new GetObjectCommand({ - Bucket: this.bucketName, - Key: storageKey, - }); - - const response = await this.s3Client.send(command); - const stream = response.Body as Readable; - const chunks: Buffer[] = []; - - for await (const chunk of stream) { - chunks.push(chunk); + private readonly logger = new Logger(FileStorageService.name); + private readonly s3Client: S3Client; + private readonly bucketName: string; + constructor(private configService: ConfigService) { + this.bucketName = this.configService.get('AWS_S3_BUCKET', ''); + this.s3Client = new S3Client({ + region: this.configService.get('AWS_REGION', 'us-east-1'), + credentials: { + accessKeyId: this.configService.get('AWS_ACCESS_KEY_ID', ''), + secretAccessKey: this.configService.get('AWS_SECRET_ACCESS_KEY', ''), + }, + }); + } + // Legacy method for backward compatibility + async uploadFile(file: UploadedFile, metadata: ContentMetadata): Promise<{ + url: string; + etag?: string; + }> { + const key = `${metadata.contentId}/${Date.now()}_${file.originalname}`; + await this.uploadProcessedFile(file.buffer, key, file.mimetype); + return { + url: `https://${this.bucketName}.s3.amazonaws.com/${key}`, + etag: undefined, + }; + } + // Legacy method for backward compatibility + async getSignedUrl(keyOrUrl: string, _expiresSec = 300): Promise { + // If a full URL is provided, return as-is + if (keyOrUrl.startsWith('http')) + return keyOrUrl; + const _command = new GetObjectCommand({ + Bucket: this.bucketName, + Key: keyOrUrl, + }); + // For simplicity, return the key as URL (in production, generate proper signed URL) + return `https://${this.bucketName}.s3.amazonaws.com/${keyOrUrl}`; + } + async uploadProcessedFile(buffer: Buffer, key: string, contentType: string): Promise { + const command = new PutObjectCommand({ + Bucket: this.bucketName, + Key: key, + Body: buffer, + ContentType: contentType, + }); + await this.s3Client.send(command); + this.logger.log(`Uploaded file to ${key}`); + } + async downloadFile(storageKey: string): Promise { + const command = new GetObjectCommand({ + Bucket: this.bucketName, + Key: storageKey, + }); + const response = await this.s3Client.send(command); + const stream = response.Body as Readable; + const chunks: Buffer[] = []; + for await (const chunk of stream) { + chunks.push(chunk); + } + return Buffer.concat(chunks); + } + async deleteFile(storageKey: string): Promise { + const command = new DeleteObjectCommand({ + Bucket: this.bucketName, + Key: storageKey, + }); + await this.s3Client.send(command); + this.logger.log(`Deleted file ${storageKey}`); } - - return Buffer.concat(chunks); - } - - async deleteFile(storageKey: string): Promise { - const command = new DeleteObjectCommand({ - Bucket: this.bucketName, - Key: storageKey, - }); - - await this.s3Client.send(command); - this.logger.log(`Deleted file ${storageKey}`); - } } diff --git a/src/media/validation/file-validation.constants.ts b/src/media/validation/file-validation.constants.ts index 829458c1..dbeca814 100644 --- a/src/media/validation/file-validation.constants.ts +++ b/src/media/validation/file-validation.constants.ts @@ -1,165 +1,136 @@ export const ALLOWED_FILE_TYPES = { - // Images - IMAGES: [ - 'image/jpeg', - 'image/jpg', - 'image/png', - 'image/gif', - 'image/webp', - 'image/svg+xml', - 'image/bmp', - 'image/tiff', - 'image/avif', - ], - - // Videos - VIDEOS: [ - 'video/mp4', - 'video/webm', - 'video/ogg', - 'video/quicktime', - 'video/x-msvideo', - 'video/x-matroska', - 'video/mpeg', - ], - - // Documents - DOCUMENTS: [ - 'application/pdf', - 'application/msword', - 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', - 'application/vnd.ms-excel', - 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', - 'application/vnd.ms-powerpoint', - 'application/vnd.openxmlformats-officedocument.presentationml.presentation', - 'text/plain', - 'text/csv', - 'application/rtf', - ], - - // Audio - AUDIO: [ - 'audio/mpeg', - 'audio/mp3', - 'audio/wav', - 'audio/ogg', - 'audio/aac', - 'audio/webm', - 'audio/flac', - ], - - // Archives (limited) - ARCHIVES: ['application/zip', 'application/x-zip-compressed'], + // Images + IMAGES: [ + 'image/jpeg', + 'image/jpg', + 'image/png', + 'image/gif', + 'image/webp', + 'image/svg+xml', + 'image/bmp', + 'image/tiff', + 'image/avif', + ], + // Videos + VIDEOS: [ + 'video/mp4', + 'video/webm', + 'video/ogg', + 'video/quicktime', + 'video/x-msvideo', + 'video/x-matroska', + 'video/mpeg', + ], + // Documents + DOCUMENTS: [ + 'application/pdf', + 'application/msword', + 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', + 'application/vnd.ms-excel', + 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', + 'application/vnd.ms-powerpoint', + 'application/vnd.openxmlformats-officedocument.presentationml.presentation', + 'text/plain', + 'text/csv', + 'application/rtf', + ], + // Audio + AUDIO: [ + 'audio/mpeg', + 'audio/mp3', + 'audio/wav', + 'audio/ogg', + 'audio/aac', + 'audio/webm', + 'audio/flac', + ], + // Archives (limited) + ARCHIVES: ['application/zip', 'application/x-zip-compressed'], } as const; - export const ALLOWED_EXTENSIONS = { - IMAGES: ['.jpg', '.jpeg', '.png', '.gif', '.webp', '.svg', '.bmp', '.tiff', '.avif'], - VIDEOS: ['.mp4', '.webm', '.ogg', '.mov', '.avi', '.mkv', '.mpeg'], - DOCUMENTS: ['.pdf', '.doc', '.docx', '.xls', '.xlsx', '.ppt', '.pptx', '.txt', '.csv', '.rtf'], - AUDIO: ['.mp3', '.wav', '.ogg', '.aac', '.webm', '.flac'], - ARCHIVES: ['.zip'], + IMAGES: ['.jpg', '.jpeg', '.png', '.gif', '.webp', '.svg', '.bmp', '.tiff', '.avif'], + VIDEOS: ['.mp4', '.webm', '.ogg', '.mov', '.avi', '.mkv', '.mpeg'], + DOCUMENTS: ['.pdf', '.doc', '.docx', '.xls', '.xlsx', '.ppt', '.pptx', '.txt', '.csv', '.rtf'], + AUDIO: ['.mp3', '.wav', '.ogg', '.aac', '.webm', '.flac'], + ARCHIVES: ['.zip'], } as const; - export const FILE_SIZE_LIMITS = { - // Images: 20MB - IMAGE_MAX_SIZE: 20 * 1024 * 1024, - - // Videos: 500MB - VIDEO_MAX_SIZE: 500 * 1024 * 1024, - - // Documents: 50MB - DOCUMENT_MAX_SIZE: 50 * 1024 * 1024, - - // Audio: 100MB - AUDIO_MAX_SIZE: 100 * 1024 * 1024, - - // Archives: 100MB - ARCHIVE_MAX_SIZE: 100 * 1024 * 1024, - - // Default: 10MB - DEFAULT_MAX_SIZE: 10 * 1024 * 1024, + // Images: 20MB + IMAGE_MAX_SIZE: 20 * 1024 * 1024, + // Videos: 500MB + VIDEO_MAX_SIZE: 500 * 1024 * 1024, + // Documents: 50MB + DOCUMENT_MAX_SIZE: 50 * 1024 * 1024, + // Audio: 100MB + AUDIO_MAX_SIZE: 100 * 1024 * 1024, + // Archives: 100MB + ARCHIVE_MAX_SIZE: 100 * 1024 * 1024, + // Default: 10MB + DEFAULT_MAX_SIZE: 10 * 1024 * 1024, } as const; - export const ALL_ALLOWED_FILE_TYPES = Object.values(ALLOWED_FILE_TYPES).flat(); -export const MAX_UPLOAD_FILE_SIZE = Math.max( - FILE_SIZE_LIMITS.IMAGE_MAX_SIZE, - FILE_SIZE_LIMITS.VIDEO_MAX_SIZE, - FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE, - FILE_SIZE_LIMITS.AUDIO_MAX_SIZE, - FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE, -); - +export const MAX_UPLOAD_FILE_SIZE = Math.max(FILE_SIZE_LIMITS.IMAGE_MAX_SIZE, FILE_SIZE_LIMITS.VIDEO_MAX_SIZE, FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE, FILE_SIZE_LIMITS.AUDIO_MAX_SIZE, FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE); export const IMAGE_DIMENSION_LIMITS = { - MIN_WIDTH: 1, - MIN_HEIGHT: 1, - MAX_WIDTH: 16384, - MAX_HEIGHT: 16384, - MAX_PIXELS: 100_000_000, // 100 megapixels + MIN_WIDTH: 1, + MIN_HEIGHT: 1, + MAX_WIDTH: 16384, + MAX_HEIGHT: 16384, + MAX_PIXELS: 100000000, // 100 megapixels } as const; - export const THUMBNAIL_CONFIG = { - SIZES: [ - { name: 'thumbnail', width: 150, height: 150 }, - { name: 'small', width: 300, height: 300 }, - { name: 'medium', width: 600, height: 600 }, - { name: 'large', width: 1200, height: 1200 }, - ], - DEFAULT_FORMAT: 'webp' as const, - DEFAULT_QUALITY: 80, + SIZES: [ + { name: 'thumbnail', width: 150, height: 150 }, + { name: 'small', width: 300, height: 300 }, + { name: 'medium', width: 600, height: 600 }, + { name: 'large', width: 1200, height: 1200 }, + ], + DEFAULT_FORMAT: 'webp' as const, + DEFAULT_QUALITY: 80, } as const; - export const COMPRESSION_CONFIG = { - JPEG_QUALITY: 85, - PNG_COMPRESSION_LEVEL: 9, - WEBP_QUALITY: 85, - AVIF_QUALITY: 80, - MAX_DIMENSION: 4096, + JPEG_QUALITY: 85, + PNG_COMPRESSION_LEVEL: 9, + WEBP_QUALITY: 85, + AVIF_QUALITY: 80, + MAX_DIMENSION: 4096, } as const; - export const MALWARE_SCAN_CONFIG = { - MAX_FILE_SIZE_FOR_SCAN: 100 * 1024 * 1024, // 100MB - TIMEOUT_MS: 30000, // 30 seconds - RETRY_ATTEMPTS: 3, + MAX_FILE_SIZE_FOR_SCAN: 100 * 1024 * 1024, // 100MB + TIMEOUT_MS: 30000, // 30 seconds + RETRY_ATTEMPTS: 3, } as const; - export const UPLOAD_PROGRESS_CONFIG = { - REDIS_KEY_PREFIX: 'upload:progress:', - EXPIRY_SECONDS: 3600, // 1 hour - UPDATE_INTERVAL_MS: 500, // Update every 500ms + REDIS_KEY_PREFIX: 'upload:progress:', + EXPIRY_SECONDS: 3600, // 1 hour + UPDATE_INTERVAL_MS: 500, // Update every 500ms } as const; - export const MAGIC_NUMBERS: Record = { - // Images - 'image/jpeg': [Buffer.from([0xff, 0xd8, 0xff])], - 'image/png': [Buffer.from([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])], - 'image/gif': [ - Buffer.from([0x47, 0x49, 0x46, 0x38, 0x37, 0x61]), - Buffer.from([0x47, 0x49, 0x46, 0x38, 0x39, 0x61]), - ], - 'image/webp': [Buffer.from([0x52, 0x49, 0x46, 0x46])], // RIFF header - 'image/bmp': [Buffer.from([0x42, 0x4d])], // BM - 'image/tiff': [Buffer.from([0x49, 0x49, 0x2a, 0x00]), Buffer.from([0x4d, 0x4d, 0x00, 0x2a])], - - // PDF - 'application/pdf': [Buffer.from([0x25, 0x50, 0x44, 0x46])], // %PDF - - // ZIP (also for docx, xlsx, pptx) - 'application/zip': [Buffer.from([0x50, 0x4b, 0x03, 0x04])], - - // MP4 - 'video/mp4': [ - Buffer.from([0x00, 0x00, 0x00, 0x18, 0x66, 0x74, 0x79, 0x70]), - Buffer.from([0x00, 0x00, 0x00, 0x20, 0x66, 0x74, 0x79, 0x70]), - ], - - // MP3 - 'audio/mpeg': [ - Buffer.from([0xff, 0xfb]), - Buffer.from([0xff, 0xf3]), - Buffer.from([0xff, 0xf2]), - Buffer.from([0x49, 0x44, 0x33]), - ], - - // WAV - 'audio/wav': [Buffer.from([0x52, 0x49, 0x46, 0x46])], // RIFF + // Images + 'image/jpeg': [Buffer.from([0xff, 0xd8, 0xff])], + 'image/png': [Buffer.from([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])], + 'image/gif': [ + Buffer.from([0x47, 0x49, 0x46, 0x38, 0x37, 0x61]), + Buffer.from([0x47, 0x49, 0x46, 0x38, 0x39, 0x61]), + ], + 'image/webp': [Buffer.from([0x52, 0x49, 0x46, 0x46])], // RIFF header + 'image/bmp': [Buffer.from([0x42, 0x4d])], // BM + 'image/tiff': [Buffer.from([0x49, 0x49, 0x2a, 0x00]), Buffer.from([0x4d, 0x4d, 0x00, 0x2a])], + // PDF + 'application/pdf': [Buffer.from([0x25, 0x50, 0x44, 0x46])], // %PDF + // ZIP (also for docx, xlsx, pptx) + 'application/zip': [Buffer.from([0x50, 0x4b, 0x03, 0x04])], + // MP4 + 'video/mp4': [ + Buffer.from([0x00, 0x00, 0x00, 0x18, 0x66, 0x74, 0x79, 0x70]), + Buffer.from([0x00, 0x00, 0x00, 0x20, 0x66, 0x74, 0x79, 0x70]), + ], + // MP3 + 'audio/mpeg': [ + Buffer.from([0xff, 0xfb]), + Buffer.from([0xff, 0xf3]), + Buffer.from([0xff, 0xf2]), + Buffer.from([0x49, 0x44, 0x33]), + ], + // WAV + 'audio/wav': [Buffer.from([0x52, 0x49, 0x46, 0x46])], // RIFF } as const; diff --git a/src/media/validation/file-validation.service.ts b/src/media/validation/file-validation.service.ts index 05ab1994..25f83267 100644 --- a/src/media/validation/file-validation.service.ts +++ b/src/media/validation/file-validation.service.ts @@ -1,432 +1,374 @@ import { Injectable, Logger, BadRequestException } from '@nestjs/common'; import sharp from 'sharp'; import * as path from 'path'; -import { - ALLOWED_FILE_TYPES, - ALLOWED_EXTENSIONS, - FILE_SIZE_LIMITS, - IMAGE_DIMENSION_LIMITS, - MAGIC_NUMBERS, -} from './file-validation.constants'; +import { ALLOWED_FILE_TYPES, ALLOWED_EXTENSIONS, FILE_SIZE_LIMITS, IMAGE_DIMENSION_LIMITS, MAGIC_NUMBERS, } from './file-validation.constants'; import { UploadedFile } from '../../common/types/file.types'; - export interface FileValidationResult { - valid: boolean; - mimeType: string; - fileType: 'image' | 'video' | 'document' | 'audio' | 'archive' | 'unknown'; - size: number; - maxSize: number; - errors: string[]; - warnings: string[]; - metadata?: { - width?: number; - height?: number; - format?: string; - hasAlpha?: boolean; - }; + valid: boolean; + mimeType: string; + fileType: 'image' | 'video' | 'document' | 'audio' | 'archive' | 'unknown'; + size: number; + maxSize: number; + errors: string[]; + warnings: string[]; + metadata?: { + width?: number; + height?: number; + format?: string; + hasAlpha?: boolean; + }; } - export interface ImageDimensions { - width: number; - height: number; + width: number; + height: number; } - @Injectable() export class FileValidationService { - private readonly logger = new Logger(FileValidationService.name); - - private readonly allAllowedTypes: string[] = [ - ...ALLOWED_FILE_TYPES.IMAGES, - ...ALLOWED_FILE_TYPES.VIDEOS, - ...ALLOWED_FILE_TYPES.DOCUMENTS, - ...ALLOWED_FILE_TYPES.AUDIO, - ...ALLOWED_FILE_TYPES.ARCHIVES, - ]; - - private readonly allAllowedExtensions: string[] = [ - ...ALLOWED_EXTENSIONS.IMAGES, - ...ALLOWED_EXTENSIONS.VIDEOS, - ...ALLOWED_EXTENSIONS.DOCUMENTS, - ...ALLOWED_EXTENSIONS.AUDIO, - ...ALLOWED_EXTENSIONS.ARCHIVES, - ]; - - /** - * Validate file comprehensively - */ - async validateFile(file: UploadedFile): Promise { - const errors: string[] = []; - const warnings: string[] = []; - - // 1. Check if file exists - if (!file || !file.buffer || file.buffer.length === 0) { - return { - valid: false, - mimeType: '', - fileType: 'unknown', - size: 0, - maxSize: 0, - errors: ['No file provided or file is empty'], - warnings: [], - }; - } - - // 2. Validate MIME type against whitelist - const mimeValidation = this.validateMimeType(file.mimetype); - if (!mimeValidation.valid) { - errors.push(`File type "${file.mimetype}" is not allowed`); - } - - // 3. Validate file extension - const extValidation = this.validateExtension(file.originalname); - if (!extValidation.valid) { - errors.push(`File extension "${extValidation.extension}" is not allowed`); - } - - // 4. Validate magic numbers (file signature) - const signatureValidation = await this.validateFileSignature(file); - if (!signatureValidation.valid) { - errors.push('File signature does not match the declared type. Possible spoofing attempt.'); - } - - // 5. Get file type category - const fileType = this.getFileType(file.mimetype); - - // 6. Validate file size - const sizeValidation = this.validateFileSize(file.size, fileType); - if (!sizeValidation.valid) { - errors.push( - `File size ${this.formatBytes(file.size)} exceeds maximum allowed ${this.formatBytes(sizeValidation.maxSize)}`, - ); - } - - // 7. Validate image dimensions if it's an image - let imageMetadata: - | { width?: number; height?: number; format?: string; hasAlpha?: boolean } - | undefined; - if (fileType === 'image') { - const dimValidation = await this.validateImageDimensions(file.buffer); - if (!dimValidation.valid && dimValidation.dimensions) { - errors.push( - `Image dimensions (${dimValidation.dimensions.width}x${dimValidation.dimensions.height}) are outside allowed limits`, - ); - } - if (dimValidation.dimensions) { - imageMetadata = { - width: dimValidation.dimensions.width, - height: dimValidation.dimensions.height, - format: dimValidation.dimensions.format, - hasAlpha: dimValidation.dimensions.hasAlpha, + private readonly logger = new Logger(FileValidationService.name); + private readonly allAllowedTypes: string[] = [ + ...ALLOWED_FILE_TYPES.IMAGES, + ...ALLOWED_FILE_TYPES.VIDEOS, + ...ALLOWED_FILE_TYPES.DOCUMENTS, + ...ALLOWED_FILE_TYPES.AUDIO, + ...ALLOWED_FILE_TYPES.ARCHIVES, + ]; + private readonly allAllowedExtensions: string[] = [ + ...ALLOWED_EXTENSIONS.IMAGES, + ...ALLOWED_EXTENSIONS.VIDEOS, + ...ALLOWED_EXTENSIONS.DOCUMENTS, + ...ALLOWED_EXTENSIONS.AUDIO, + ...ALLOWED_EXTENSIONS.ARCHIVES, + ]; + /** + * Validate file comprehensively + */ + async validateFile(file: UploadedFile): Promise { + const errors: string[] = []; + const warnings: string[] = []; + // 1. Check if file exists + if (!file || !file.buffer || file.buffer.length === 0) { + return { + valid: false, + mimeType: '', + fileType: 'unknown', + size: 0, + maxSize: 0, + errors: ['No file provided or file is empty'], + warnings: [], + }; + } + // 2. Validate MIME type against whitelist + const mimeValidation = this.validateMimeType(file.mimetype); + if (!mimeValidation.valid) { + errors.push(`File type "${file.mimetype}" is not allowed`); + } + // 3. Validate file extension + const extValidation = this.validateExtension(file.originalname); + if (!extValidation.valid) { + errors.push(`File extension "${extValidation.extension}" is not allowed`); + } + // 4. Validate magic numbers (file signature) + const signatureValidation = await this.validateFileSignature(file); + if (!signatureValidation.valid) { + errors.push('File signature does not match the declared type. Possible spoofing attempt.'); + } + // 5. Get file type category + const fileType = this.getFileType(file.mimetype); + // 6. Validate file size + const sizeValidation = this.validateFileSize(file.size, fileType); + if (!sizeValidation.valid) { + errors.push(`File size ${this.formatBytes(file.size)} exceeds maximum allowed ${this.formatBytes(sizeValidation.maxSize)}`); + } + // 7. Validate image dimensions if it's an image + let imageMetadata: { + width?: number; + height?: number; + format?: string; + hasAlpha?: boolean; + } | undefined; + if (fileType === 'image') { + const dimValidation = await this.validateImageDimensions(file.buffer); + if (!dimValidation.valid && dimValidation.dimensions) { + errors.push(`Image dimensions (${dimValidation.dimensions.width}x${dimValidation.dimensions.height}) are outside allowed limits`); + } + if (dimValidation.dimensions) { + imageMetadata = { + width: dimValidation.dimensions.width, + height: dimValidation.dimensions.height, + format: dimValidation.dimensions.format, + hasAlpha: dimValidation.dimensions.hasAlpha, + }; + } + if (dimValidation.warnings) { + warnings.push(...dimValidation.warnings); + } + } + // 8. Check for suspicious patterns + const securityCheck = await this.checkSecurityPatterns(file); + if (!securityCheck.valid) { + errors.push(...securityCheck.issues); + } + return { + valid: errors.length === 0, + mimeType: file.mimetype, + fileType, + size: file.size, + maxSize: sizeValidation.maxSize, + errors, + warnings, + metadata: imageMetadata, }; - } - if (dimValidation.warnings) { - warnings.push(...dimValidation.warnings); - } } - - // 8. Check for suspicious patterns - const securityCheck = await this.checkSecurityPatterns(file); - if (!securityCheck.valid) { - errors.push(...securityCheck.issues); + /** + * Validate MIME type against whitelist + */ + private validateMimeType(mimeType: string): { + valid: boolean; + } { + return { + valid: this.allAllowedTypes.includes(mimeType.toLowerCase()), + }; } - - return { - valid: errors.length === 0, - mimeType: file.mimetype, - fileType, - size: file.size, - maxSize: sizeValidation.maxSize, - errors, - warnings, - metadata: imageMetadata, - }; - } - - /** - * Validate MIME type against whitelist - */ - private validateMimeType(mimeType: string): { valid: boolean } { - return { - valid: this.allAllowedTypes.includes(mimeType.toLowerCase()), - }; - } - - /** - * Validate file extension - */ - private validateExtension(filename: string): { valid: boolean; extension: string } { - const ext = path.extname(filename).toLowerCase(); - return { - valid: this.allAllowedExtensions.includes(ext), - extension: ext, - }; - } - - /** - * Validate file signature (magic numbers) - */ - private async validateFileSignature( - file: UploadedFile, - ): Promise<{ valid: boolean; detectedType?: string }> { - const buffer = file.buffer; - - // Check if we have magic numbers for this MIME type - const expectedSignatures = MAGIC_NUMBERS[file.mimetype]; - if (!expectedSignatures) { - // No magic numbers defined for this type, skip validation - this.logger.warn(`No magic numbers defined for MIME type: ${file.mimetype}`); - return { valid: true }; + /** + * Validate file extension + */ + private validateExtension(filename: string): { + valid: boolean; + extension: string; + } { + const ext = path.extname(filename).toLowerCase(); + return { + valid: this.allAllowedExtensions.includes(ext), + extension: ext, + }; } - - // Check if any signature matches - for (const signature of expectedSignatures) { - if (buffer.length >= signature.length) { - const fileHeader = buffer.slice(0, signature.length); - if (fileHeader.equals(signature)) { - return { valid: true }; + /** + * Validate file signature (magic numbers) + */ + private async validateFileSignature(file: UploadedFile): Promise<{ + valid: boolean; + detectedType?: string; + }> { + const buffer = file.buffer; + // Check if we have magic numbers for this MIME type + const expectedSignatures = MAGIC_NUMBERS[file.mimetype]; + if (!expectedSignatures) { + // No magic numbers defined for this type, skip validation + this.logger.warn(`No magic numbers defined for MIME type: ${file.mimetype}`); + return { valid: true }; } - } - } - - // Special case: Office documents (docx, xlsx, pptx) are ZIP-based - if (file.mimetype.includes('officedocument') || file.mimetype.includes('vnd.openxmlformats')) { - const zipSignature = MAGIC_NUMBERS['application/zip']; - if (zipSignature && buffer.length >= zipSignature[0].length) { - const fileHeader = buffer.slice(0, zipSignature[0].length); - if (fileHeader.equals(zipSignature[0])) { - return { valid: true }; + // Check if any signature matches + for (const signature of expectedSignatures) { + if (buffer.length >= signature.length) { + const fileHeader = buffer.slice(0, signature.length); + if (fileHeader.equals(signature)) { + return { valid: true }; + } + } + } + // Special case: Office documents (docx, xlsx, pptx) are ZIP-based + if (file.mimetype.includes('officedocument') || file.mimetype.includes('vnd.openxmlformats')) { + const zipSignature = MAGIC_NUMBERS['application/zip']; + if (zipSignature && buffer.length >= zipSignature[0].length) { + const fileHeader = buffer.slice(0, zipSignature[0].length); + if (fileHeader.equals(zipSignature[0])) { + return { valid: true }; + } + } } - } + return { valid: false }; } - - return { valid: false }; - } - - /** - * Get file type category - */ - private getFileType( - mimeType: string, - ): 'image' | 'video' | 'document' | 'audio' | 'archive' | 'unknown' { - const normalizedMime = mimeType.toLowerCase(); - if ( - ALLOWED_FILE_TYPES.IMAGES.includes( - normalizedMime as (typeof ALLOWED_FILE_TYPES.IMAGES)[number], - ) - ) - return 'image'; - if ( - ALLOWED_FILE_TYPES.VIDEOS.includes( - normalizedMime as (typeof ALLOWED_FILE_TYPES.VIDEOS)[number], - ) - ) - return 'video'; - if ( - ALLOWED_FILE_TYPES.DOCUMENTS.includes( - normalizedMime as (typeof ALLOWED_FILE_TYPES.DOCUMENTS)[number], - ) - ) - return 'document'; - if ( - ALLOWED_FILE_TYPES.AUDIO.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.AUDIO)[number]) - ) - return 'audio'; - if ( - ALLOWED_FILE_TYPES.ARCHIVES.includes( - normalizedMime as (typeof ALLOWED_FILE_TYPES.ARCHIVES)[number], - ) - ) - return 'archive'; - return 'unknown'; - } - - /** - * Validate file size based on type - */ - private validateFileSize(size: number, fileType: string): { valid: boolean; maxSize: number } { - let maxSize = FILE_SIZE_LIMITS.DEFAULT_MAX_SIZE; - - switch (fileType) { - case 'image': - maxSize = FILE_SIZE_LIMITS.IMAGE_MAX_SIZE; - break; - case 'video': - maxSize = FILE_SIZE_LIMITS.VIDEO_MAX_SIZE; - break; - case 'document': - maxSize = FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE; - break; - case 'audio': - maxSize = FILE_SIZE_LIMITS.AUDIO_MAX_SIZE; - break; - case 'archive': - maxSize = FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE; - break; + /** + * Get file type category + */ + private getFileType(mimeType: string): 'image' | 'video' | 'document' | 'audio' | 'archive' | 'unknown' { + const normalizedMime = mimeType.toLowerCase(); + if (ALLOWED_FILE_TYPES.IMAGES.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.IMAGES)[number])) + return 'image'; + if (ALLOWED_FILE_TYPES.VIDEOS.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.VIDEOS)[number])) + return 'video'; + if (ALLOWED_FILE_TYPES.DOCUMENTS.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.DOCUMENTS)[number])) + return 'document'; + if (ALLOWED_FILE_TYPES.AUDIO.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.AUDIO)[number])) + return 'audio'; + if (ALLOWED_FILE_TYPES.ARCHIVES.includes(normalizedMime as (typeof ALLOWED_FILE_TYPES.ARCHIVES)[number])) + return 'archive'; + return 'unknown'; } - - return { - valid: size <= maxSize, - maxSize, - }; - } - - /** - * Validate image dimensions - */ - async validateImageDimensions(buffer: Buffer): Promise<{ - valid: boolean; - dimensions?: ImageDimensions & { format?: string; hasAlpha?: boolean }; - warnings?: string[]; - }> { - try { - const metadata = await sharp(buffer).metadata(); - - if (!metadata.width || !metadata.height) { + /** + * Validate file size based on type + */ + private validateFileSize(size: number, fileType: string): { + valid: boolean; + maxSize: number; + } { + let maxSize = FILE_SIZE_LIMITS.DEFAULT_MAX_SIZE; + switch (fileType) { + case 'image': + maxSize = FILE_SIZE_LIMITS.IMAGE_MAX_SIZE; + break; + case 'video': + maxSize = FILE_SIZE_LIMITS.VIDEO_MAX_SIZE; + break; + case 'document': + maxSize = FILE_SIZE_LIMITS.DOCUMENT_MAX_SIZE; + break; + case 'audio': + maxSize = FILE_SIZE_LIMITS.AUDIO_MAX_SIZE; + break; + case 'archive': + maxSize = FILE_SIZE_LIMITS.ARCHIVE_MAX_SIZE; + break; + } return { - valid: false, - warnings: ['Could not determine image dimensions'], + valid: size <= maxSize, + maxSize, }; - } - - const warnings: string[] = []; - const { MIN_WIDTH, MIN_HEIGHT, MAX_WIDTH, MAX_HEIGHT, MAX_PIXELS } = IMAGE_DIMENSION_LIMITS; - - // Check minimum dimensions - if (metadata.width < MIN_WIDTH || metadata.height < MIN_HEIGHT) { - return { - valid: false, - dimensions: { - width: metadata.width, - height: metadata.height, - format: metadata.format, - hasAlpha: metadata.hasAlpha, - }, + } + /** + * Validate image dimensions + */ + async validateImageDimensions(buffer: Buffer): Promise<{ + valid: boolean; + dimensions?: ImageDimensions & { + format?: string; + hasAlpha?: boolean; }; - } - - // Check maximum dimensions - if (metadata.width > MAX_WIDTH || metadata.height > MAX_HEIGHT) { + warnings?: string[]; + }> { + try { + const metadata = await sharp(buffer).metadata(); + if (!metadata.width || !metadata.height) { + return { + valid: false, + warnings: ['Could not determine image dimensions'], + }; + } + const warnings: string[] = []; + const { MIN_WIDTH, MIN_HEIGHT, MAX_WIDTH, MAX_HEIGHT, MAX_PIXELS } = IMAGE_DIMENSION_LIMITS; + // Check minimum dimensions + if (metadata.width < MIN_WIDTH || metadata.height < MIN_HEIGHT) { + return { + valid: false, + dimensions: { + width: metadata.width, + height: metadata.height, + format: metadata.format, + hasAlpha: metadata.hasAlpha, + }, + }; + } + // Check maximum dimensions + if (metadata.width > MAX_WIDTH || metadata.height > MAX_HEIGHT) { + return { + valid: false, + dimensions: { + width: metadata.width, + height: metadata.height, + format: metadata.format, + hasAlpha: metadata.hasAlpha, + }, + }; + } + // Check total pixel count + const totalPixels = metadata.width * metadata.height; + if (totalPixels > MAX_PIXELS) { + warnings.push(`Image has ${totalPixels.toLocaleString()} pixels, which is very large`); + } + // Warn about very large dimensions + if (metadata.width > 8192 || metadata.height > 8192) { + warnings.push('Image dimensions are very large and may cause performance issues'); + } + return { + valid: true, + dimensions: { + width: metadata.width, + height: metadata.height, + format: metadata.format, + hasAlpha: metadata.hasAlpha, + }, + warnings: warnings.length > 0 ? warnings : undefined, + }; + } + catch (error) { + this.logger.error('Failed to validate image dimensions:', error); + return { + valid: false, + warnings: ['Failed to read image metadata'], + }; + } + } + /** + * Check for security patterns in file + */ + private async checkSecurityPatterns(file: UploadedFile): Promise<{ + valid: boolean; + issues: string[]; + }> { + const issues: string[] = []; + // Check for double extensions (e.g., file.jpg.exe) + const basename = path.basename(file.originalname, path.extname(file.originalname)); + if (path.extname(basename)) { + issues.push('File has multiple extensions which may indicate malicious intent'); + } + // Check for null bytes in filename + if (file.originalname.includes('\x00')) { + issues.push('Filename contains null bytes'); + } + // Check for path traversal + if (file.originalname.includes('..') || + file.originalname.includes('/') || + file.originalname.includes('\\')) { + issues.push('Filename contains path traversal characters'); + } + // Check for control characters + // eslint-disable-next-line no-control-regex + if (/[\x00-\x1f\x7f-\x9f]/.test(file.originalname)) { + issues.push('Filename contains control characters'); + } + // Check file content for executable signatures (basic check) + const executableSignatures = [ + Buffer.from([0x4d, 0x5a]), // Windows executable (MZ header) + Buffer.from([0x7f, 0x45, 0x4c, 0x46]), // ELF (Linux executable) + Buffer.from([0xca, 0xfe, 0xba, 0xbe]), // Java class file + Buffer.from([0xcf, 0xfa, 0xed, 0xfe]), // macOS Mach-O + ]; + for (const signature of executableSignatures) { + if (file.buffer.length >= signature.length) { + const fileHeader = file.buffer.slice(0, signature.length); + if (fileHeader.equals(signature)) { + issues.push('File appears to be an executable, which is not allowed'); + break; + } + } + } return { - valid: false, - dimensions: { - width: metadata.width, - height: metadata.height, - format: metadata.format, - hasAlpha: metadata.hasAlpha, - }, + valid: issues.length === 0, + issues, }; - } - - // Check total pixel count - const totalPixels = metadata.width * metadata.height; - if (totalPixels > MAX_PIXELS) { - warnings.push(`Image has ${totalPixels.toLocaleString()} pixels, which is very large`); - } - - // Warn about very large dimensions - if (metadata.width > 8192 || metadata.height > 8192) { - warnings.push('Image dimensions are very large and may cause performance issues'); - } - - return { - valid: true, - dimensions: { - width: metadata.width, - height: metadata.height, - format: metadata.format, - hasAlpha: metadata.hasAlpha, - }, - warnings: warnings.length > 0 ? warnings : undefined, - }; - } catch (error) { - this.logger.error('Failed to validate image dimensions:', error); - return { - valid: false, - warnings: ['Failed to read image metadata'], - }; - } - } - - /** - * Check for security patterns in file - */ - private async checkSecurityPatterns( - file: UploadedFile, - ): Promise<{ valid: boolean; issues: string[] }> { - const issues: string[] = []; - - // Check for double extensions (e.g., file.jpg.exe) - const basename = path.basename(file.originalname, path.extname(file.originalname)); - if (path.extname(basename)) { - issues.push('File has multiple extensions which may indicate malicious intent'); } - - // Check for null bytes in filename - if (file.originalname.includes('\x00')) { - issues.push('Filename contains null bytes'); + /** + * Format bytes to human readable string + */ + private formatBytes(bytes: number): string { + if (bytes === 0) + return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`; } - - // Check for path traversal - if ( - file.originalname.includes('..') || - file.originalname.includes('/') || - file.originalname.includes('\\') - ) { - issues.push('Filename contains path traversal characters'); - } - - // Check for control characters - // eslint-disable-next-line no-control-regex - if (/[\x00-\x1f\x7f-\x9f]/.test(file.originalname)) { - issues.push('Filename contains control characters'); - } - - // Check file content for executable signatures (basic check) - const executableSignatures = [ - Buffer.from([0x4d, 0x5a]), // Windows executable (MZ header) - Buffer.from([0x7f, 0x45, 0x4c, 0x46]), // ELF (Linux executable) - Buffer.from([0xca, 0xfe, 0xba, 0xbe]), // Java class file - Buffer.from([0xcf, 0xfa, 0xed, 0xfe]), // macOS Mach-O - ]; - - for (const signature of executableSignatures) { - if (file.buffer.length >= signature.length) { - const fileHeader = file.buffer.slice(0, signature.length); - if (fileHeader.equals(signature)) { - issues.push('File appears to be an executable, which is not allowed'); - break; + /** + * Quick validation for controller use + */ + async quickValidate(file: UploadedFile): Promise { + const result = await this.validateFile(file); + if (!result.valid) { + throw new BadRequestException({ + message: 'File validation failed', + errors: result.errors, + warnings: result.warnings, + }); } - } - } - - return { - valid: issues.length === 0, - issues, - }; - } - - /** - * Format bytes to human readable string - */ - private formatBytes(bytes: number): string { - if (bytes === 0) return '0 Bytes'; - const k = 1024; - const sizes = ['Bytes', 'KB', 'MB', 'GB']; - const i = Math.floor(Math.log(bytes) / Math.log(k)); - return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`; - } - - /** - * Quick validation for controller use - */ - async quickValidate(file: UploadedFile): Promise { - const result = await this.validateFile(file); - if (!result.valid) { - throw new BadRequestException({ - message: 'File validation failed', - errors: result.errors, - warnings: result.warnings, - }); } - } } diff --git a/src/media/validation/malware-scanning.service.ts b/src/media/validation/malware-scanning.service.ts index c332020e..ea83b80c 100644 --- a/src/media/validation/malware-scanning.service.ts +++ b/src/media/validation/malware-scanning.service.ts @@ -3,287 +3,246 @@ import { ConfigService } from '@nestjs/config'; import axios, { AxiosInstance } from 'axios'; import { MALWARE_SCAN_CONFIG } from './file-validation.constants'; import { UploadedFile } from '../../common/types/file.types'; - export interface MalwareScanResult { - clean: boolean; - threats: string[]; - scanTime: number; - scannerVersion?: string; - error?: string; + clean: boolean; + threats: string[]; + scanTime: number; + scannerVersion?: string; + error?: string; } - export interface VirusTotalReport { - data: { - attributes: { - last_analysis_stats: { - malicious: number; - suspicious: number; - undetected: number; - harmless: number; - timeout: number; - }; - last_analysis_results: Record< - string, - { - category: string; - engine_name: string; - result: string | null; - } - >; - size: number; - type_description: string; + data: { + attributes: { + last_analysis_stats: { + malicious: number; + suspicious: number; + undetected: number; + harmless: number; + timeout: number; + }; + last_analysis_results: Record; + size: number; + type_description: string; + }; }; - }; } - @Injectable() export class MalwareScanningService { - private readonly logger = new Logger(MalwareScanningService.name); - private readonly virusTotalClient: AxiosInstance | null = null; - private readonly clamAvClient: AxiosInstance | null = null; - private readonly enableScanning: boolean; - - constructor(private readonly configService: ConfigService) { - this.enableScanning = this.configService.get('ENABLE_MALWARE_SCANNING', false); - - // Initialize VirusTotal client if API key is available - const vtApiKey = this.configService.get('VIRUSTOTAL_API_KEY'); - if (vtApiKey) { - this.virusTotalClient = axios.create({ - baseURL: 'https://www.virustotal.com/api/v3', - headers: { - 'x-apikey': vtApiKey, - 'Content-Type': 'application/json', - }, - timeout: MALWARE_SCAN_CONFIG.TIMEOUT_MS, - }); - } - - // Initialize ClamAV REST client if configured - const clamAvUrl = this.configService.get('CLAMAV_URL'); - if (clamAvUrl) { - this.clamAvClient = axios.create({ - baseURL: clamAvUrl, - timeout: MALWARE_SCAN_CONFIG.TIMEOUT_MS, - }); - } - } - - /** - * Scan file for malware - */ - async scanFile(file: UploadedFile): Promise { - const startTime = Date.now(); - - // Skip scanning if disabled or file too large - if (!this.enableScanning) { - this.logger.debug('Malware scanning is disabled'); - return { - clean: true, - threats: [], - scanTime: 0, - }; - } - - if (file.size > MALWARE_SCAN_CONFIG.MAX_FILE_SIZE_FOR_SCAN) { - this.logger.warn(`File too large for malware scan: ${file.size} bytes`); - return { - clean: true, - threats: [], - scanTime: 0, - error: 'File too large for malware scanning', - }; - } - - try { - // Try ClamAV first (local scanning, faster) - if (this.clamAvClient) { - return await this.scanWithClamAv(file); - } - - // Fall back to VirusTotal - if (this.virusTotalClient) { - return await this.scanWithVirusTotal(file); - } - - // No scanning available - this.logger.warn('No malware scanning service configured'); - return { - clean: true, - threats: [], - scanTime: Date.now() - startTime, - error: 'No malware scanning service available', - }; - } catch (error) { - this.logger.error('Malware scan failed:', error); - return { - clean: false, - threats: [], - scanTime: Date.now() - startTime, - error: `Scan failed: ${error instanceof Error ? error.message : 'Unknown error'}`, - }; - } - } - - /** - * Scan file using ClamAV REST API - */ - private async scanWithClamAv(file: UploadedFile): Promise { - const startTime = Date.now(); - - if (!this.clamAvClient) { - throw new Error('ClamAV client not initialized'); + private readonly logger = new Logger(MalwareScanningService.name); + private readonly virusTotalClient: AxiosInstance | null = null; + private readonly clamAvClient: AxiosInstance | null = null; + private readonly enableScanning: boolean; + constructor(private readonly configService: ConfigService) { + this.enableScanning = this.configService.get('ENABLE_MALWARE_SCANNING', false); + // Initialize VirusTotal client if API key is available + const vtApiKey = this.configService.get('VIRUSTOTAL_API_KEY'); + if (vtApiKey) { + this.virusTotalClient = axios.create({ + baseURL: 'https://www.virustotal.com/api/v3', + headers: { + 'x-apikey': vtApiKey, + 'Content-Type': 'application/json', + }, + timeout: MALWARE_SCAN_CONFIG.TIMEOUT_MS, + }); + } + // Initialize ClamAV REST client if configured + const clamAvUrl = this.configService.get('CLAMAV_URL'); + if (clamAvUrl) { + this.clamAvClient = axios.create({ + baseURL: clamAvUrl, + timeout: MALWARE_SCAN_CONFIG.TIMEOUT_MS, + }); + } } - - const formData = new FormData(); - const blob = new Blob([file.buffer as unknown as BlobPart], { type: file.mimetype }); - formData.append('file', blob, file.originalname); - - const response = await this.clamAvClient.post('/scan', formData, { - headers: { - 'Content-Type': 'multipart/form-data', - }, - }); - - const result = response.data; - const threats: string[] = []; - - if (result.infected) { - threats.push(result.virus || 'Unknown malware detected'); + /** + * Scan file for malware + */ + async scanFile(file: UploadedFile): Promise { + const startTime = Date.now(); + // Skip scanning if disabled or file too large + if (!this.enableScanning) { + this.logger.debug('Malware scanning is disabled'); + return { + clean: true, + threats: [], + scanTime: 0, + }; + } + if (file.size > MALWARE_SCAN_CONFIG.MAX_FILE_SIZE_FOR_SCAN) { + this.logger.warn(`File too large for malware scan: ${file.size} bytes`); + return { + clean: true, + threats: [], + scanTime: 0, + error: 'File too large for malware scanning', + }; + } + try { + // Try ClamAV first (local scanning, faster) + if (this.clamAvClient) { + return await this.scanWithClamAv(file); + } + // Fall back to VirusTotal + if (this.virusTotalClient) { + return await this.scanWithVirusTotal(file); + } + // No scanning available + this.logger.warn('No malware scanning service configured'); + return { + clean: true, + threats: [], + scanTime: Date.now() - startTime, + error: 'No malware scanning service available', + }; + } + catch (error) { + this.logger.error('Malware scan failed:', error); + return { + clean: false, + threats: [], + scanTime: Date.now() - startTime, + error: `Scan failed: ${error instanceof Error ? error.message : 'Unknown error'}`, + }; + } } - - return { - clean: !result.infected, - threats, - scanTime: Date.now() - startTime, - scannerVersion: result.engineVersion, - }; - } - - /** - * Scan file using VirusTotal API - */ - private async scanWithVirusTotal(file: UploadedFile): Promise { - const startTime = Date.now(); - - if (!this.virusTotalClient) { - throw new Error('VirusTotal client not initialized'); + /** + * Scan file using ClamAV REST API + */ + private async scanWithClamAv(file: UploadedFile): Promise { + const startTime = Date.now(); + if (!this.clamAvClient) { + throw new Error('ClamAV client not initialized'); + } + const formData = new FormData(); + const blob = new Blob([file.buffer as unknown as BlobPart], { type: file.mimetype }); + formData.append('file', blob, file.originalname); + const response = await this.clamAvClient.post('/scan', formData, { + headers: { + 'Content-Type': 'multipart/form-data', + }, + }); + const result = response.data; + const threats: string[] = []; + if (result.infected) { + threats.push(result.virus || 'Unknown malware detected'); + } + return { + clean: !result.infected, + threats, + scanTime: Date.now() - startTime, + scannerVersion: result.engineVersion, + }; } - - // Step 1: Upload file for scanning - const formData = new FormData(); - const blob = new Blob([file.buffer as unknown as BlobPart], { type: file.mimetype }); - formData.append('file', blob, file.originalname); - - const uploadResponse = await this.virusTotalClient.post('/files', formData, { - headers: { - 'Content-Type': 'multipart/form-data', - }, - }); - - const analysisId = uploadResponse.data.data.id; - - // Step 2: Poll for results - let attempts = 0; - let report: VirusTotalReport | null = null; - - while (attempts < MALWARE_SCAN_CONFIG.RETRY_ATTEMPTS) { - await this.delay(5000); // Wait 5 seconds between polls - - try { - const reportResponse = await this.virusTotalClient.get(`/analyses/${analysisId}`); - report = reportResponse.data; - - // Check if analysis is complete + /** + * Scan file using VirusTotal API + */ + private async scanWithVirusTotal(file: UploadedFile): Promise { + const startTime = Date.now(); + if (!this.virusTotalClient) { + throw new Error('VirusTotal client not initialized'); + } + // Step 1: Upload file for scanning + const formData = new FormData(); + const blob = new Blob([file.buffer as unknown as BlobPart], { type: file.mimetype }); + formData.append('file', blob, file.originalname); + const uploadResponse = await this.virusTotalClient.post('/files', formData, { + headers: { + 'Content-Type': 'multipart/form-data', + }, + }); + const analysisId = uploadResponse.data.data.id; + // Step 2: Poll for results + let attempts = 0; + let report: VirusTotalReport | null = null; + while (attempts < MALWARE_SCAN_CONFIG.RETRY_ATTEMPTS) { + await this.delay(5000); // Wait 5 seconds between polls + try { + const reportResponse = await this.virusTotalClient.get(`/analyses/${analysisId}`); + report = reportResponse.data; + // Check if analysis is complete + const stats = report.data.attributes.last_analysis_stats; + if (stats.malicious > 0 || stats.suspicious > 0 || stats.undetected > 0) { + break; // Analysis complete + } + } + catch (error) { + this.logger.warn(`Poll attempt ${attempts + 1} failed:`, error); + } + attempts++; + } + if (!report) { + return { + clean: false, + threats: [], + scanTime: Date.now() - startTime, + error: 'Failed to retrieve scan results', + }; + } + // Process results const stats = report.data.attributes.last_analysis_stats; - if (stats.malicious > 0 || stats.suspicious > 0 || stats.undetected > 0) { - break; // Analysis complete + const threats: string[] = []; + if (stats.malicious > 0 || stats.suspicious > 0) { + const results = report.data.attributes.last_analysis_results; + for (const [engine, result] of Object.entries(results)) { + if (result.category === 'malicious' || result.category === 'suspicious') { + threats.push(`${engine}: ${result.result || 'Threat detected'}`); + } + } } - } catch (error) { - this.logger.warn(`Poll attempt ${attempts + 1} failed:`, error); - } - - attempts++; + return { + clean: stats.malicious === 0 && stats.suspicious === 0, + threats, + scanTime: Date.now() - startTime, + scannerVersion: 'VirusTotal v3', + }; } - - if (!report) { - return { - clean: false, - threats: [], - scanTime: Date.now() - startTime, - error: 'Failed to retrieve scan results', - }; - } - - // Process results - const stats = report.data.attributes.last_analysis_stats; - const threats: string[] = []; - - if (stats.malicious > 0 || stats.suspicious > 0) { - const results = report.data.attributes.last_analysis_results; - - for (const [engine, result] of Object.entries(results)) { - if (result.category === 'malicious' || result.category === 'suspicious') { - threats.push(`${engine}: ${result.result || 'Threat detected'}`); + /** + * Quick hash-based lookup (check if file was already scanned) + */ + async checkHash(fileHash: string): Promise { + if (!this.virusTotalClient) { + return null; + } + try { + const response = await this.virusTotalClient.get(`/files/${fileHash}`); + const report: VirusTotalReport = response.data; + const stats = report.data.attributes.last_analysis_stats; + const threats: string[] = []; + if (stats.malicious > 0 || stats.suspicious > 0) { + const results = report.data.attributes.last_analysis_results; + for (const [engine, result] of Object.entries(results)) { + if (result.category === 'malicious' || result.category === 'suspicious') { + threats.push(`${engine}: ${result.result || 'Threat detected'}`); + } + } + } + return { + clean: stats.malicious === 0 && stats.suspicious === 0, + threats, + scanTime: 0, + scannerVersion: 'VirusTotal v3 (cached)', + }; + } + catch (_error) { + // File not found in database, needs to be scanned + return null; } - } } - - return { - clean: stats.malicious === 0 && stats.suspicious === 0, - threats, - scanTime: Date.now() - startTime, - scannerVersion: 'VirusTotal v3', - }; - } - - /** - * Quick hash-based lookup (check if file was already scanned) - */ - async checkHash(fileHash: string): Promise { - if (!this.virusTotalClient) { - return null; + /** + * Utility delay function + */ + private delay(ms: number): Promise { + return new Promise((resolve) => setTimeout(resolve, ms)); } - - try { - const response = await this.virusTotalClient.get(`/files/${fileHash}`); - const report: VirusTotalReport = response.data; - - const stats = report.data.attributes.last_analysis_stats; - const threats: string[] = []; - - if (stats.malicious > 0 || stats.suspicious > 0) { - const results = report.data.attributes.last_analysis_results; - for (const [engine, result] of Object.entries(results)) { - if (result.category === 'malicious' || result.category === 'suspicious') { - threats.push(`${engine}: ${result.result || 'Threat detected'}`); - } - } - } - - return { - clean: stats.malicious === 0 && stats.suspicious === 0, - threats, - scanTime: 0, - scannerVersion: 'VirusTotal v3 (cached)', - }; - } catch (_error) { - // File not found in database, needs to be scanned - return null; + /** + * Check if scanning is available + */ + isScanningAvailable(): boolean { + return this.enableScanning && (this.clamAvClient !== null || this.virusTotalClient !== null); } - } - - /** - * Utility delay function - */ - private delay(ms: number): Promise { - return new Promise((resolve) => setTimeout(resolve, ms)); - } - - /** - * Check if scanning is available - */ - isScanningAvailable(): boolean { - return this.enableScanning && (this.clamAvClient !== null || this.virusTotalClient !== null); - } } diff --git a/src/media/validation/upload-progress.service.ts b/src/media/validation/upload-progress.service.ts index 0795e07f..cef4f72f 100644 --- a/src/media/validation/upload-progress.service.ts +++ b/src/media/validation/upload-progress.service.ts @@ -1,322 +1,275 @@ import { Injectable, Logger } from '@nestjs/common'; import Redis from 'ioredis'; import { UPLOAD_PROGRESS_CONFIG } from './file-validation.constants'; - export interface UploadProgress { - uploadId: string; - status: - | 'pending' - | 'validating' - | 'scanning' - | 'processing' - | 'uploading' - | 'completed' - | 'failed'; - progress: number; // 0-100 - fileName: string; - fileSize: number; - bytesProcessed: number; - stage: string; - message: string; - startedAt: string; - updatedAt: string; - completedAt?: string; - error?: string; - result?: { - contentId?: string; - url?: string; - thumbnails?: string[]; - }; + uploadId: string; + status: 'pending' | 'validating' | 'scanning' | 'processing' | 'uploading' | 'completed' | 'failed'; + progress: number; // 0-100 + fileName: string; + fileSize: number; + bytesProcessed: number; + stage: string; + message: string; + startedAt: string; + updatedAt: string; + completedAt?: string; + error?: string; + result?: { + contentId?: string; + url?: string; + thumbnails?: string[]; + }; } - export interface ProgressUpdate { - status?: UploadProgress['status']; - progress?: number; - stage?: string; - message?: string; - bytesProcessed?: number; - error?: string; - result?: UploadProgress['result']; + status?: UploadProgress['status']; + progress?: number; + stage?: string; + message?: string; + bytesProcessed?: number; + error?: string; + result?: UploadProgress['result']; } - @Injectable() export class UploadProgressService { - private readonly logger = new Logger(UploadProgressService.name); - private readonly redis: Redis; - - constructor() { - // Initialize Redis client - this.redis = new Redis(process.env.REDIS_URL || 'redis://localhost:6379'); - this.redis.on('error', () => { - // Prevent unhandled error events during Redis outages - }); - } - - /** - * Initialize upload progress tracking - */ - async initializeUpload( - uploadId: string, - fileName: string, - fileSize: number, - ): Promise { - const progress: UploadProgress = { - uploadId, - status: 'pending', - progress: 0, - fileName, - fileSize, - bytesProcessed: 0, - stage: 'initialized', - message: 'Upload initialized', - startedAt: new Date().toISOString(), - updatedAt: new Date().toISOString(), - }; - - await this.saveProgress(uploadId, progress); - this.logger.log(`Initialized upload tracking for ${uploadId}: ${fileName}`); - - return progress; - } - - /** - * Update upload progress - */ - async updateProgress(uploadId: string, update: ProgressUpdate): Promise { - const existing = await this.getProgress(uploadId); - if (!existing) { - throw new Error(`Upload ${uploadId} not found`); + private readonly logger = new Logger(UploadProgressService.name); + private readonly redis: Redis; + constructor() { + // Initialize Redis client + this.redis = new Redis(process.env.REDIS_URL || 'redis://localhost:6379'); + this.redis.on('error', () => { + // Prevent unhandled error events during Redis outages + }); } - - const progress: UploadProgress = { - ...existing, - ...update, - updatedAt: new Date().toISOString(), - }; - - // Ensure progress stays within bounds - if (progress.progress !== undefined) { - progress.progress = Math.max(0, Math.min(100, progress.progress)); + /** + * Initialize upload progress tracking + */ + async initializeUpload(uploadId: string, fileName: string, fileSize: number): Promise { + const progress: UploadProgress = { + uploadId, + status: 'pending', + progress: 0, + fileName, + fileSize, + bytesProcessed: 0, + stage: 'initialized', + message: 'Upload initialized', + startedAt: new Date().toISOString(), + updatedAt: new Date().toISOString(), + }; + await this.saveProgress(uploadId, progress); + this.logger.log(`Initialized upload tracking for ${uploadId}: ${fileName}`); + return progress; } - - // Mark as completed if progress is 100% - if (progress.progress === 100 && progress.status !== 'failed') { - progress.status = 'completed'; - progress.completedAt = new Date().toISOString(); + /** + * Update upload progress + */ + async updateProgress(uploadId: string, update: ProgressUpdate): Promise { + const existing = await this.getProgress(uploadId); + if (!existing) { + throw new Error(`Upload ${uploadId} not found`); + } + const progress: UploadProgress = { + ...existing, + ...update, + updatedAt: new Date().toISOString(), + }; + // Ensure progress stays within bounds + if (progress.progress !== undefined) { + progress.progress = Math.max(0, Math.min(100, progress.progress)); + } + // Mark as completed if progress is 100% + if (progress.progress === 100 && progress.status !== 'failed') { + progress.status = 'completed'; + progress.completedAt = new Date().toISOString(); + } + await this.saveProgress(uploadId, progress); + return progress; } - - await this.saveProgress(uploadId, progress); - return progress; - } - - /** - * Get upload progress - */ - async getProgress(uploadId: string): Promise { - try { - const key = this.getRedisKey(uploadId); - const data = await this.redis.get(key); - - if (!data) { - return null; - } - - return JSON.parse(data) as UploadProgress; - } catch (error) { - this.logger.error(`Failed to get progress for ${uploadId}:`, error); - return null; + /** + * Get upload progress + */ + async getProgress(uploadId: string): Promise { + try { + const key = this.getRedisKey(uploadId); + const data = await this.redis.get(key); + if (!data) { + return null; + } + return JSON.parse(data) as UploadProgress; + } + catch (error) { + this.logger.error(`Failed to get progress for ${uploadId}:`, error); + return null; + } } - } - - /** - * Mark upload as failed - */ - async markFailed(uploadId: string, error: string): Promise { - return this.updateProgress(uploadId, { - status: 'failed', - progress: 0, - stage: 'failed', - message: `Upload failed: ${error}`, - error, - }); - } - - /** - * Mark upload as completed - */ - async markCompleted(uploadId: string, result: UploadProgress['result']): Promise { - return this.updateProgress(uploadId, { - status: 'completed', - progress: 100, - stage: 'completed', - message: 'Upload completed successfully', - result, - }); - } - - /** - * Delete upload progress (cleanup) - */ - async deleteProgress(uploadId: string): Promise { - const key = this.getRedisKey(uploadId); - await this.redis.del(key); - this.logger.log(`Deleted upload progress for ${uploadId}`); - } - - /** - * List active uploads - */ - async listActiveUploads(): Promise { - try { - const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; - const keys = await this.redis.keys(pattern); - - if (keys.length === 0) { - return []; - } - - const values = await this.redis.mget(...keys); - const uploads: UploadProgress[] = []; - - for (const value of values) { - if (value) { - const progress = JSON.parse(value) as UploadProgress; - // Only include non-completed and non-failed uploads - if (progress.status !== 'completed' && progress.status !== 'failed') { - uploads.push(progress); - } + /** + * Mark upload as failed + */ + async markFailed(uploadId: string, error: string): Promise { + return this.updateProgress(uploadId, { + status: 'failed', + progress: 0, + stage: 'failed', + message: `Upload failed: ${error}`, + error, + }); + } + /** + * Mark upload as completed + */ + async markCompleted(uploadId: string, result: UploadProgress['result']): Promise { + return this.updateProgress(uploadId, { + status: 'completed', + progress: 100, + stage: 'completed', + message: 'Upload completed successfully', + result, + }); + } + /** + * Delete upload progress (cleanup) + */ + async deleteProgress(uploadId: string): Promise { + const key = this.getRedisKey(uploadId); + await this.redis.del(key); + this.logger.log(`Deleted upload progress for ${uploadId}`); + } + /** + * List active uploads + */ + async listActiveUploads(): Promise { + try { + const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; + const keys = await this.redis.keys(pattern); + if (keys.length === 0) { + return []; + } + const values = await this.redis.mget(...keys); + const uploads: UploadProgress[] = []; + for (const value of values) { + if (value) { + const progress = JSON.parse(value) as UploadProgress; + // Only include non-completed and non-failed uploads + if (progress.status !== 'completed' && progress.status !== 'failed') { + uploads.push(progress); + } + } + } + return uploads.sort((a, b) => new Date(b.updatedAt).getTime() - new Date(a.updatedAt).getTime()); + } + catch (error) { + this.logger.error('Failed to list active uploads:', error); + return []; } - } - - return uploads.sort( - (a, b) => new Date(b.updatedAt).getTime() - new Date(a.updatedAt).getTime(), - ); - } catch (error) { - this.logger.error('Failed to list active uploads:', error); - return []; } - } - - /** - * Cleanup old completed uploads - */ - async cleanupOldUploads(maxAgeHours: number = 24): Promise { - try { - const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; - const keys = await this.redis.keys(pattern); - - if (keys.length === 0) { - return 0; - } - - const values = await this.redis.mget(...keys); - const now = new Date().getTime(); - const maxAgeMs = maxAgeHours * 60 * 60 * 1000; - let deletedCount = 0; - - for (let i = 0; i < values.length; i++) { - const value = values[i]; - if (value) { - const progress = JSON.parse(value) as UploadProgress; - const updatedAt = new Date(progress.updatedAt).getTime(); - - // Delete if old and completed/failed - if ( - (progress.status === 'completed' || progress.status === 'failed') && - now - updatedAt > maxAgeMs - ) { - await this.redis.del(keys[i]); - deletedCount++; - } + /** + * Cleanup old completed uploads + */ + async cleanupOldUploads(maxAgeHours: number = 24): Promise { + try { + const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; + const keys = await this.redis.keys(pattern); + if (keys.length === 0) { + return 0; + } + const values = await this.redis.mget(...keys); + const now = new Date().getTime(); + const maxAgeMs = maxAgeHours * 60 * 60 * 1000; + let deletedCount = 0; + for (let i = 0; i < values.length; i++) { + const value = values[i]; + if (value) { + const progress = JSON.parse(value) as UploadProgress; + const updatedAt = new Date(progress.updatedAt).getTime(); + // Delete if old and completed/failed + if ((progress.status === 'completed' || progress.status === 'failed') && + now - updatedAt > maxAgeMs) { + await this.redis.del(keys[i]); + deletedCount++; + } + } + } + this.logger.log(`Cleaned up ${deletedCount} old upload records`); + return deletedCount; + } + catch (error) { + this.logger.error('Failed to cleanup old uploads:', error); + return 0; } - } - - this.logger.log(`Cleaned up ${deletedCount} old upload records`); - return deletedCount; - } catch (error) { - this.logger.error('Failed to cleanup old uploads:', error); - return 0; } - } - - /** - * Get upload statistics - */ - async getStatistics(): Promise<{ - total: number; - pending: number; - validating: number; - scanning: number; - processing: number; - uploading: number; - completed: number; - failed: number; - }> { - try { - const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; - const keys = await this.redis.keys(pattern); - - if (keys.length === 0) { - return { - total: 0, - pending: 0, - validating: 0, - scanning: 0, - processing: 0, - uploading: 0, - completed: 0, - failed: 0, - }; - } - - const values = await this.redis.mget(...keys); - const stats = { - total: 0, - pending: 0, - validating: 0, - scanning: 0, - processing: 0, - uploading: 0, - completed: 0, - failed: 0, - }; - - for (const value of values) { - if (value) { - const progress = JSON.parse(value) as UploadProgress; - stats.total++; - stats[progress.status]++; + /** + * Get upload statistics + */ + async getStatistics(): Promise<{ + total: number; + pending: number; + validating: number; + scanning: number; + processing: number; + uploading: number; + completed: number; + failed: number; + }> { + try { + const pattern = `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}*`; + const keys = await this.redis.keys(pattern); + if (keys.length === 0) { + return { + total: 0, + pending: 0, + validating: 0, + scanning: 0, + processing: 0, + uploading: 0, + completed: 0, + failed: 0, + }; + } + const values = await this.redis.mget(...keys); + const stats = { + total: 0, + pending: 0, + validating: 0, + scanning: 0, + processing: 0, + uploading: 0, + completed: 0, + failed: 0, + }; + for (const value of values) { + if (value) { + const progress = JSON.parse(value) as UploadProgress; + stats.total++; + stats[progress.status]++; + } + } + return stats; + } + catch (error) { + this.logger.error('Failed to get upload statistics:', error); + return { + total: 0, + pending: 0, + validating: 0, + scanning: 0, + processing: 0, + uploading: 0, + completed: 0, + failed: 0, + }; } - } - - return stats; - } catch (error) { - this.logger.error('Failed to get upload statistics:', error); - return { - total: 0, - pending: 0, - validating: 0, - scanning: 0, - processing: 0, - uploading: 0, - completed: 0, - failed: 0, - }; } - } - - /** - * Save progress to Redis - */ - private async saveProgress(uploadId: string, progress: UploadProgress): Promise { - const key = this.getRedisKey(uploadId); - await this.redis.setex(key, UPLOAD_PROGRESS_CONFIG.EXPIRY_SECONDS, JSON.stringify(progress)); - } - - /** - * Get Redis key for upload - */ - private getRedisKey(uploadId: string): string { - return `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}${uploadId}`; - } + /** + * Save progress to Redis + */ + private async saveProgress(uploadId: string, progress: UploadProgress): Promise { + const key = this.getRedisKey(uploadId); + await this.redis.setex(key, UPLOAD_PROGRESS_CONFIG.EXPIRY_SECONDS, JSON.stringify(progress)); + } + /** + * Get Redis key for upload + */ + private getRedisKey(uploadId: string): string { + return `${UPLOAD_PROGRESS_CONFIG.REDIS_KEY_PREFIX}${uploadId}`; + } } diff --git a/src/media/validation/upload-validation.util.spec.ts b/src/media/validation/upload-validation.util.spec.ts index e9d0bec4..9e5dfed1 100644 --- a/src/media/validation/upload-validation.util.spec.ts +++ b/src/media/validation/upload-validation.util.spec.ts @@ -1,45 +1,27 @@ -import { - buildUploadValidationDetails, - MEDIA_UPLOAD_INTERCEPTOR_OPTIONS, -} from './upload-validation.util'; +import { buildUploadValidationDetails, MEDIA_UPLOAD_INTERCEPTOR_OPTIONS, } from './upload-validation.util'; import { ALL_ALLOWED_FILE_TYPES, MAX_UPLOAD_FILE_SIZE } from './file-validation.constants'; - describe('upload validation util', () => { - it('accepts files with whitelisted MIME types', () => { - const callback = jest.fn(); - const req: Record = {}; - - MEDIA_UPLOAD_INTERCEPTOR_OPTIONS.fileFilter( - req, - { mimetype: ALL_ALLOWED_FILE_TYPES[0] }, - callback, - ); - - expect(callback).toHaveBeenCalledWith(null, true); - expect(req).not.toHaveProperty('uploadValidationError'); - }); - - it('rejects files with non-whitelisted MIME types', () => { - const callback = jest.fn(); - const req: Record = {}; - - MEDIA_UPLOAD_INTERCEPTOR_OPTIONS.fileFilter( - req, - { mimetype: 'application/x-msdownload' }, - callback, - ); - - expect(callback).toHaveBeenCalledWith(null, false); - expect(req).toHaveProperty('uploadValidationError'); - expect(req['uploadValidationError']).toMatchObject({ - message: 'File type "application/x-msdownload" is not allowed', + it('accepts files with whitelisted MIME types', () => { + const callback = jest.fn(); + const req: Record = {}; + MEDIA_UPLOAD_INTERCEPTOR_OPTIONS.fileFilter(req, { mimetype: ALL_ALLOWED_FILE_TYPES[0] }, callback); + expect(callback).toHaveBeenCalledWith(null, true); + expect(req).not.toHaveProperty('uploadValidationError'); }); - }); - - it('exposes upload validation details for API responses', () => { - expect(buildUploadValidationDetails()).toMatchObject({ - allowedTypes: ALL_ALLOWED_FILE_TYPES, - maxUploadSize: MAX_UPLOAD_FILE_SIZE, + it('rejects files with non-whitelisted MIME types', () => { + const callback = jest.fn(); + const req: Record = {}; + MEDIA_UPLOAD_INTERCEPTOR_OPTIONS.fileFilter(req, { mimetype: 'application/x-msdownload' }, callback); + expect(callback).toHaveBeenCalledWith(null, false); + expect(req).toHaveProperty('uploadValidationError'); + expect(req['uploadValidationError']).toMatchObject({ + message: 'File type "application/x-msdownload" is not allowed', + }); + }); + it('exposes upload validation details for API responses', () => { + expect(buildUploadValidationDetails()).toMatchObject({ + allowedTypes: ALL_ALLOWED_FILE_TYPES, + maxUploadSize: MAX_UPLOAD_FILE_SIZE, + }); }); - }); }); diff --git a/src/media/validation/upload-validation.util.ts b/src/media/validation/upload-validation.util.ts index 337ba984..13ed60fa 100644 --- a/src/media/validation/upload-validation.util.ts +++ b/src/media/validation/upload-validation.util.ts @@ -1,54 +1,39 @@ -import { - FILE_SIZE_LIMITS, - ALL_ALLOWED_FILE_TYPES, - MAX_UPLOAD_FILE_SIZE, -} from './file-validation.constants'; - +import { FILE_SIZE_LIMITS, ALL_ALLOWED_FILE_TYPES, MAX_UPLOAD_FILE_SIZE, } from './file-validation.constants'; export interface UploadValidationRequestLike { - uploadValidationError?: { - message: string; - allowedMimeTypes: string[]; - }; + uploadValidationError?: { + message: string; + allowedMimeTypes: string[]; + }; } - export interface UploadValidationFileLike { - mimetype?: string; + mimetype?: string; } - export interface UploadFilterCallback { - (error: Error | null, acceptFile: boolean): void; + (error: Error | null, acceptFile: boolean): void; } - export const MEDIA_UPLOAD_INTERCEPTOR_OPTIONS = { - limits: { - fileSize: MAX_UPLOAD_FILE_SIZE, - files: 1, - }, - fileFilter: ( - req: UploadValidationRequestLike, - file: UploadValidationFileLike, - callback: UploadFilterCallback, - ): void => { - const allowedMimeTypes = ALL_ALLOWED_FILE_TYPES as readonly string[]; - const normalizedMimeType = file.mimetype?.toLowerCase().trim() || ''; - - if (!allowedMimeTypes.includes(normalizedMimeType)) { - req.uploadValidationError = { - message: `File type "${file.mimetype || 'unknown'}" is not allowed`, - allowedMimeTypes: ALL_ALLOWED_FILE_TYPES, - }; - callback(null, false); - return; - } - - callback(null, true); - }, + limits: { + fileSize: MAX_UPLOAD_FILE_SIZE, + files: 1, + }, + fileFilter: (req: UploadValidationRequestLike, file: UploadValidationFileLike, callback: UploadFilterCallback): void => { + const allowedMimeTypes = ALL_ALLOWED_FILE_TYPES as readonly string[]; + const normalizedMimeType = file.mimetype?.toLowerCase().trim() || ''; + if (!allowedMimeTypes.includes(normalizedMimeType)) { + req.uploadValidationError = { + message: `File type "${file.mimetype || 'unknown'}" is not allowed`, + allowedMimeTypes: ALL_ALLOWED_FILE_TYPES, + }; + callback(null, false); + return; + } + callback(null, true); + }, } as const; - export function buildUploadValidationDetails() { - return { - allowedTypes: ALL_ALLOWED_FILE_TYPES, - sizeLimits: FILE_SIZE_LIMITS, - maxUploadSize: MAX_UPLOAD_FILE_SIZE, - }; + return { + allowedTypes: ALL_ALLOWED_FILE_TYPES, + sizeLimits: FILE_SIZE_LIMITS, + maxUploadSize: MAX_UPLOAD_FILE_SIZE, + }; } diff --git a/src/messaging/circuit-breaker/circuit-breaker.service.ts b/src/messaging/circuit-breaker/circuit-breaker.service.ts index 7cccb141..0026fa3f 100644 --- a/src/messaging/circuit-breaker/circuit-breaker.service.ts +++ b/src/messaging/circuit-breaker/circuit-breaker.service.ts @@ -1,150 +1,126 @@ import { Injectable, Logger } from '@nestjs/common'; import { TracingService } from '../tracing/tracing.service'; - export interface CircuitBreakerConfig { - failureThreshold: number; // Number of failures to open circuit - recoveryTimeout: number; // Time in ms to wait before attempting recovery - monitoringPeriod: number; // Time in ms to monitor failures + failureThreshold: number; // Number of failures to open circuit + recoveryTimeout: number; // Time in ms to wait before attempting recovery + monitoringPeriod: number; // Time in ms to monitor failures } - export type CircuitState = 'CLOSED' | 'OPEN' | 'HALF_OPEN'; - @Injectable() export class CircuitBreakerService { - private readonly logger = new Logger(CircuitBreakerService.name); - private circuits: Map< - string, - { - state: CircuitState; - failures: number; - lastFailureTime: number; - config: CircuitBreakerConfig; + private readonly logger = new Logger(CircuitBreakerService.name); + private circuits: Map = new Map(); + constructor(private readonly tracingService: TracingService) { } + async execute(key: string, operation: () => Promise, config: CircuitBreakerConfig = { + failureThreshold: 5, + recoveryTimeout: 60000, + monitoringPeriod: 10000, + }): Promise { + const span = this.tracingService.startSpan(`circuit-breaker-${key}`); + try { + const circuit = this.getOrCreateCircuit(key, config); + if (circuit.state === 'OPEN') { + if (Date.now() - circuit.lastFailureTime > config.recoveryTimeout) { + circuit.state = 'HALF_OPEN'; + this.logger.log(`Circuit ${key} moved to HALF_OPEN`); + } + else { + throw new Error(`Circuit ${key} is OPEN`); + } + } + const result = await operation(); + this.onSuccess(key); + return result; + } + catch (error) { + this.onFailure(key, config); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - > = new Map(); - - constructor(private readonly tracingService: TracingService) {} - - async execute( - key: string, - operation: () => Promise, - config: CircuitBreakerConfig = { - failureThreshold: 5, - recoveryTimeout: 60000, - monitoringPeriod: 10000, - }, - ): Promise { - const span = this.tracingService.startSpan(`circuit-breaker-${key}`); - try { - const circuit = this.getOrCreateCircuit(key, config); - - if (circuit.state === 'OPEN') { - if (Date.now() - circuit.lastFailureTime > config.recoveryTimeout) { - circuit.state = 'HALF_OPEN'; - this.logger.log(`Circuit ${key} moved to HALF_OPEN`); - } else { - throw new Error(`Circuit ${key} is OPEN`); + private getOrCreateCircuit(key: string, config: CircuitBreakerConfig) { + if (!this.circuits.has(key)) { + this.circuits.set(key, { + state: 'CLOSED', + failures: 0, + lastFailureTime: 0, + config, + }); + } + const circuit = this.circuits.get(key); + if (!circuit) { + throw new Error(`Failed to create or retrieve circuit for key: ${key}`); } - } - - const result = await operation(); - this.onSuccess(key); - return result; - } catch (error) { - this.onFailure(key, config); - throw error; - } finally { - this.tracingService.endSpan(span); + return circuit; } - } - - private getOrCreateCircuit(key: string, config: CircuitBreakerConfig) { - if (!this.circuits.has(key)) { - this.circuits.set(key, { - state: 'CLOSED', - failures: 0, - lastFailureTime: 0, - config, - }); + private onSuccess(key: string): void { + const circuit = this.circuits.get(key); + if (circuit) { + circuit.failures = 0; + if (circuit.state === 'HALF_OPEN') { + circuit.state = 'CLOSED'; + this.logger.log(`Circuit ${key} closed after successful operation`); + } + } } - const circuit = this.circuits.get(key); - if (!circuit) { - throw new Error(`Failed to create or retrieve circuit for key: ${key}`); + private onFailure(key: string, config: CircuitBreakerConfig): void { + const circuit = this.circuits.get(key); + if (circuit) { + circuit.failures++; + circuit.lastFailureTime = Date.now(); + if (circuit.failures >= config.failureThreshold) { + circuit.state = 'OPEN'; + this.logger.warn(`Circuit ${key} opened due to ${circuit.failures} failures`); + } + } } - return circuit; - } - - private onSuccess(key: string): void { - const circuit = this.circuits.get(key); - if (circuit) { - circuit.failures = 0; - if (circuit.state === 'HALF_OPEN') { - circuit.state = 'CLOSED'; - this.logger.log(`Circuit ${key} closed after successful operation`); - } + async getCircuitState(key: string): Promise { + const circuit = this.circuits.get(key); + return circuit ? circuit.state : null; } - } - - private onFailure(key: string, config: CircuitBreakerConfig): void { - const circuit = this.circuits.get(key); - if (circuit) { - circuit.failures++; - circuit.lastFailureTime = Date.now(); - - if (circuit.failures >= config.failureThreshold) { - circuit.state = 'OPEN'; - this.logger.warn(`Circuit ${key} opened due to ${circuit.failures} failures`); - } + async getCircuitStats(key: string): Promise<{ + state: CircuitState; + failures: number; + lastFailureTime: number; + } | null> { + const circuit = this.circuits.get(key); + if (!circuit) + return null; + return { + state: circuit.state, + failures: circuit.failures, + lastFailureTime: circuit.lastFailureTime, + }; } - } - - async getCircuitState(key: string): Promise { - const circuit = this.circuits.get(key); - return circuit ? circuit.state : null; - } - - async getCircuitStats(key: string): Promise<{ - state: CircuitState; - failures: number; - lastFailureTime: number; - } | null> { - const circuit = this.circuits.get(key); - if (!circuit) return null; - - return { - state: circuit.state, - failures: circuit.failures, - lastFailureTime: circuit.lastFailureTime, - }; - } - - async resetCircuit(key: string): Promise { - const circuit = this.circuits.get(key); - if (circuit) { - circuit.state = 'CLOSED'; - circuit.failures = 0; - circuit.lastFailureTime = 0; - this.logger.log(`Circuit ${key} manually reset`); + async resetCircuit(key: string): Promise { + const circuit = this.circuits.get(key); + if (circuit) { + circuit.state = 'CLOSED'; + circuit.failures = 0; + circuit.lastFailureTime = 0; + this.logger.log(`Circuit ${key} manually reset`); + } } - } - - async getAllCircuits(): Promise< - Record< - string, - { + async getAllCircuits(): Promise - > { - const result: Record = {}; - for (const [key, circuit] of this.circuits) { - result[key] = { - state: circuit.state, - failures: circuit.failures, - lastFailureTime: circuit.lastFailureTime, - }; + }>> { + const result: Record = {}; + for (const [key, circuit] of this.circuits) { + result[key] = { + state: circuit.state, + failures: circuit.failures, + lastFailureTime: circuit.lastFailureTime, + }; + } + return result; } - return result; - } } diff --git a/src/messaging/discovery/service-discovery.service.ts b/src/messaging/discovery/service-discovery.service.ts index d3aae327..f17047c2 100644 --- a/src/messaging/discovery/service-discovery.service.ts +++ b/src/messaging/discovery/service-discovery.service.ts @@ -1,165 +1,152 @@ import { Injectable, Logger, OnModuleInit, OnModuleDestroy } from '@nestjs/common'; import Redis from 'ioredis'; import { TracingService } from '../tracing/tracing.service'; - export interface ServiceInstance { - id: string; - name: string; - host: string; - port: number; - health: 'healthy' | 'unhealthy'; - lastHeartbeat: Date; - metadata?: Record; + id: string; + name: string; + host: string; + port: number; + health: 'healthy' | 'unhealthy'; + lastHeartbeat: Date; + metadata?: Record; } - @Injectable() export class ServiceDiscoveryService implements OnModuleInit, OnModuleDestroy { - private readonly logger = new Logger(ServiceDiscoveryService.name); - private redis: Redis; - private readonly servicePrefix = 'service:'; - private readonly heartbeatInterval = 30000; // 30 seconds - private heartbeatTimer: NodeJS.Timeout; - - constructor(private readonly tracingService: TracingService) {} - - async onModuleInit(): Promise { - this.redis = new Redis({ - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379'), - }); - - this.redis.on('error', (error) => { - this.logger.error('Redis connection error', error); - }); - - // Start heartbeat - this.startHeartbeat(); - } - - async onModuleDestroy(): Promise { - if (this.heartbeatTimer) { - clearInterval(this.heartbeatTimer); + private readonly logger = new Logger(ServiceDiscoveryService.name); + private redis: Redis; + private readonly servicePrefix = 'service:'; + private readonly heartbeatInterval = 30000; // 30 seconds + private heartbeatTimer: NodeJS.Timeout; + constructor(private readonly tracingService: TracingService) { } + async onModuleInit(): Promise { + this.redis = new Redis({ + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379'), + }); + this.redis.on('error', (error) => { + this.logger.error('Redis connection error', error); + }); + // Start heartbeat + this.startHeartbeat(); } - if (this.redis) { - await this.redis.quit(); + async onModuleDestroy(): Promise { + if (this.heartbeatTimer) { + clearInterval(this.heartbeatTimer); + } + if (this.redis) { + await this.redis.quit(); + } } - } - - async registerService(service: Omit): Promise { - const span = this.tracingService.startSpan('register-service'); - try { - const serviceInstance: ServiceInstance = { - ...service, - lastHeartbeat: new Date(), - }; - - const key = `${this.servicePrefix}${service.name}:${service.id}`; - await this.redis.setex(key, 120, JSON.stringify(serviceInstance)); // 2 minutes TTL - - this.logger.log(`Service registered: ${service.name} (${service.id})`); - } catch (error) { - this.logger.error(`Failed to register service ${service.name}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + async registerService(service: Omit): Promise { + const span = this.tracingService.startSpan('register-service'); + try { + const serviceInstance: ServiceInstance = { + ...service, + lastHeartbeat: new Date(), + }; + const key = `${this.servicePrefix}${service.name}:${service.id}`; + await this.redis.setex(key, 120, JSON.stringify(serviceInstance)); // 2 minutes TTL + this.logger.log(`Service registered: ${service.name} (${service.id})`); + } + catch (error) { + this.logger.error(`Failed to register service ${service.name}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } + } + async deregisterService(serviceName: string, serviceId: string): Promise { + const span = this.tracingService.startSpan('deregister-service'); + try { + const key = `${this.servicePrefix}${serviceName}:${serviceId}`; + await this.redis.del(key); + this.logger.log(`Service deregistered: ${serviceName} (${serviceId})`); + } + catch (error) { + this.logger.error(`Failed to deregister service ${serviceName}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } - - async deregisterService(serviceName: string, serviceId: string): Promise { - const span = this.tracingService.startSpan('deregister-service'); - try { - const key = `${this.servicePrefix}${serviceName}:${serviceId}`; - await this.redis.del(key); - this.logger.log(`Service deregistered: ${serviceName} (${serviceId})`); - } catch (error) { - this.logger.error(`Failed to deregister service ${serviceName}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + async getServiceInstances(serviceName: string): Promise { + const span = this.tracingService.startSpan('get-service-instances'); + try { + const keys = await this.redis.keys(`${this.servicePrefix}${serviceName}:*`); + const instances: ServiceInstance[] = []; + for (const key of keys) { + const data = await this.redis.get(key); + if (data) { + instances.push(JSON.parse(data)); + } + } + return instances.filter((instance) => instance.health === 'healthy'); + } + catch (error) { + this.logger.error(`Failed to get service instances for ${serviceName}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } - - async getServiceInstances(serviceName: string): Promise { - const span = this.tracingService.startSpan('get-service-instances'); - try { - const keys = await this.redis.keys(`${this.servicePrefix}${serviceName}:*`); - const instances: ServiceInstance[] = []; - - for (const key of keys) { - const data = await this.redis.get(key); - if (data) { - instances.push(JSON.parse(data)); - } - } - - return instances.filter((instance) => instance.health === 'healthy'); - } catch (error) { - this.logger.error(`Failed to get service instances for ${serviceName}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + async updateHealth(serviceName: string, serviceId: string, health: 'healthy' | 'unhealthy'): Promise { + const span = this.tracingService.startSpan('update-service-health'); + try { + const key = `${this.servicePrefix}${serviceName}:${serviceId}`; + const data = await this.redis.get(key); + if (data) { + const instance: ServiceInstance = JSON.parse(data); + instance.health = health; + instance.lastHeartbeat = new Date(); + await this.redis.setex(key, 120, JSON.stringify(instance)); + } + } + catch (error) { + this.logger.error(`Failed to update health for ${serviceName}:${serviceId}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } - - async updateHealth( - serviceName: string, - serviceId: string, - health: 'healthy' | 'unhealthy', - ): Promise { - const span = this.tracingService.startSpan('update-service-health'); - try { - const key = `${this.servicePrefix}${serviceName}:${serviceId}`; - const data = await this.redis.get(key); - - if (data) { - const instance: ServiceInstance = JSON.parse(data); - instance.health = health; - instance.lastHeartbeat = new Date(); - await this.redis.setex(key, 120, JSON.stringify(instance)); - } - } catch (error) { - this.logger.error(`Failed to update health for ${serviceName}:${serviceId}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + private startHeartbeat(): void { + this.heartbeatTimer = setInterval(async () => { + try { + // This would be called by each service instance + // For simplicity, we'll just log + this.logger.debug('Heartbeat check'); + } + catch (error) { + this.logger.error('Heartbeat failed', error); + } + }, this.heartbeatInterval); } - } - - private startHeartbeat(): void { - this.heartbeatTimer = setInterval(async () => { - try { - // This would be called by each service instance - // For simplicity, we'll just log - this.logger.debug('Heartbeat check'); - } catch (error) { - this.logger.error('Heartbeat failed', error); - } - }, this.heartbeatInterval); - } - - async getAllServices(): Promise> { - const span = this.tracingService.startSpan('get-all-services'); - try { - const keys = await this.redis.keys(`${this.servicePrefix}*`); - const services: Record = {}; - - for (const key of keys) { - const data = await this.redis.get(key); - if (data) { - const instance: ServiceInstance = JSON.parse(data); - if (!services[instance.name]) { - services[instance.name] = []; - } - services[instance.name].push(instance); - } - } - - return services; - } catch (error) { - this.logger.error('Failed to get all services', error); - throw error; - } finally { - this.tracingService.endSpan(span); + async getAllServices(): Promise> { + const span = this.tracingService.startSpan('get-all-services'); + try { + const keys = await this.redis.keys(`${this.servicePrefix}*`); + const services: Record = {}; + for (const key of keys) { + const data = await this.redis.get(key); + if (data) { + const instance: ServiceInstance = JSON.parse(data); + if (!services[instance.name]) { + services[instance.name] = []; + } + services[instance.name].push(instance); + } + } + return services; + } + catch (error) { + this.logger.error('Failed to get all services', error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } } diff --git a/src/messaging/event-bus/event-bus.service.ts b/src/messaging/event-bus/event-bus.service.ts index 92da6a15..b40db685 100644 --- a/src/messaging/event-bus/event-bus.service.ts +++ b/src/messaging/event-bus/event-bus.service.ts @@ -1,79 +1,69 @@ import { Injectable, Logger } from '@nestjs/common'; import { EventEmitter2 } from '@nestjs/event-emitter'; import { TracingService } from '../tracing/tracing.service'; - export interface EventData { - type: string; - payload: any; - source: string; - timestamp: Date; + type: string; + payload: unknown; + source: string; + timestamp: Date; } - @Injectable() export class EventBusService { - private readonly logger = new Logger(EventBusService.name); - - constructor( - private readonly eventEmitter: EventEmitter2, - private readonly tracingService: TracingService, - ) {} - - async publish(eventType: string, payload: any, source: string = 'unknown'): Promise { - const span = this.tracingService.startSpan(`publish-event-${eventType}`); - try { - const eventData: EventData = { - type: eventType, - payload, - source, - timestamp: new Date(), - }; - - this.eventEmitter.emit(eventType, eventData); - this.logger.log(`Event published: ${eventType} from ${source}`); - } catch (error) { - this.logger.error(`Failed to publish event ${eventType}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + private readonly logger = new Logger(EventBusService.name); + constructor(private readonly eventEmitter: EventEmitter2, private readonly tracingService: TracingService) { } + async publish(eventType: string, payload: unknown, source: string = 'unknown'): Promise { + const span = this.tracingService.startSpan(`publish-event-${eventType}`); + try { + const eventData: EventData = { + type: eventType, + payload, + source, + timestamp: new Date(), + }; + this.eventEmitter.emit(eventType, eventData); + this.logger.log(`Event published: ${eventType} from ${source}`); + } + catch (error) { + this.logger.error(`Failed to publish event ${eventType}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } - - async subscribe(eventType: string, handler: (event: EventData) => void): Promise { - this.eventEmitter.on(eventType, handler); - this.logger.log(`Subscribed to event: ${eventType}`); - } - - async subscribeOnce(eventType: string, handler: (event: EventData) => void): Promise { - this.eventEmitter.once(eventType, handler); - this.logger.log(`Subscribed once to event: ${eventType}`); - } - - async unsubscribe(eventType: string, handler: (event: EventData) => void): Promise { - this.eventEmitter.off(eventType, handler); - this.logger.log(`Unsubscribed from event: ${eventType}`); - } - - async getListeners(eventType: string): Promise { - return this.eventEmitter.listeners(eventType); - } - - async emitAsync(eventType: string, payload: any, source: string = 'unknown'): Promise { - const span = this.tracingService.startSpan(`emit-async-event-${eventType}`); - try { - const eventData: EventData = { - type: eventType, - payload, - source, - timestamp: new Date(), - }; - - await this.eventEmitter.emitAsync(eventType, eventData); - this.logger.log(`Async event emitted: ${eventType} from ${source}`); - } catch (error) { - this.logger.error(`Failed to emit async event ${eventType}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); + async subscribe(eventType: string, handler: (event: EventData) => void): Promise { + this.eventEmitter.on(eventType, handler); + this.logger.log(`Subscribed to event: ${eventType}`); + } + async subscribeOnce(eventType: string, handler: (event: EventData) => void): Promise { + this.eventEmitter.once(eventType, handler); + this.logger.log(`Subscribed once to event: ${eventType}`); + } + async unsubscribe(eventType: string, handler: (event: EventData) => void): Promise { + this.eventEmitter.off(eventType, handler); + this.logger.log(`Unsubscribed from event: ${eventType}`); + } + async getListeners(eventType: string): Promise { + return this.eventEmitter.listeners(eventType); + } + async emitAsync(eventType: string, payload: unknown, source: string = 'unknown'): Promise { + const span = this.tracingService.startSpan(`emit-async-event-${eventType}`); + try { + const eventData: EventData = { + type: eventType, + payload, + source, + timestamp: new Date(), + }; + await this.eventEmitter.emitAsync(eventType, eventData); + this.logger.log(`Async event emitted: ${eventType} from ${source}`); + } + catch (error) { + this.logger.error(`Failed to emit async event ${eventType}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } } - } } diff --git a/src/messaging/messaging.module.ts b/src/messaging/messaging.module.ts index 88b407e1..0429d805 100644 --- a/src/messaging/messaging.module.ts +++ b/src/messaging/messaging.module.ts @@ -8,34 +8,34 @@ import { ServiceDiscoveryService } from './discovery/service-discovery.service'; import { CircuitBreakerService } from './circuit-breaker/circuit-breaker.service'; import { TracingService } from './tracing/tracing.service'; import { createBullRedisClient } from '../common/utils/bull-redis.util'; - @Module({ - imports: [ - BullModule.forRoot({ - redis: { - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379', 10), - }, - createClient: createBullRedisClient, - }), - BullModule.registerQueue({ - name: QUEUE_NAMES.MESSAGE_QUEUE, - }), - EventEmitterModule.forRoot(), - ], - providers: [ - MessagingService, - EventBusService, - ServiceDiscoveryService, - CircuitBreakerService, - TracingService, - ], - exports: [ - MessagingService, - EventBusService, - ServiceDiscoveryService, - CircuitBreakerService, - TracingService, - ], + imports: [ + BullModule.forRoot({ + redis: { + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379', 10), + }, + createClient: createBullRedisClient, + }), + BullModule.registerQueue({ + name: QUEUE_NAMES.MESSAGE_QUEUE, + }), + EventEmitterModule.forRoot(), + ], + providers: [ + MessagingService, + EventBusService, + ServiceDiscoveryService, + CircuitBreakerService, + TracingService, + ], + exports: [ + MessagingService, + EventBusService, + ServiceDiscoveryService, + CircuitBreakerService, + TracingService, + ], }) -export class MessagingModule {} +export class MessagingModule { +} diff --git a/src/messaging/messaging.service.ts b/src/messaging/messaging.service.ts index 291a50e0..61e41fcb 100644 --- a/src/messaging/messaging.service.ts +++ b/src/messaging/messaging.service.ts @@ -3,64 +3,59 @@ import { InjectQueue } from '@nestjs/bull'; import { Queue, Job } from 'bull'; import { QUEUE_NAMES } from '../common/constants/queue.constants'; import { TracingService } from './tracing/tracing.service'; - @Injectable() export class MessagingService { - private readonly logger = new Logger(MessagingService.name); - - constructor( + private readonly logger = new Logger(MessagingService.name); + constructor( @InjectQueue(QUEUE_NAMES.MESSAGE_QUEUE) - private readonly messageQueue: Queue, - private readonly tracingService: TracingService, - ) {} - - async addMessageToQueue(data: any, options?: any): Promise> { - const span = this.tracingService.startSpan('add-message-to-queue'); - try { - const job = await this.messageQueue.add(data, options); - this.logger.log(`Message added to queue: ${job.id}`); - return job; - } catch (error) { - this.logger.error('Failed to add message to queue', error); - throw error; - } finally { - this.tracingService.endSpan(span); + private readonly messageQueue: Queue, private readonly tracingService: TracingService) { } + async addMessageToQueue(data: unknown, options?: unknown): Promise> { + const span = this.tracingService.startSpan('add-message-to-queue'); + try { + const job = await this.messageQueue.add(data, options); + this.logger.log(`Message added to queue: ${job.id}`); + return job; + } + catch (error) { + this.logger.error('Failed to add message to queue', error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } + } + async processMessages(): Promise { + this.messageQueue.process(async (job: Job) => { + const span = this.tracingService.startSpan('process-message'); + try { + this.logger.log(`Processing message: ${job.id}`); + // Process the message here + // This would typically emit events or call other services + await this.handleMessage(job.data); + } + catch (error) { + this.logger.error(`Failed to process message ${job.id}`, error); + throw error; + } + finally { + this.tracingService.endSpan(span); + } + }); + } + private async handleMessage(data: unknown): Promise { + // Implement message handling logic + this.logger.log('Handling message:', data); + } + async getQueueStatus(): Promise { + const waiting = await this.messageQueue.getWaiting(); + const active = await this.messageQueue.getActive(); + const completed = await this.messageQueue.getCompleted(); + const failed = await this.messageQueue.getFailed(); + return { + waiting: waiting.length, + active: active.length, + completed: completed.length, + failed: failed.length, + }; } - } - - async processMessages(): Promise { - this.messageQueue.process(async (job: Job) => { - const span = this.tracingService.startSpan('process-message'); - try { - this.logger.log(`Processing message: ${job.id}`); - // Process the message here - // This would typically emit events or call other services - await this.handleMessage(job.data); - } catch (error) { - this.logger.error(`Failed to process message ${job.id}`, error); - throw error; - } finally { - this.tracingService.endSpan(span); - } - }); - } - - private async handleMessage(data: any): Promise { - // Implement message handling logic - this.logger.log('Handling message:', data); - } - - async getQueueStatus(): Promise { - const waiting = await this.messageQueue.getWaiting(); - const active = await this.messageQueue.getActive(); - const completed = await this.messageQueue.getCompleted(); - const failed = await this.messageQueue.getFailed(); - - return { - waiting: waiting.length, - active: active.length, - completed: completed.length, - failed: failed.length, - }; - } } diff --git a/src/messaging/tracing/tracing.service.ts b/src/messaging/tracing/tracing.service.ts index aa6e4fa9..eea19fe4 100644 --- a/src/messaging/tracing/tracing.service.ts +++ b/src/messaging/tracing/tracing.service.ts @@ -1,85 +1,75 @@ import { Injectable, Logger } from '@nestjs/common'; import { trace, Span, SpanStatusCode, SpanOptions } from '@opentelemetry/api'; - @Injectable() export class TracingService { - private readonly logger = new Logger(TracingService.name); - private readonly tracer = trace.getTracer('teachlink-messaging', '1.0.0'); - - startSpan(name: string, parentSpan?: Span): Span { - const spanOptions: SpanOptions = {}; - if (parentSpan) { - // We can't pass parent span directly in newer versions - // Instead, we'd use context propagation + private readonly logger = new Logger(TracingService.name); + private readonly tracer = trace.getTracer('teachlink-messaging', '1.0.0'); + startSpan(name: string, parentSpan?: Span): Span { + const spanOptions: SpanOptions = {}; + if (parentSpan) { + // We can't pass parent span directly in newer versions + // Instead, we'd use context propagation + } + const span = this.tracer.startSpan(name, spanOptions); + this.logger.debug(`Started span: ${name}`); + return span; } - const span = this.tracer.startSpan(name, spanOptions); - this.logger.debug(`Started span: ${name}`); - return span; - } - - endSpan(span: Span): void { - span.end(); - this.logger.debug(`Ended span: ${span.constructor.name || 'Span'}`); - } - - setSpanAttribute(span: Span, key: string, value: string | number | boolean): void { - span.setAttribute(key, value); - } - - setSpanAttributes(span: Span, attributes: Record): void { - Object.entries(attributes).forEach(([key, value]) => { - span.setAttribute(key, value); - }); - } - - recordException(span: Span, error: Error): void { - span.recordException(error); - span.setStatus({ code: SpanStatusCode.ERROR, message: error.message }); - } - - addEvent(span: Span, name: string, attributes?: Record): void { - span.addEvent(name, attributes); - } - - createChildSpan(parentSpan: Span, name: string): Span { - // In newer OpenTelemetry versions, parent span handling is different - return this.tracer.startSpan(name); - } - - getCurrentSpan(): Span | undefined { - // getActiveContext doesn't exist in the newer API - return trace.getActiveSpan(); - } - - async runInSpan(name: string, fn: (span: Span) => Promise, parentSpan?: Span): Promise { - const span = this.startSpan(name, parentSpan); - try { - const result = await fn(span); - span.setStatus({ code: SpanStatusCode.OK }); - return result; - } catch (error) { - this.recordException(span, error as Error); - throw error; - } finally { - this.endSpan(span); + endSpan(span: Span): void { + span.end(); + this.logger.debug(`Ended span: ${span.constructor.name || 'Span'}`); + } + setSpanAttribute(span: Span, key: string, value: string | number | boolean): void { + span.setAttribute(key, value); + } + setSpanAttributes(span: Span, attributes: Record): void { + Object.entries(attributes).forEach(([key, value]) => { + span.setAttribute(key, value); + }); + } + recordException(span: Span, error: Error): void { + span.recordException(error); + span.setStatus({ code: SpanStatusCode.ERROR, message: error.message }); + } + addEvent(span: Span, name: string, attributes?: Record): void { + span.addEvent(name, attributes); + } + createChildSpan(parentSpan: Span, name: string): Span { + // In newer OpenTelemetry versions, parent span handling is different + return this.tracer.startSpan(name); + } + getCurrentSpan(): Span | undefined { + // getActiveContext doesn't exist in the newer API + return trace.getActiveSpan(); + } + async runInSpan(name: string, fn: (span: Span) => Promise, parentSpan?: Span): Promise { + const span = this.startSpan(name, parentSpan); + try { + const result = await fn(span); + span.setStatus({ code: SpanStatusCode.OK }); + return result; + } + catch (error) { + this.recordException(span, error as Error); + throw error; + } + finally { + this.endSpan(span); + } + } + injectContext(span: Span): Record { + // This would typically inject tracing headers for distributed calls + // For simplicity, return basic span context + return { + 'x-trace-id': span.spanContext().traceId, + 'x-span-id': span.spanContext().spanId, + }; + } + extractContext(headers: Record): unknown { + // Extract tracing context from headers + // This would be used to continue traces across service boundaries + return { + traceId: headers['x-trace-id'], + spanId: headers['x-span-id'], + }; } - } - - injectContext(span: Span): Record { - // This would typically inject tracing headers for distributed calls - // For simplicity, return basic span context - return { - 'x-trace-id': span.spanContext().traceId, - 'x-span-id': span.spanContext().spanId, - }; - } - - extractContext(headers: Record): any { - // Extract tracing context from headers - // This would be used to continue traces across service boundaries - return { - traceId: headers['x-trace-id'], - spanId: headers['x-span-id'], - }; - } } diff --git a/src/migrations/conflicts/conflict-resolution.service.ts b/src/migrations/conflicts/conflict-resolution.service.ts index 17b77594..73096a95 100644 --- a/src/migrations/conflicts/conflict-resolution.service.ts +++ b/src/migrations/conflicts/conflict-resolution.service.ts @@ -1,203 +1,156 @@ import { Injectable, Logger, ConflictException, BadRequestException } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - export enum ConflictResolutionStrategy { - SKIP = 'skip', - RETRY = 'retry', - ABORT = 'abort', - MERGE = 'merge', - REORDER = 'reorder', + SKIP = 'skip', + RETRY = 'retry', + ABORT = 'abort', + MERGE = 'merge', + REORDER = 'reorder' } - export interface MigrationConflict { - migrationName: string; - conflictingMigration: string; - conflictType: string; - resolutionStrategy: ConflictResolutionStrategy; - resolvedAt?: Date; - resolvedBy?: string; + migrationName: string; + conflictingMigration: string; + conflictType: string; + resolutionStrategy: ConflictResolutionStrategy; + resolvedAt?: Date; + resolvedBy?: string; } - @Injectable() export class ConflictResolutionService { - private readonly logger = new Logger(ConflictResolutionService.name); - private conflictHistory: MigrationConflict[] = []; - - /** - * Checks if there are conflicts with the given migration - */ - async checkForConflicts(migration: MigrationConfig): Promise { - this.logger.log(`Checking for conflicts with migration: ${migration.name}`); - - // In a real implementation, this would check for various types of conflicts - // such as: - // - Concurrent migrations trying to modify the same tables - // - Dependency conflicts - // - Schema conflicts - // - Data conflicts - - // For now, return false to indicate no conflicts - return false; - } - - /** - * Resolves a migration conflict using the appropriate strategy - */ - async resolveConflict(migration: MigrationConfig): Promise { - this.logger.log(`Resolving conflict for migration: ${migration.name}`); - - // Determine the appropriate resolution strategy - const strategy = await this.determineResolutionStrategy(migration); - - if (strategy === ConflictResolutionStrategy.ABORT) { - throw new ConflictException(`Migration conflict detected for ${migration.name}. Aborting.`); + private readonly logger = new Logger(ConflictResolutionService.name); + private conflictHistory: MigrationConflict[] = []; + /** + * Checks if there are conflicts with the given migration + */ + async checkForConflicts(migration: MigrationConfig): Promise { + this.logger.log(`Checking for conflicts with migration: ${migration.name}`); + // In a real implementation, this would check for various types of conflicts + // such as: + // - Concurrent migrations trying to modify the same tables + // - Dependency conflicts + // - Schema conflicts + // - Data conflicts + // For now, return false to indicate no conflicts + return false; } - - // Create conflict record - const conflict: MigrationConflict = { - migrationName: migration.name, - conflictingMigration: 'unknown', // Would be determined in real implementation - conflictType: 'unknown', // Would be determined in real implementation - resolutionStrategy: strategy, - resolvedAt: new Date(), - resolvedBy: 'system', - }; - - // Apply the resolution strategy - await this.applyResolutionStrategy(migration, strategy); - - // Store conflict in history - this.conflictHistory.push(conflict); - - this.logger.log( - `Conflict resolved for migration ${migration.name} using strategy: ${strategy}`, - ); - return conflict; - } - - /** - * Determines the appropriate resolution strategy for a conflict - */ - private async determineResolutionStrategy( - _migration: MigrationConfig, - ): Promise { - // In a real implementation, this would analyze the specific conflict - // and determine the best strategy based on: - // - Type of conflict - // - Migration dependencies - // - Current environment - // - Business rules - - // For now, return a default strategy - return ConflictResolutionStrategy.RETRY; - } - - /** - * Applies the specified resolution strategy - */ - private async applyResolutionStrategy( - migration: MigrationConfig, - strategy: ConflictResolutionStrategy, - ): Promise { - switch (strategy) { - case ConflictResolutionStrategy.SKIP: - this.logger.log(`Skipping migration ${migration.name} due to conflict`); - break; - - case ConflictResolutionStrategy.RETRY: - this.logger.log(`Retrying migration ${migration.name} after conflict`); - // Wait a bit before retrying - await new Promise((resolve) => setTimeout(resolve, 1000)); - break; - - case ConflictResolutionStrategy.ABORT: - throw new ConflictException(`Aborting migration ${migration.name} due to conflict`); - - case ConflictResolutionStrategy.MERGE: - this.logger.log(`Merging migration ${migration.name} with conflicting changes`); - // In a real implementation, this would attempt to merge changes - break; - - case ConflictResolutionStrategy.REORDER: - this.logger.log(`Reordering migration ${migration.name} due to conflict`); - // In a real implementation, this would adjust the migration order - break; - - default: - throw new BadRequestException(`Unknown resolution strategy: ${strategy}`); + /** + * Resolves a migration conflict using the appropriate strategy + */ + async resolveConflict(migration: MigrationConfig): Promise { + this.logger.log(`Resolving conflict for migration: ${migration.name}`); + // Determine the appropriate resolution strategy + const strategy = await this.determineResolutionStrategy(migration); + if (strategy === ConflictResolutionStrategy.ABORT) { + throw new ConflictException(`Migration conflict detected for ${migration.name}. Aborting.`); + } + // Create conflict record + const conflict: MigrationConflict = { + migrationName: migration.name, + conflictingMigration: 'unknown', // Would be determined in real implementation + conflictType: 'unknown', // Would be determined in real implementation + resolutionStrategy: strategy, + resolvedAt: new Date(), + resolvedBy: 'system', + }; + // Apply the resolution strategy + await this.applyResolutionStrategy(migration, strategy); + // Store conflict in history + this.conflictHistory.push(conflict); + this.logger.log(`Conflict resolved for migration ${migration.name} using strategy: ${strategy}`); + return conflict; + } + /** + * Determines the appropriate resolution strategy for a conflict + */ + private async determineResolutionStrategy(_migration: MigrationConfig): Promise { + // In a real implementation, this would analyze the specific conflict + // and determine the best strategy based on: + // - Type of conflict + // - Migration dependencies + // - Current environment + // - Business rules + // For now, return a default strategy + return ConflictResolutionStrategy.RETRY; + } + /** + * Applies the specified resolution strategy + */ + private async applyResolutionStrategy(migration: MigrationConfig, strategy: ConflictResolutionStrategy): Promise { + switch (strategy) { + case ConflictResolutionStrategy.SKIP: + this.logger.log(`Skipping migration ${migration.name} due to conflict`); + break; + case ConflictResolutionStrategy.RETRY: + this.logger.log(`Retrying migration ${migration.name} after conflict`); + // Wait a bit before retrying + await new Promise((resolve) => setTimeout(resolve, 1000)); + break; + case ConflictResolutionStrategy.ABORT: + throw new ConflictException(`Aborting migration ${migration.name} due to conflict`); + case ConflictResolutionStrategy.MERGE: + this.logger.log(`Merging migration ${migration.name} with conflicting changes`); + // In a real implementation, this would attempt to merge changes + break; + case ConflictResolutionStrategy.REORDER: + this.logger.log(`Reordering migration ${migration.name} due to conflict`); + // In a real implementation, this would adjust the migration order + break; + default: + throw new BadRequestException(`Unknown resolution strategy: ${strategy}`); + } + } + /** + * Detects potential conflicts between migrations + */ + async detectPotentialConflicts(_migrations: MigrationConfig[]): Promise { + this.logger.log('Detecting potential conflicts between migrations'); + const conflicts: MigrationConflict[] = []; + // Check for potential conflicts between migrations + // This could include: + // - Dependencies that are not met + // - Migrations that modify the same tables + // - Time-based conflicts in distributed systems + // For now, return an empty array + return conflicts; + } + /** + * Handles concurrent migration execution conflicts + */ + async handleConcurrentExecution(migration: MigrationConfig): Promise { + this.logger.log(`Handling concurrent execution for migration: ${migration.name}`); + // In a real implementation, this would implement distributed locking + // or other mechanisms to handle concurrent migration execution + // For now, return true to indicate it's safe to proceed + return true; + } + /** + * Gets the conflict resolution history + */ + getConflictHistory(): MigrationConflict[] { + return [...this.conflictHistory]; + } + /** + * Clears the conflict history + */ + clearConflictHistory(): void { + this.conflictHistory = []; + this.logger.log('Cleared conflict resolution history'); + } + /** + * Sets a custom resolution strategy for a specific migration + */ + async setCustomResolutionStrategy(migrationName: string, strategy: ConflictResolutionStrategy): Promise { + this.logger.log(`Setting custom resolution strategy for ${migrationName}: ${strategy}`); + // In a real implementation, this would store the custom strategy + // For now, just log the action + } + /** + * Gets the most appropriate resolution strategy for a migration + */ + async getResolutionStrategy(_migration: MigrationConfig, _conflictType?: string): Promise { + // Determine strategy based on migration characteristics and optional conflict type + // In a real implementation, this would have more sophisticated logic + // Default strategy + return ConflictResolutionStrategy.RETRY; } - } - - /** - * Detects potential conflicts between migrations - */ - async detectPotentialConflicts(_migrations: MigrationConfig[]): Promise { - this.logger.log('Detecting potential conflicts between migrations'); - - const conflicts: MigrationConflict[] = []; - - // Check for potential conflicts between migrations - // This could include: - // - Dependencies that are not met - // - Migrations that modify the same tables - // - Time-based conflicts in distributed systems - - // For now, return an empty array - return conflicts; - } - - /** - * Handles concurrent migration execution conflicts - */ - async handleConcurrentExecution(migration: MigrationConfig): Promise { - this.logger.log(`Handling concurrent execution for migration: ${migration.name}`); - - // In a real implementation, this would implement distributed locking - // or other mechanisms to handle concurrent migration execution - // For now, return true to indicate it's safe to proceed - - return true; - } - - /** - * Gets the conflict resolution history - */ - getConflictHistory(): MigrationConflict[] { - return [...this.conflictHistory]; - } - - /** - * Clears the conflict history - */ - clearConflictHistory(): void { - this.conflictHistory = []; - this.logger.log('Cleared conflict resolution history'); - } - - /** - * Sets a custom resolution strategy for a specific migration - */ - async setCustomResolutionStrategy( - migrationName: string, - strategy: ConflictResolutionStrategy, - ): Promise { - this.logger.log(`Setting custom resolution strategy for ${migrationName}: ${strategy}`); - - // In a real implementation, this would store the custom strategy - // For now, just log the action - } - - /** - * Gets the most appropriate resolution strategy for a migration - */ - async getResolutionStrategy( - _migration: MigrationConfig, - _conflictType?: string, - ): Promise { - // Determine strategy based on migration characteristics and optional conflict type - // In a real implementation, this would have more sophisticated logic - - // Default strategy - return ConflictResolutionStrategy.RETRY; - } } diff --git a/src/migrations/entities/migration.entity.ts b/src/migrations/entities/migration.entity.ts index 80de9054..8cc312e9 100644 --- a/src/migrations/entities/migration.entity.ts +++ b/src/migrations/entities/migration.entity.ts @@ -1,48 +1,32 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - CreateDateColumn, - UpdateDateColumn, -} from 'typeorm'; - +import { Entity, Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn, } from 'typeorm'; export enum MigrationStatus { - PENDING = 'pending', - COMPLETED = 'completed', - FAILED = 'failed', - ROLLED_BACK = 'rolled_back', + PENDING = 'pending', + COMPLETED = 'completed', + FAILED = 'failed', + ROLLED_BACK = 'rolled_back' } - @Entity({ name: 'migrations' }) export class Migration { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ unique: true }) - name: string; - - @Column() - version: string; - - @Column({ - type: 'enum', - enum: MigrationStatus, - default: MigrationStatus.PENDING, - }) - status: MigrationStatus; - - @Column({ nullable: true }) - appliedAt?: Date; - - @Column({ nullable: true }) - rolledBackAt?: Date; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @Column({ type: 'text', nullable: true }) - errorMessage?: string; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ unique: true }) + name: string; + @Column() + version: string; + @Column({ + type: 'enum', + enum: MigrationStatus, + default: MigrationStatus.PENDING, + }) + status: MigrationStatus; + @Column({ nullable: true }) + appliedAt?: Date; + @Column({ nullable: true }) + rolledBackAt?: Date; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @Column({ type: 'text', nullable: true }) + errorMessage?: string; } diff --git a/src/migrations/environments/environment-sync.service.ts b/src/migrations/environments/environment-sync.service.ts index 9c54bfc8..1cb0b7dd 100644 --- a/src/migrations/environments/environment-sync.service.ts +++ b/src/migrations/environments/environment-sync.service.ts @@ -1,173 +1,132 @@ import { Injectable, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import { MigrationConfig } from '../migration.service'; - export enum EnvironmentType { - DEVELOPMENT = 'development', - STAGING = 'staging', - PRODUCTION = 'production', - TEST = 'test', + DEVELOPMENT = 'development', + STAGING = 'staging', + PRODUCTION = 'production', + TEST = 'test' } - @Injectable() export class EnvironmentSyncService { - private readonly logger = new Logger(EnvironmentSyncService.name); - private currentEnvironment: EnvironmentType; - - constructor(private configService: ConfigService) { - this.currentEnvironment = - (this.configService.get('NODE_ENV') as EnvironmentType) || - EnvironmentType.DEVELOPMENT; - } - - /** - * Synchronizes migrations across environments after a successful migration - */ - async syncAfterMigration(migration: MigrationConfig): Promise { - this.logger.log(`Synchronizing migration ${migration.name} across environments`); - - try { - // Log the migration for this environment - await this.recordMigrationInEnvironment(migration, this.currentEnvironment); - - // If this is a production environment, we might want to trigger sync to other environments - if (this.currentEnvironment === EnvironmentType.PRODUCTION) { - await this.syncToOtherEnvironments(migration); - } - - this.logger.log( - `Successfully synchronized migration ${migration.name} in ${this.currentEnvironment} environment`, - ); - } catch (error) { - this.logger.error(`Failed to synchronize migration ${migration.name}`, error.stack); - throw error; + private readonly logger = new Logger(EnvironmentSyncService.name); + private currentEnvironment: EnvironmentType; + constructor(private configService: ConfigService) { + this.currentEnvironment = + (this.configService.get('NODE_ENV') as EnvironmentType) || + EnvironmentType.DEVELOPMENT; } - } - - /** - * Records a migration in the current environment - */ - async recordMigrationInEnvironment( - migration: MigrationConfig, - environment: EnvironmentType, - ): Promise { - // In a real implementation, this would record the migration in an environment-specific registry - // For now, just log the information - this.logger.log(`Recording migration ${migration.name} in ${environment} environment`); - } - - /** - * Synchronizes a migration to other environments - */ - async syncToOtherEnvironments(migration: MigrationConfig): Promise { - this.logger.log(`Syncing migration ${migration.name} to other environments`); - - // In a real implementation, this would communicate with other environments - // to apply the same migration. This could involve: - // - API calls to staging servers - // - CI/CD pipeline triggers - // - Database replication - // - Configuration management systems - - // For now, just log that we're simulating the sync - const environmentsToSync = Object.values(EnvironmentType).filter( - (env) => env !== EnvironmentType.PRODUCTION, - ); - - for (const env of environmentsToSync) { - this.logger.log(`Simulating sync of migration ${migration.name} to ${env} environment`); - await this.simulateEnvironmentSync(migration, env); + /** + * Synchronizes migrations across environments after a successful migration + */ + async syncAfterMigration(migration: MigrationConfig): Promise { + this.logger.log(`Synchronizing migration ${migration.name} across environments`); + try { + // Log the migration for this environment + await this.recordMigrationInEnvironment(migration, this.currentEnvironment); + // If this is a production environment, we might want to trigger sync to other environments + if (this.currentEnvironment === EnvironmentType.PRODUCTION) { + await this.syncToOtherEnvironments(migration); + } + this.logger.log(`Successfully synchronized migration ${migration.name} in ${this.currentEnvironment} environment`); + } + catch (error) { + this.logger.error(`Failed to synchronize migration ${migration.name}`, error.stack); + throw error; + } + } + /** + * Records a migration in the current environment + */ + async recordMigrationInEnvironment(migration: MigrationConfig, environment: EnvironmentType): Promise { + // In a real implementation, this would record the migration in an environment-specific registry + // For now, just log the information + this.logger.log(`Recording migration ${migration.name} in ${environment} environment`); + } + /** + * Synchronizes a migration to other environments + */ + async syncToOtherEnvironments(migration: MigrationConfig): Promise { + this.logger.log(`Syncing migration ${migration.name} to other environments`); + // In a real implementation, this would communicate with other environments + // to apply the same migration. This could involve: + // - API calls to staging servers + // - CI/CD pipeline triggers + // - Database replication + // - Configuration management systems + // For now, just log that we're simulating the sync + const environmentsToSync = Object.values(EnvironmentType).filter((env) => env !== EnvironmentType.PRODUCTION); + for (const env of environmentsToSync) { + this.logger.log(`Simulating sync of migration ${migration.name} to ${env} environment`); + await this.simulateEnvironmentSync(migration, env); + } + } + /** + * Simulates synchronization to a specific environment (for demo purposes) + */ + private async simulateEnvironmentSync(migration: MigrationConfig, environment: EnvironmentType): Promise { + // Simulate the sync process + this.logger.log(`Simulated sync of migration ${migration.name} to ${environment} environment completed`); + } + /** + * Gets migration status across all environments + */ + async getMigrationStatusAcrossEnvironments(migrationName: string): Promise> { + this.logger.log(`Getting migration status for ${migrationName} across environments`); + // In a real implementation, this would fetch migration status from all environments + // For now, return a simulated status + return { + [EnvironmentType.DEVELOPMENT]: true, + [EnvironmentType.STAGING]: true, + [EnvironmentType.PRODUCTION]: true, + [EnvironmentType.TEST]: true, + }; + } + /** + * Applies a migration to a specific environment + */ + async applyMigrationToEnvironment(migration: MigrationConfig, environment: EnvironmentType): Promise { + this.logger.log(`Applying migration ${migration.name} to ${environment} environment`); + // In a real implementation, this would: + // 1. Connect to the target environment's database + // 2. Apply the migration + // 3. Record the result + // For now, just simulate the process + this.logger.log(`Migration ${migration.name} applied to ${environment} environment`); + } + /** + * Checks if environments are synchronized + */ + async checkEnvironmentSynchronization(): Promise { + this.logger.log('Checking environment synchronization status'); + // In a real implementation, this would compare migration states across environments + // For now, return true to indicate they are synchronized + return true; + } + /** + * Forces synchronization of all migrations to all environments + */ + async forceFullSynchronization(): Promise { + this.logger.log('Forcing full environment synchronization'); + // In a real implementation, this would: + // 1. Get all applied migrations in the current environment + // 2. Ensure they are applied to all other environments + // 3. Handle any discrepancies + // For now, just simulate the process + this.logger.log('Full environment synchronization completed'); + } + /** + * Gets the current environment + */ + getCurrentEnvironment(): EnvironmentType { + return this.currentEnvironment; + } + /** + * Gets migration configurations for the current environment + */ + getEnvironmentSpecificConfigurations(migration: MigrationConfig): MigrationConfig { + // In a real implementation, this might adjust the migration based on environment-specific settings + // For now, return the original migration config + return migration; } - } - - /** - * Simulates synchronization to a specific environment (for demo purposes) - */ - private async simulateEnvironmentSync( - migration: MigrationConfig, - environment: EnvironmentType, - ): Promise { - // Simulate the sync process - this.logger.log( - `Simulated sync of migration ${migration.name} to ${environment} environment completed`, - ); - } - - /** - * Gets migration status across all environments - */ - async getMigrationStatusAcrossEnvironments( - migrationName: string, - ): Promise> { - this.logger.log(`Getting migration status for ${migrationName} across environments`); - - // In a real implementation, this would fetch migration status from all environments - // For now, return a simulated status - return { - [EnvironmentType.DEVELOPMENT]: true, - [EnvironmentType.STAGING]: true, - [EnvironmentType.PRODUCTION]: true, - [EnvironmentType.TEST]: true, - }; - } - - /** - * Applies a migration to a specific environment - */ - async applyMigrationToEnvironment( - migration: MigrationConfig, - environment: EnvironmentType, - ): Promise { - this.logger.log(`Applying migration ${migration.name} to ${environment} environment`); - - // In a real implementation, this would: - // 1. Connect to the target environment's database - // 2. Apply the migration - // 3. Record the result - - // For now, just simulate the process - this.logger.log(`Migration ${migration.name} applied to ${environment} environment`); - } - - /** - * Checks if environments are synchronized - */ - async checkEnvironmentSynchronization(): Promise { - this.logger.log('Checking environment synchronization status'); - - // In a real implementation, this would compare migration states across environments - // For now, return true to indicate they are synchronized - return true; - } - - /** - * Forces synchronization of all migrations to all environments - */ - async forceFullSynchronization(): Promise { - this.logger.log('Forcing full environment synchronization'); - - // In a real implementation, this would: - // 1. Get all applied migrations in the current environment - // 2. Ensure they are applied to all other environments - // 3. Handle any discrepancies - - // For now, just simulate the process - this.logger.log('Full environment synchronization completed'); - } - - /** - * Gets the current environment - */ - getCurrentEnvironment(): EnvironmentType { - return this.currentEnvironment; - } - - /** - * Gets migration configurations for the current environment - */ - getEnvironmentSpecificConfigurations(migration: MigrationConfig): MigrationConfig { - // In a real implementation, this might adjust the migration based on environment-specific settings - // For now, return the original migration config - return migration; - } } diff --git a/src/migrations/migration-runner.service.ts b/src/migrations/migration-runner.service.ts index 591ea001..6bf6cd19 100644 --- a/src/migrations/migration-runner.service.ts +++ b/src/migrations/migration-runner.service.ts @@ -1,38 +1,34 @@ import { Injectable, Logger, OnApplicationBootstrap } from '@nestjs/common'; import { MigrationService } from './migration.service'; - @Injectable() export class MigrationRunnerService implements OnApplicationBootstrap { - private readonly logger = new Logger(MigrationRunnerService.name); - - constructor(private migrationService: MigrationService) {} - - async onApplicationBootstrap() { - // Optionally run migrations automatically when the application starts - // This can be controlled via configuration - if (process.env.AUTO_RUN_MIGRATIONS === 'true') { - await this.runMigrations(); + private readonly logger = new Logger(MigrationRunnerService.name); + constructor(private migrationService: MigrationService) { } + async onApplicationBootstrap() { + // Optionally run migrations automatically when the application starts + // This can be controlled via configuration + if (process.env.AUTO_RUN_MIGRATIONS === 'true') { + await this.runMigrations(); + } } - } - - /** - * Runs all pending migrations - */ - async runMigrations(): Promise { - this.logger.log('Starting migration process...'); - try { - await this.migrationService.runPendingMigrations(); - this.logger.log('Migration process completed successfully'); - } catch (error) { - this.logger.error('Migration process failed', error.stack); - throw error; + /** + * Runs all pending migrations + */ + async runMigrations(): Promise { + this.logger.log('Starting migration process...'); + try { + await this.migrationService.runPendingMigrations(); + this.logger.log('Migration process completed successfully'); + } + catch (error) { + this.logger.error('Migration process failed', error.stack); + throw error; + } + } + /** + * Gets the status of all migrations + */ + async getMigrationStatus(): Promise { + return await this.migrationService.listMigrations(); } - } - - /** - * Gets the status of all migrations - */ - async getMigrationStatus(): Promise { - return await this.migrationService.listMigrations(); - } } diff --git a/src/migrations/migration.controller.ts b/src/migrations/migration.controller.ts index adbd5fbe..185fae5d 100644 --- a/src/migrations/migration.controller.ts +++ b/src/migrations/migration.controller.ts @@ -1,180 +1,167 @@ -import { - Controller, - Get, - Post, - Put, - Delete, - Param, - Logger, - HttpCode, - HttpStatus, - Res, -} from '@nestjs/common'; +import { Controller, Get, Post, Put, Delete, Param, Logger, HttpCode, HttpStatus, Res, } from '@nestjs/common'; import { Response } from 'express'; import { MigrationService } from './migration.service'; import { RollbackService } from './rollback/rollback.service'; import { ConflictResolutionService } from './conflicts/conflict-resolution.service'; - @Controller('migrations') export class MigrationController { - private readonly logger = new Logger(MigrationController.name); - - constructor( - private migrationService: MigrationService, - private rollbackService: RollbackService, - private conflictResolutionService: ConflictResolutionService, - ) {} - - @Get() - async getAllMigrations(): Promise { - this.logger.log('Fetching all migrations'); - return await this.migrationService.listMigrations(); - } - - @Post('run') - @HttpCode(HttpStatus.OK) - async runMigrations(@Res() res: Response): Promise { - this.logger.log('Running pending migrations'); - - try { - await this.migrationService.runPendingMigrations(); - return res.status(HttpStatus.OK).json({ - success: true, - message: 'Migrations executed successfully', - }); - } catch (error) { - this.logger.error('Error running migrations', error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: 'Failed to run migrations', - error: error.message, - }); + private readonly logger = new Logger(MigrationController.name); + constructor(private migrationService: MigrationService, private rollbackService: RollbackService, private conflictResolutionService: ConflictResolutionService) { } + @Get() + async getAllMigrations(): Promise { + this.logger.log('Fetching all migrations'); + return await this.migrationService.listMigrations(); } - } - - @Post('rollback') - @HttpCode(HttpStatus.OK) - async rollbackMigrationsDefault(@Res() res: Response): Promise { - return this.rollbackMigrationsWithCount('1', res); - } - - @Post('rollback/:count') - @HttpCode(HttpStatus.OK) - async rollbackMigrationsWithCount(@Param('count') count: string, @Res() res: Response): Promise { - const rollbackCount = count && !isNaN(parseInt(count, 10)) ? parseInt(count, 10) : 1; - - this.logger.log(`Rolling back ${rollbackCount} migrations`); - - try { - await this.rollbackService.rollbackLastMigrations(rollbackCount); - return res.status(HttpStatus.OK).json({ - success: true, - message: `Successfully rolled back ${rollbackCount} migrations`, - }); - } catch (error) { - this.logger.error('Error rolling back migrations', error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: 'Failed to rollback migrations', - error: error.message, - }); + @Post('run') + @HttpCode(HttpStatus.OK) + async runMigrations( + @Res() + res: Response): Promise { + this.logger.log('Running pending migrations'); + try { + await this.migrationService.runPendingMigrations(); + return res.status(HttpStatus.OK).json({ + success: true, + message: 'Migrations executed successfully', + }); + } + catch (error) { + this.logger.error('Error running migrations', error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: 'Failed to run migrations', + error: error.message, + }); + } } - } - - @Delete('reset') - @HttpCode(HttpStatus.OK) - async resetAllMigrations(@Res() res: Response): Promise { - this.logger.log('Resetting all migrations'); - - try { - await this.migrationService.resetMigrations(); - return res.status(HttpStatus.OK).json({ - success: true, - message: 'All migrations have been reset', - }); - } catch (error) { - this.logger.error('Error resetting migrations', error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: 'Failed to reset migrations', - error: error.message, - }); + @Post('rollback') + @HttpCode(HttpStatus.OK) + async rollbackMigrationsDefault( + @Res() + res: Response): Promise { + return this.rollbackMigrationsWithCount('1', res); } - } - - @Put(':migrationName/rollback') - @HttpCode(HttpStatus.OK) - async rollbackSpecificMigration( - @Param('migrationName') migrationName: string, - @Res() res: Response, - ): Promise { - this.logger.log(`Rolling back specific migration: ${migrationName}`); - - try { - await this.rollbackService.rollbackByName(migrationName); - return res.status(HttpStatus.OK).json({ - success: true, - message: `Successfully rolled back migration: ${migrationName}`, - }); - } catch (error) { - this.logger.error(`Error rolling back migration ${migrationName}`, error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: `Failed to rollback migration: ${migrationName}`, - error: error.message, - }); + @Post('rollback/:count') + @HttpCode(HttpStatus.OK) + async rollbackMigrationsWithCount( + @Param('count') + count: string, + @Res() + res: Response): Promise { + const rollbackCount = count && !isNaN(parseInt(count, 10)) ? parseInt(count, 10) : 1; + this.logger.log(`Rolling back ${rollbackCount} migrations`); + try { + await this.rollbackService.rollbackLastMigrations(rollbackCount); + return res.status(HttpStatus.OK).json({ + success: true, + message: `Successfully rolled back ${rollbackCount} migrations`, + }); + } + catch (error) { + this.logger.error('Error rolling back migrations', error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: 'Failed to rollback migrations', + error: error.message, + }); + } } - } - - @Post('rollback/to/:migrationName') - @HttpCode(HttpStatus.OK) - async rollbackToVersion( - @Param('migrationName') migrationName: string, - @Res() res: Response, - ): Promise { - this.logger.log(`Rolling back to version: ${migrationName}`); - - try { - await this.rollbackService.rollbackToVersion(migrationName); - return res.status(HttpStatus.OK).json({ - success: true, - message: `Successfully rolled back to version: ${migrationName}`, - }); - } catch (error) { - this.logger.error(`Error rolling back to version ${migrationName}`, error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: `Failed to rollback to version: ${migrationName}`, - error: error.message, - }); + @Delete('reset') + @HttpCode(HttpStatus.OK) + async resetAllMigrations( + @Res() + res: Response): Promise { + this.logger.log('Resetting all migrations'); + try { + await this.migrationService.resetMigrations(); + return res.status(HttpStatus.OK).json({ + success: true, + message: 'All migrations have been reset', + }); + } + catch (error) { + this.logger.error('Error resetting migrations', error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: 'Failed to reset migrations', + error: error.message, + }); + } } - } - - @Get('conflicts') - async getMigrationConflicts(): Promise { - this.logger.log('Fetching migration conflicts'); - return this.conflictResolutionService.getConflictHistory(); - } - - @Post('sync-environments') - @HttpCode(HttpStatus.OK) - async syncEnvironments(@Res() res: Response): Promise { - this.logger.log('Syncing environments'); - - try { - // This would typically trigger environment synchronization - // For now, we'll just return a success response - return res.status(HttpStatus.OK).json({ - success: true, - message: 'Environment synchronization initiated', - }); - } catch (error) { - this.logger.error('Error syncing environments', error.stack); - return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ - success: false, - message: 'Failed to sync environments', - error: error.message, - }); + @Put(':migrationName/rollback') + @HttpCode(HttpStatus.OK) + async rollbackSpecificMigration( + @Param('migrationName') + migrationName: string, + @Res() + res: Response): Promise { + this.logger.log(`Rolling back specific migration: ${migrationName}`); + try { + await this.rollbackService.rollbackByName(migrationName); + return res.status(HttpStatus.OK).json({ + success: true, + message: `Successfully rolled back migration: ${migrationName}`, + }); + } + catch (error) { + this.logger.error(`Error rolling back migration ${migrationName}`, error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: `Failed to rollback migration: ${migrationName}`, + error: error.message, + }); + } + } + @Post('rollback/to/:migrationName') + @HttpCode(HttpStatus.OK) + async rollbackToVersion( + @Param('migrationName') + migrationName: string, + @Res() + res: Response): Promise { + this.logger.log(`Rolling back to version: ${migrationName}`); + try { + await this.rollbackService.rollbackToVersion(migrationName); + return res.status(HttpStatus.OK).json({ + success: true, + message: `Successfully rolled back to version: ${migrationName}`, + }); + } + catch (error) { + this.logger.error(`Error rolling back to version ${migrationName}`, error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: `Failed to rollback to version: ${migrationName}`, + error: error.message, + }); + } + } + @Get('conflicts') + async getMigrationConflicts(): Promise { + this.logger.log('Fetching migration conflicts'); + return this.conflictResolutionService.getConflictHistory(); + } + @Post('sync-environments') + @HttpCode(HttpStatus.OK) + async syncEnvironments( + @Res() + res: Response): Promise { + this.logger.log('Syncing environments'); + try { + // This would typically trigger environment synchronization + // For now, we'll just return a success response + return res.status(HttpStatus.OK).json({ + success: true, + message: 'Environment synchronization initiated', + }); + } + catch (error) { + this.logger.error('Error syncing environments', error.stack); + return res.status(HttpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: 'Failed to sync environments', + error: error.message, + }); + } } - } } diff --git a/src/migrations/migration.module.ts b/src/migrations/migration.module.ts index f5eba3b6..8aeded7b 100644 --- a/src/migrations/migration.module.ts +++ b/src/migrations/migration.module.ts @@ -8,25 +8,25 @@ import { EnvironmentSyncService } from './environments/environment-sync.service' import { ConflictResolutionService } from './conflicts/conflict-resolution.service'; import { MigrationController } from './migration.controller'; import { MigrationRunnerService } from './migration-runner.service'; - @Module({ - imports: [TypeOrmModule.forFeature([Migration])], - controllers: [MigrationController], - providers: [ - MigrationService, - RollbackService, - SchemaValidationService, - EnvironmentSyncService, - ConflictResolutionService, - MigrationRunnerService, - ], - exports: [ - MigrationService, - RollbackService, - SchemaValidationService, - EnvironmentSyncService, - ConflictResolutionService, - MigrationRunnerService, - ], + imports: [TypeOrmModule.forFeature([Migration])], + controllers: [MigrationController], + providers: [ + MigrationService, + RollbackService, + SchemaValidationService, + EnvironmentSyncService, + ConflictResolutionService, + MigrationRunnerService, + ], + exports: [ + MigrationService, + RollbackService, + SchemaValidationService, + EnvironmentSyncService, + ConflictResolutionService, + MigrationRunnerService, + ], }) -export class MigrationModule {} +export class MigrationModule { +} diff --git a/src/migrations/migration.registry.ts b/src/migrations/migration.registry.ts index 992a9c25..d2d3aa8c 100644 --- a/src/migrations/migration.registry.ts +++ b/src/migrations/migration.registry.ts @@ -5,7 +5,6 @@ import { CreateCourseModulesTableMigration } from './samples/003-create-course-m import { CreateLessonsTableMigration } from './samples/004-create-lessons-table.migration'; import { CreateEnrollmentsTableMigration } from './samples/005-create-enrollments-table.migration'; import { CreateMigrationsTrackingTableMigration } from './samples/006-create-migrations-tracking-table.migration'; - /** * Central registry of all migrations in execution order. * @@ -16,10 +15,10 @@ import { CreateMigrationsTrackingTableMigration } from './samples/006-create-mig * - Never remove or reorder existing entries — only append new ones. */ export const MIGRATION_REGISTRY: MigrationConfig[] = [ - new CreateMigrationsTrackingTableMigration(), - new CreateUsersTableMigration(), - new CreateCoursesTableMigration(), - new CreateCourseModulesTableMigration(), - new CreateLessonsTableMigration(), - new CreateEnrollmentsTableMigration(), + new CreateMigrationsTrackingTableMigration(), + new CreateUsersTableMigration(), + new CreateCoursesTableMigration(), + new CreateCourseModulesTableMigration(), + new CreateLessonsTableMigration(), + new CreateEnrollmentsTableMigration(), ]; diff --git a/src/migrations/migration.service.ts b/src/migrations/migration.service.ts index d19ad9a6..b71b8e3e 100644 --- a/src/migrations/migration.service.ts +++ b/src/migrations/migration.service.ts @@ -7,199 +7,158 @@ import { SchemaValidationService } from './validation/schema-validation.service' import { RollbackService } from './rollback/rollback.service'; import { EnvironmentSyncService } from './environments/environment-sync.service'; import { MIGRATION_REGISTRY } from './migration.registry'; - export interface MigrationConfig { - name: string; - up: (connection: any) => Promise; - down: (connection: any) => Promise; - version: string; - dependencies?: string[]; + name: string; + up: (connection: unknown) => Promise; + down: (connection: unknown) => Promise; + version: string; + dependencies?: string[]; } - @Injectable() export class MigrationService { - private readonly logger = new Logger(MigrationService.name); - - constructor( + private readonly logger = new Logger(MigrationService.name); + constructor( @InjectRepository(Migration) - private migrationRepository: Repository, - private conflictResolutionService: ConflictResolutionService, - private schemaValidationService: SchemaValidationService, - private rollbackService: RollbackService, - private environmentSyncService: EnvironmentSyncService, - ) {} - - /** - * Executes pending migrations - */ - async runPendingMigrations(): Promise { - this.logger.log('Starting pending migrations...'); - - // Get all applied migrations - const appliedMigrations = await this.migrationRepository.find({ - order: { createdAt: 'ASC' }, - }); - - // Get all registered migrations - const registeredMigrations = this.getRegisteredMigrations(); - - // Filter for unapplied migrations - const pendingMigrations = registeredMigrations.filter((registered) => { - return !appliedMigrations.some((applied) => applied.name === registered.name); - }); - - // Validate dependencies - for (const migration of pendingMigrations) { - if (!this.validateDependencies(migration, appliedMigrations)) { - throw new Error(`Dependency not met for migration: ${migration.name}`); - } + private migrationRepository: Repository, private conflictResolutionService: ConflictResolutionService, private schemaValidationService: SchemaValidationService, private rollbackService: RollbackService, private environmentSyncService: EnvironmentSyncService) { } + /** + * Executes pending migrations + */ + async runPendingMigrations(): Promise { + this.logger.log('Starting pending migrations...'); + // Get all applied migrations + const appliedMigrations = await this.migrationRepository.find({ + order: { createdAt: 'ASC' }, + }); + // Get all registered migrations + const registeredMigrations = this.getRegisteredMigrations(); + // Filter for unapplied migrations + const pendingMigrations = registeredMigrations.filter((registered) => { + return !appliedMigrations.some((applied) => applied.name === registered.name); + }); + // Validate dependencies + for (const migration of pendingMigrations) { + if (!this.validateDependencies(migration, appliedMigrations)) { + throw new Error(`Dependency not met for migration: ${migration.name}`); + } + } + // Execute pending migrations + for (const migration of pendingMigrations) { + await this.executeMigration(migration); + } + this.logger.log('All pending migrations completed.'); } - - // Execute pending migrations - for (const migration of pendingMigrations) { - await this.executeMigration(migration); + /** + * Executes a single migration + */ + private async executeMigration(migration: MigrationConfig): Promise { + this.logger.log(`Executing migration: ${migration.name}`); + try { + // Check for conflicts + const hasConflict = await this.conflictResolutionService.checkForConflicts(migration); + if (hasConflict) { + await this.conflictResolutionService.resolveConflict(migration); + } + // Validate schema before applying migration + await this.schemaValidationService.validateBeforeMigration(migration); + // Execute the migration + const connection = await this.getConnection(); + await migration.up(connection); + // Record migration in the database + const migrationRecord = new Migration(); + migrationRecord.name = migration.name; + migrationRecord.version = migration.version; + migrationRecord.status = MigrationStatus.COMPLETED; + migrationRecord.appliedAt = new Date(); + await this.migrationRepository.save(migrationRecord); + // Sync environment after successful migration + await this.environmentSyncService.syncAfterMigration(migration); + this.logger.log(`Successfully executed migration: ${migration.name}`); + } + catch (error) { + this.logger.error(`Failed to execute migration: ${migration.name}`, error.stack); + // Attempt rollback on failure + await this.rollbackService.rollbackMigration(migration); + throw error; + } } - - this.logger.log('All pending migrations completed.'); - } - - /** - * Executes a single migration - */ - private async executeMigration(migration: MigrationConfig): Promise { - this.logger.log(`Executing migration: ${migration.name}`); - - try { - // Check for conflicts - const hasConflict = await this.conflictResolutionService.checkForConflicts(migration); - if (hasConflict) { - await this.conflictResolutionService.resolveConflict(migration); - } - - // Validate schema before applying migration - await this.schemaValidationService.validateBeforeMigration(migration); - - // Execute the migration - const connection = await this.getConnection(); - await migration.up(connection); - - // Record migration in the database - const migrationRecord = new Migration(); - migrationRecord.name = migration.name; - migrationRecord.version = migration.version; - migrationRecord.status = MigrationStatus.COMPLETED; - migrationRecord.appliedAt = new Date(); - - await this.migrationRepository.save(migrationRecord); - - // Sync environment after successful migration - await this.environmentSyncService.syncAfterMigration(migration); - - this.logger.log(`Successfully executed migration: ${migration.name}`); - } catch (error) { - this.logger.error(`Failed to execute migration: ${migration.name}`, error.stack); - - // Attempt rollback on failure - await this.rollbackService.rollbackMigration(migration); - - throw error; + /** + * Gets all registered migrations + */ + private getRegisteredMigrations(): MigrationConfig[] { + return MIGRATION_REGISTRY; + } + /** + * Validates migration dependencies + */ + private validateDependencies(migration: MigrationConfig, appliedMigrations: Migration[]): boolean { + if (!migration.dependencies || migration.dependencies.length === 0) { + return true; + } + return migration.dependencies.every((depName) => { + return appliedMigrations.some((m) => m.name === depName); + }); } - } - - /** - * Gets all registered migrations - */ - private getRegisteredMigrations(): MigrationConfig[] { - return MIGRATION_REGISTRY; - } - - /** - * Validates migration dependencies - */ - private validateDependencies( - migration: MigrationConfig, - appliedMigrations: Migration[], - ): boolean { - if (!migration.dependencies || migration.dependencies.length === 0) { - return true; + /** + * Gets database connection + */ + private async getConnection(): Promise { + // In a real implementation, this would return the actual database connection + // For now, returning a mock connection + return {}; + } + /** + * Lists all migrations with their status + */ + async listMigrations(): Promise> { + const appliedMigrations = await this.migrationRepository.find({ + order: { createdAt: 'ASC' }, + }); + const registeredMigrations = this.getRegisteredMigrations(); + // Combine applied and registered migrations + const allMigrations = [ + ...appliedMigrations.map((m) => ({ + name: m.name, + version: m.version, + status: m.status, + appliedAt: m.appliedAt, + })), + ]; + // Add unapplied migrations + registeredMigrations.forEach((registered) => { + const exists = appliedMigrations.some((m) => m.name === registered.name); + if (!exists) { + allMigrations.push({ + name: registered.name, + version: registered.version, + status: MigrationStatus.PENDING, + appliedAt: undefined, + }); + } + }); + return allMigrations; } - - return migration.dependencies.every((depName) => { - return appliedMigrations.some((m) => m.name === depName); - }); - } - - /** - * Gets database connection - */ - private async getConnection(): Promise { - // In a real implementation, this would return the actual database connection - // For now, returning a mock connection - return {}; - } - - /** - * Lists all migrations with their status - */ - async listMigrations(): Promise< - Array<{ name: string; version: string; status: string; appliedAt?: Date }> - > { - const appliedMigrations = await this.migrationRepository.find({ - order: { createdAt: 'ASC' }, - }); - - const registeredMigrations = this.getRegisteredMigrations(); - - // Combine applied and registered migrations - const allMigrations = [ - ...appliedMigrations.map((m) => ({ - name: m.name, - version: m.version, - status: m.status, - appliedAt: m.appliedAt, - })), - ]; - - // Add unapplied migrations - registeredMigrations.forEach((registered) => { - const exists = appliedMigrations.some((m) => m.name === registered.name); - if (!exists) { - allMigrations.push({ - name: registered.name, - version: registered.version, - status: MigrationStatus.PENDING, - appliedAt: undefined, + /** + * Resets all migrations (for development purposes) + */ + async resetMigrations(): Promise { + this.logger.log('Resetting all migrations...'); + const appliedMigrations = await this.migrationRepository.find({ + order: { createdAt: 'DESC' }, }); - } - }); - - return allMigrations; - } - - /** - * Resets all migrations (for development purposes) - */ - async resetMigrations(): Promise { - this.logger.log('Resetting all migrations...'); - - const appliedMigrations = await this.migrationRepository.find({ - order: { createdAt: 'DESC' }, - }); - - for (const migration of appliedMigrations) { - // Find the actual migration config to get the 'down' function - const registeredMigration = this.getRegisteredMigrations().find( - (m) => m.name === migration.name, - ); - - if (registeredMigration) { - await this.rollbackService.rollbackMigration(registeredMigration); - } - - // Remove from migration history - await this.migrationRepository.remove(migration); + for (const migration of appliedMigrations) { + // Find the actual migration config to get the 'down' function + const registeredMigration = this.getRegisteredMigrations().find((m) => m.name === migration.name); + if (registeredMigration) { + await this.rollbackService.rollbackMigration(registeredMigration); + } + // Remove from migration history + await this.migrationRepository.remove(migration); + } + this.logger.log('All migrations have been reset.'); } - - this.logger.log('All migrations have been reset.'); - } } diff --git a/src/migrations/rollback/rollback.service.ts b/src/migrations/rollback/rollback.service.ts index e2c983d7..66a8004d 100644 --- a/src/migrations/rollback/rollback.service.ts +++ b/src/migrations/rollback/rollback.service.ts @@ -4,201 +4,166 @@ import { Repository, Raw } from 'typeorm'; import { Migration, MigrationStatus } from '../entities/migration.entity'; import { MigrationConfig } from '../migration.service'; import { MIGRATION_REGISTRY } from '../migration.registry'; - @Injectable() export class RollbackService { - private readonly logger = new Logger(RollbackService.name); - - constructor( + private readonly logger = new Logger(RollbackService.name); + constructor( @InjectRepository(Migration) - private migrationRepository: Repository, - ) {} - - /** - * Rolls back a specific migration - */ - async rollbackMigration(migration: MigrationConfig): Promise { - this.logger.log(`Rolling back migration: ${migration.name}`); - - try { - // Get database connection - const connection = await this.getConnection(); - - // Execute the down migration - await migration.down(connection); - - // Update migration record - const existingMigration = await this.migrationRepository.findOne({ - where: { name: migration.name }, - }); - - if (existingMigration) { - existingMigration.status = MigrationStatus.ROLLED_BACK; - existingMigration.rolledBackAt = new Date(); - await this.migrationRepository.save(existingMigration); - } - - this.logger.log(`Successfully rolled back migration: ${migration.name}`); - } catch (error) { - this.logger.error(`Failed to roll back migration: ${migration.name}`, error.stack); - throw error; + private migrationRepository: Repository) { } + /** + * Rolls back a specific migration + */ + async rollbackMigration(migration: MigrationConfig): Promise { + this.logger.log(`Rolling back migration: ${migration.name}`); + try { + // Get database connection + const connection = await this.getConnection(); + // Execute the down migration + await migration.down(connection); + // Update migration record + const existingMigration = await this.migrationRepository.findOne({ + where: { name: migration.name }, + }); + if (existingMigration) { + existingMigration.status = MigrationStatus.ROLLED_BACK; + existingMigration.rolledBackAt = new Date(); + await this.migrationRepository.save(existingMigration); + } + this.logger.log(`Successfully rolled back migration: ${migration.name}`); + } + catch (error) { + this.logger.error(`Failed to roll back migration: ${migration.name}`, error.stack); + throw error; + } } - } - - /** - * Rolls back the last N migrations - */ - async rollbackLastMigrations(count: number = 1): Promise { - this.logger.log(`Rolling back last ${count} migrations`); - - // Get the last N applied migrations - const lastAppliedMigrations = await this.migrationRepository.find({ - where: { status: MigrationStatus.COMPLETED }, - order: { appliedAt: 'DESC' }, - take: count, - }); - - // Get all registered migrations to find the down functions - const registeredMigrations = this.getRegisteredMigrations(); - - for (const appliedMigration of lastAppliedMigrations) { - const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); - - if (migrationConfig) { - await this.rollbackMigration(migrationConfig); - } else { - this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); - } + /** + * Rolls back the last N migrations + */ + async rollbackLastMigrations(count: number = 1): Promise { + this.logger.log(`Rolling back last ${count} migrations`); + // Get the last N applied migrations + const lastAppliedMigrations = await this.migrationRepository.find({ + where: { status: MigrationStatus.COMPLETED }, + order: { appliedAt: 'DESC' }, + take: count, + }); + // Get all registered migrations to find the down functions + const registeredMigrations = this.getRegisteredMigrations(); + for (const appliedMigration of lastAppliedMigrations) { + const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); + if (migrationConfig) { + await this.rollbackMigration(migrationConfig); + } + else { + this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); + } + } } - } - - /** - * Rolls back all migrations - */ - async rollbackAllMigrations(): Promise { - this.logger.log('Rolling back all migrations'); - - // Get all applied migrations in reverse order - const allAppliedMigrations = await this.migrationRepository.find({ - where: { status: MigrationStatus.COMPLETED }, - order: { appliedAt: 'DESC' }, - }); - - // Get all registered migrations to find the down functions - const registeredMigrations = this.getRegisteredMigrations(); - - for (const appliedMigration of allAppliedMigrations) { - const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); - - if (migrationConfig) { - await this.rollbackMigration(migrationConfig); - } else { - this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); - } + /** + * Rolls back all migrations + */ + async rollbackAllMigrations(): Promise { + this.logger.log('Rolling back all migrations'); + // Get all applied migrations in reverse order + const allAppliedMigrations = await this.migrationRepository.find({ + where: { status: MigrationStatus.COMPLETED }, + order: { appliedAt: 'DESC' }, + }); + // Get all registered migrations to find the down functions + const registeredMigrations = this.getRegisteredMigrations(); + for (const appliedMigration of allAppliedMigrations) { + const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); + if (migrationConfig) { + await this.rollbackMigration(migrationConfig); + } + else { + this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); + } + } } - } - - /** - * Gets database connection - */ - private async getConnection(): Promise { - // In a real implementation, this would return the actual database connection - // For now, returning a mock connection - return {}; - } - - /** - * Gets all registered migrations - */ - private getRegisteredMigrations(): MigrationConfig[] { - return MIGRATION_REGISTRY; - } - - /** - * Rolls back all migrations down to (but not including) the target version. - * Migrations are rolled back in reverse-applied order. - */ - async rollbackToVersion(targetMigrationName: string): Promise { - this.logger.log(`Rolling back to version: ${targetMigrationName}`); - - const registeredMigrations = this.getRegisteredMigrations(); - - // Verify the target migration exists in the registry - const targetExists = registeredMigrations.some((m) => m.name === targetMigrationName); - if (!targetExists) { - throw new Error(`Target migration not found in registry: ${targetMigrationName}`); + /** + * Gets database connection + */ + private async getConnection(): Promise { + // In a real implementation, this would return the actual database connection + // For now, returning a mock connection + return {}; } - - // Get all applied migrations that were applied AFTER the target, in reverse order - const targetRecord = await this.migrationRepository.findOne({ - where: { name: targetMigrationName }, - }); - - const whereClause = targetRecord?.appliedAt - ? { - status: MigrationStatus.COMPLETED, - appliedAt: Raw((alias) => `${alias} > :appliedAt`, { appliedAt: targetRecord.appliedAt }), - } - : { status: MigrationStatus.COMPLETED }; - - const migrationsToRollback = await this.migrationRepository.find({ - where: whereClause, - order: { appliedAt: 'DESC' }, - }); - - for (const appliedMigration of migrationsToRollback) { - const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); - if (migrationConfig) { - await this.rollbackMigration(migrationConfig); - } else { - this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); - } + /** + * Gets all registered migrations + */ + private getRegisteredMigrations(): MigrationConfig[] { + return MIGRATION_REGISTRY; } - - this.logger.log(`Rollback to version ${targetMigrationName} complete.`); - } - - /** - * Rolls back a specific named migration regardless of order. - */ - async rollbackByName(migrationName: string): Promise { - this.logger.log(`Rolling back specific migration by name: ${migrationName}`); - - const registeredMigrations = this.getRegisteredMigrations(); - const migrationConfig = registeredMigrations.find((m) => m.name === migrationName); - - if (!migrationConfig) { - throw new Error(`Migration not found in registry: ${migrationName}`); + /** + * Rolls back all migrations down to (but not including) the target version. + * Migrations are rolled back in reverse-applied order. + */ + async rollbackToVersion(targetMigrationName: string): Promise { + this.logger.log(`Rolling back to version: ${targetMigrationName}`); + const registeredMigrations = this.getRegisteredMigrations(); + // Verify the target migration exists in the registry + const targetExists = registeredMigrations.some((m) => m.name === targetMigrationName); + if (!targetExists) { + throw new Error(`Target migration not found in registry: ${targetMigrationName}`); + } + // Get all applied migrations that were applied AFTER the target, in reverse order + const targetRecord = await this.migrationRepository.findOne({ + where: { name: targetMigrationName }, + }); + const whereClause = targetRecord?.appliedAt + ? { + status: MigrationStatus.COMPLETED, + appliedAt: Raw((alias) => `${alias} > :appliedAt`, { appliedAt: targetRecord.appliedAt }), + } + : { status: MigrationStatus.COMPLETED }; + const migrationsToRollback = await this.migrationRepository.find({ + where: whereClause, + order: { appliedAt: 'DESC' }, + }); + for (const appliedMigration of migrationsToRollback) { + const migrationConfig = registeredMigrations.find((m) => m.name === appliedMigration.name); + if (migrationConfig) { + await this.rollbackMigration(migrationConfig); + } + else { + this.logger.warn(`Could not find migration config for: ${appliedMigration.name}`); + } + } + this.logger.log(`Rollback to version ${targetMigrationName} complete.`); } - - const canRollback = await this.canRollbackMigration(migrationName); - if (!canRollback) { - throw new Error( - `Cannot roll back migration ${migrationName}: it either hasn't been applied or has dependent migrations applied after it.`, - ); + /** + * Rolls back a specific named migration regardless of order. + */ + async rollbackByName(migrationName: string): Promise { + this.logger.log(`Rolling back specific migration by name: ${migrationName}`); + const registeredMigrations = this.getRegisteredMigrations(); + const migrationConfig = registeredMigrations.find((m) => m.name === migrationName); + if (!migrationConfig) { + throw new Error(`Migration not found in registry: ${migrationName}`); + } + const canRollback = await this.canRollbackMigration(migrationName); + if (!canRollback) { + throw new Error(`Cannot roll back migration ${migrationName}: it either hasn't been applied or has dependent migrations applied after it.`); + } + await this.rollbackMigration(migrationConfig); } - - await this.rollbackMigration(migrationConfig); - } - async canRollbackMigration(migrationName: string): Promise { - // Check if the migration exists and is completed - const migration = await this.migrationRepository.findOne({ - where: { name: migrationName, status: MigrationStatus.COMPLETED }, - }); - - if (!migration) { - return false; + async canRollbackMigration(migrationName: string): Promise { + // Check if the migration exists and is completed + const migration = await this.migrationRepository.findOne({ + where: { name: migrationName, status: MigrationStatus.COMPLETED }, + }); + if (!migration) { + return false; + } + // Check if there are dependent migrations that have been applied after this one + const laterMigrations = await this.migrationRepository.find({ + where: { + appliedAt: Raw((alias) => `${alias} > :appliedAt`, { appliedAt: migration.appliedAt }), + }, + }); + // In a real implementation, you'd check dependencies here + // For now, we'll return true if no later migrations exist + return laterMigrations.length === 0; } - - // Check if there are dependent migrations that have been applied after this one - const laterMigrations = await this.migrationRepository.find({ - where: { - appliedAt: Raw((alias) => `${alias} > :appliedAt`, { appliedAt: migration.appliedAt }), - }, - }); - - // In a real implementation, you'd check dependencies here - // For now, we'll return true if no later migrations exist - return laterMigrations.length === 0; - } } diff --git a/src/migrations/samples/001-create-users-table.migration.ts b/src/migrations/samples/001-create-users-table.migration.ts index 5fc88f3f..8f6e69fe 100644 --- a/src/migrations/samples/001-create-users-table.migration.ts +++ b/src/migrations/samples/001-create-users-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 001 — Create users table * @@ -9,30 +8,25 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateUsersTableMigration implements MigrationConfig { - name = '001-create-users-table'; - version = '1.0.0'; - dependencies: string[] = []; - - private readonly logger = new Logger(CreateUsersTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create users table'); - - await connection.query(` + name = '001-create-users-table'; + version = '1.0.0'; + dependencies: string[] = []; + private readonly logger = new Logger(CreateUsersTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create users table'); + await connection.query(` DO $$ BEGIN CREATE TYPE user_role AS ENUM ('student', 'teacher', 'admin'); EXCEPTION WHEN duplicate_object THEN NULL; END $$; `); - - await connection.query(` + await connection.query(` DO $$ BEGIN CREATE TYPE user_status AS ENUM ('active', 'inactive', 'suspended'); EXCEPTION WHEN duplicate_object THEN NULL; END $$; `); - - await connection.query(` + await connection.query(` CREATE TABLE IF NOT EXISTS users ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), email VARCHAR(255) NOT NULL UNIQUE, @@ -56,21 +50,16 @@ export class CreateUsersTableMigration implements MigrationConfig { updated_at TIMESTAMP NOT NULL DEFAULT NOW() ); `); - - await connection.query('CREATE INDEX IF NOT EXISTS idx_users_email ON users (email);'); - await connection.query('CREATE INDEX IF NOT EXISTS idx_users_username ON users (username);'); - await connection.query('CREATE INDEX IF NOT EXISTS idx_users_tenant_id ON users (tenant_id);'); - - this.logger.log('Migration applied: create users table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create users table'); - - await connection.query('DROP TABLE IF EXISTS users CASCADE;'); - await connection.query('DROP TYPE IF EXISTS user_role;'); - await connection.query('DROP TYPE IF EXISTS user_status;'); - - this.logger.log('Migration rolled back: create users table'); - } + await connection.query('CREATE INDEX IF NOT EXISTS idx_users_email ON users (email);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_users_username ON users (username);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_users_tenant_id ON users (tenant_id);'); + this.logger.log('Migration applied: create users table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create users table'); + await connection.query('DROP TABLE IF EXISTS users CASCADE;'); + await connection.query('DROP TYPE IF EXISTS user_role;'); + await connection.query('DROP TYPE IF EXISTS user_status;'); + this.logger.log('Migration rolled back: create users table'); + } } diff --git a/src/migrations/samples/002-create-courses-table.migration.ts b/src/migrations/samples/002-create-courses-table.migration.ts index c440f773..a70cfa63 100644 --- a/src/migrations/samples/002-create-courses-table.migration.ts +++ b/src/migrations/samples/002-create-courses-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 002 — Create courses table * @@ -11,16 +10,13 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateCoursesTableMigration implements MigrationConfig { - name = '002-create-courses-table'; - version = '1.0.0'; - dependencies = ['001-create-users-table']; - - private readonly logger = new Logger(CreateCoursesTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create courses table'); - - await connection.query(` + name = '002-create-courses-table'; + version = '1.0.0'; + dependencies = ['001-create-users-table']; + private readonly logger = new Logger(CreateCoursesTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create courses table'); + await connection.query(` CREATE TABLE IF NOT EXISTS course ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), title VARCHAR(255) NOT NULL, @@ -33,22 +29,13 @@ export class CreateCoursesTableMigration implements MigrationConfig { updated_at TIMESTAMP NOT NULL DEFAULT NOW() ); `); - - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_course_status ON course (status);', - ); - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_course_instructor_id ON course (instructor_id);', - ); - - this.logger.log('Migration applied: create courses table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create courses table'); - - await connection.query('DROP TABLE IF EXISTS course CASCADE;'); - - this.logger.log('Migration rolled back: create courses table'); - } + await connection.query('CREATE INDEX IF NOT EXISTS idx_course_status ON course (status);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_course_instructor_id ON course (instructor_id);'); + this.logger.log('Migration applied: create courses table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create courses table'); + await connection.query('DROP TABLE IF EXISTS course CASCADE;'); + this.logger.log('Migration rolled back: create courses table'); + } } diff --git a/src/migrations/samples/003-create-course-modules-table.migration.ts b/src/migrations/samples/003-create-course-modules-table.migration.ts index 3cd526b9..58e9f8cf 100644 --- a/src/migrations/samples/003-create-course-modules-table.migration.ts +++ b/src/migrations/samples/003-create-course-modules-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 003 — Create course_module table * @@ -11,16 +10,13 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateCourseModulesTableMigration implements MigrationConfig { - name = '003-create-course-modules-table'; - version = '1.0.0'; - dependencies = ['002-create-courses-table']; - - private readonly logger = new Logger(CreateCourseModulesTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create course_module table'); - - await connection.query(` + name = '003-create-course-modules-table'; + version = '1.0.0'; + dependencies = ['002-create-courses-table']; + private readonly logger = new Logger(CreateCourseModulesTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create course_module table'); + await connection.query(` CREATE TABLE IF NOT EXISTS course_module ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), title VARCHAR(255) NOT NULL, @@ -28,19 +24,12 @@ export class CreateCourseModulesTableMigration implements MigrationConfig { course_id UUID NOT NULL REFERENCES course(id) ON DELETE CASCADE ); `); - - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_course_module_course_id ON course_module (course_id);', - ); - - this.logger.log('Migration applied: create course_module table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create course_module table'); - - await connection.query('DROP TABLE IF EXISTS course_module CASCADE;'); - - this.logger.log('Migration rolled back: create course_module table'); - } + await connection.query('CREATE INDEX IF NOT EXISTS idx_course_module_course_id ON course_module (course_id);'); + this.logger.log('Migration applied: create course_module table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create course_module table'); + await connection.query('DROP TABLE IF EXISTS course_module CASCADE;'); + this.logger.log('Migration rolled back: create course_module table'); + } } diff --git a/src/migrations/samples/004-create-lessons-table.migration.ts b/src/migrations/samples/004-create-lessons-table.migration.ts index ac0409ac..bff2cbe9 100644 --- a/src/migrations/samples/004-create-lessons-table.migration.ts +++ b/src/migrations/samples/004-create-lessons-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 004 — Create lesson table * @@ -11,16 +10,13 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateLessonsTableMigration implements MigrationConfig { - name = '004-create-lessons-table'; - version = '1.0.0'; - dependencies = ['003-create-course-modules-table']; - - private readonly logger = new Logger(CreateLessonsTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create lesson table'); - - await connection.query(` + name = '004-create-lessons-table'; + version = '1.0.0'; + dependencies = ['003-create-course-modules-table']; + private readonly logger = new Logger(CreateLessonsTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create lesson table'); + await connection.query(` CREATE TABLE IF NOT EXISTS lesson ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), title VARCHAR(255) NOT NULL, @@ -31,19 +27,12 @@ export class CreateLessonsTableMigration implements MigrationConfig { module_id UUID NOT NULL REFERENCES course_module(id) ON DELETE CASCADE ); `); - - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_lesson_module_id ON lesson (module_id);', - ); - - this.logger.log('Migration applied: create lesson table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create lesson table'); - - await connection.query('DROP TABLE IF EXISTS lesson CASCADE;'); - - this.logger.log('Migration rolled back: create lesson table'); - } + await connection.query('CREATE INDEX IF NOT EXISTS idx_lesson_module_id ON lesson (module_id);'); + this.logger.log('Migration applied: create lesson table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create lesson table'); + await connection.query('DROP TABLE IF EXISTS lesson CASCADE;'); + this.logger.log('Migration rolled back: create lesson table'); + } } diff --git a/src/migrations/samples/005-create-enrollments-table.migration.ts b/src/migrations/samples/005-create-enrollments-table.migration.ts index 38cf1b26..8fb6ac9b 100644 --- a/src/migrations/samples/005-create-enrollments-table.migration.ts +++ b/src/migrations/samples/005-create-enrollments-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 005 — Create enrollment table * @@ -11,16 +10,13 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateEnrollmentsTableMigration implements MigrationConfig { - name = '005-create-enrollments-table'; - version = '1.0.0'; - dependencies = ['001-create-users-table', '002-create-courses-table']; - - private readonly logger = new Logger(CreateEnrollmentsTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create enrollment table'); - - await connection.query(` + name = '005-create-enrollments-table'; + version = '1.0.0'; + dependencies = ['001-create-users-table', '002-create-courses-table']; + private readonly logger = new Logger(CreateEnrollmentsTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create enrollment table'); + await connection.query(` CREATE TABLE IF NOT EXISTS enrollment ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE, @@ -31,31 +27,16 @@ export class CreateEnrollmentsTableMigration implements MigrationConfig { last_accessed_at TIMESTAMP NOT NULL DEFAULT NOW() ); `); - - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_enrollment_user_id ON enrollment (user_id);', - ); - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_enrollment_course_id ON enrollment (course_id);', - ); - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_enrollment_status ON enrollment (status);', - ); - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_enrollment_user_status ON enrollment (user_id, status);', - ); - await connection.query( - 'CREATE INDEX IF NOT EXISTS idx_enrollment_course_status ON enrollment (course_id, status);', - ); - - this.logger.log('Migration applied: create enrollment table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create enrollment table'); - - await connection.query('DROP TABLE IF EXISTS enrollment CASCADE;'); - - this.logger.log('Migration rolled back: create enrollment table'); - } + await connection.query('CREATE INDEX IF NOT EXISTS idx_enrollment_user_id ON enrollment (user_id);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_enrollment_course_id ON enrollment (course_id);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_enrollment_status ON enrollment (status);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_enrollment_user_status ON enrollment (user_id, status);'); + await connection.query('CREATE INDEX IF NOT EXISTS idx_enrollment_course_status ON enrollment (course_id, status);'); + this.logger.log('Migration applied: create enrollment table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create enrollment table'); + await connection.query('DROP TABLE IF EXISTS enrollment CASCADE;'); + this.logger.log('Migration rolled back: create enrollment table'); + } } diff --git a/src/migrations/samples/006-create-migrations-tracking-table.migration.ts b/src/migrations/samples/006-create-migrations-tracking-table.migration.ts index 1d90cdfd..f6c71d77 100644 --- a/src/migrations/samples/006-create-migrations-tracking-table.migration.ts +++ b/src/migrations/samples/006-create-migrations-tracking-table.migration.ts @@ -1,6 +1,5 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - /** * Migration 006 — Create migrations tracking table * @@ -14,23 +13,19 @@ import { MigrationConfig } from '../migration.service'; */ @Injectable() export class CreateMigrationsTrackingTableMigration implements MigrationConfig { - name = '006-create-migrations-tracking-table'; - version = '1.0.0'; - dependencies: string[] = []; - - private readonly logger = new Logger(CreateMigrationsTrackingTableMigration.name); - - async up(connection: any): Promise { - this.logger.log('Applying migration: create migrations tracking table'); - - await connection.query(` + name = '006-create-migrations-tracking-table'; + version = '1.0.0'; + dependencies: string[] = []; + private readonly logger = new Logger(CreateMigrationsTrackingTableMigration.name); + async up(connection: unknown): Promise { + this.logger.log('Applying migration: create migrations tracking table'); + await connection.query(` DO $$ BEGIN CREATE TYPE migration_status AS ENUM ('pending', 'completed', 'failed', 'rolled_back'); EXCEPTION WHEN duplicate_object THEN NULL; END $$; `); - - await connection.query(` + await connection.query(` CREATE TABLE IF NOT EXISTS migrations ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), name VARCHAR(255) NOT NULL UNIQUE, @@ -43,16 +38,12 @@ export class CreateMigrationsTrackingTableMigration implements MigrationConfig { updated_at TIMESTAMP NOT NULL DEFAULT NOW() ); `); - - this.logger.log('Migration applied: create migrations tracking table'); - } - - async down(connection: any): Promise { - this.logger.log('Rolling back migration: create migrations tracking table'); - - await connection.query('DROP TABLE IF EXISTS migrations CASCADE;'); - await connection.query('DROP TYPE IF EXISTS migration_status;'); - - this.logger.log('Migration rolled back: create migrations tracking table'); - } + this.logger.log('Migration applied: create migrations tracking table'); + } + async down(connection: unknown): Promise { + this.logger.log('Rolling back migration: create migrations tracking table'); + await connection.query('DROP TABLE IF EXISTS migrations CASCADE;'); + await connection.query('DROP TYPE IF EXISTS migration_status;'); + this.logger.log('Migration rolled back: create migrations tracking table'); + } } diff --git a/src/migrations/samples/sample-user-table.migration.ts b/src/migrations/samples/sample-user-table.migration.ts index 48bb0487..54cd7b7e 100644 --- a/src/migrations/samples/sample-user-table.migration.ts +++ b/src/migrations/samples/sample-user-table.migration.ts @@ -1,45 +1,37 @@ import { Injectable, Logger } from '@nestjs/common'; import { MigrationConfig } from '../migration.service'; - @Injectable() export class SampleUserTableMigration implements MigrationConfig { - name = 'sample-user-table'; - version = '1.0.0'; - dependencies = []; // List any dependencies this migration has - - private readonly logger = new Logger(SampleUserTableMigration.name); - - async up(_connection: any): Promise { - this.logger.log('Applying sample user table migration'); - - // In a real implementation, you would use the connection to execute SQL - // For example with TypeORM: - /* - await connection.query(` - CREATE TABLE users ( - id SERIAL PRIMARY KEY, - email VARCHAR(255) UNIQUE NOT NULL, - first_name VARCHAR(100), - last_name VARCHAR(100), - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP - ); - `); - */ - - // Mock implementation for demonstration - console.log('Creating users table...'); - } - - async down(_connection: any): Promise { - this.logger.log('Rolling back sample user table migration'); - - // In a real implementation, you would revert the changes - /* - await connection.query(`DROP TABLE IF EXISTS users;`); - */ - - // Mock implementation for demonstration - console.log('Dropping users table...'); - } + name = 'sample-user-table'; + version = '1.0.0'; + dependencies = []; // List any dependencies this migration has + private readonly logger = new Logger(SampleUserTableMigration.name); + async up(_connection: unknown): Promise { + this.logger.log('Applying sample user table migration'); + // In a real implementation, you would use the connection to execute SQL + // For example with TypeORM: + /* + await connection.query(` + CREATE TABLE users ( + id SERIAL PRIMARY KEY, + email VARCHAR(255) UNIQUE NOT NULL, + first_name VARCHAR(100), + last_name VARCHAR(100), + created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, + updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP + ); + `); + */ + // Mock implementation for demonstration + console.log('Creating users table...'); + } + async down(_connection: unknown): Promise { + this.logger.log('Rolling back sample user table migration'); + // In a real implementation, you would revert the changes + /* + await connection.query(`DROP TABLE IF EXISTS users;`); + */ + // Mock implementation for demonstration + console.log('Dropping users table...'); + } } diff --git a/src/migrations/validation/schema-validation.service.ts b/src/migrations/validation/schema-validation.service.ts index c75f9130..42cbab88 100644 --- a/src/migrations/validation/schema-validation.service.ts +++ b/src/migrations/validation/schema-validation.service.ts @@ -1,157 +1,121 @@ import { Injectable, Logger, BadRequestException } from '@nestjs/common'; // import { Connection, QueryRunner } from 'typeorm'; import { MigrationConfig } from '../migration.service'; - @Injectable() export class SchemaValidationService { - private readonly logger = new Logger(SchemaValidationService.name); - - /** - * Validates the schema before applying a migration - */ - async validateBeforeMigration(migration: MigrationConfig): Promise { - this.logger.log(`Validating schema before migration: ${migration.name}`); - - try { - // Perform pre-migration validation checks - const isValid = await this.performPreMigrationValidation(migration); - - if (!isValid) { - throw new BadRequestException(`Schema validation failed for migration: ${migration.name}`); - } - - this.logger.log(`Schema validation passed for migration: ${migration.name}`); - return true; - } catch (error) { - this.logger.error(`Schema validation failed for migration: ${migration.name}`, error.stack); - throw error; + private readonly logger = new Logger(SchemaValidationService.name); + /** + * Validates the schema before applying a migration + */ + async validateBeforeMigration(migration: MigrationConfig): Promise { + this.logger.log(`Validating schema before migration: ${migration.name}`); + try { + // Perform pre-migration validation checks + const isValid = await this.performPreMigrationValidation(migration); + if (!isValid) { + throw new BadRequestException(`Schema validation failed for migration: ${migration.name}`); + } + this.logger.log(`Schema validation passed for migration: ${migration.name}`); + return true; + } + catch (error) { + this.logger.error(`Schema validation failed for migration: ${migration.name}`, error.stack); + throw error; + } } - } - - /** - * Validates the schema after applying a migration - */ - async validateAfterMigration(migration: MigrationConfig): Promise { - this.logger.log(`Validating schema after migration: ${migration.name}`); - - try { - // Perform post-migration validation checks - const isValid = await this.performPostMigrationValidation(migration); - - if (!isValid) { - throw new BadRequestException( - `Post-migration schema validation failed for: ${migration.name}`, - ); - } - - this.logger.log(`Post-migration schema validation passed for: ${migration.name}`); - return true; - } catch (error) { - this.logger.error( - `Post-migration schema validation failed for: ${migration.name}`, - error.stack, - ); - throw error; + /** + * Validates the schema after applying a migration + */ + async validateAfterMigration(migration: MigrationConfig): Promise { + this.logger.log(`Validating schema after migration: ${migration.name}`); + try { + // Perform post-migration validation checks + const isValid = await this.performPostMigrationValidation(migration); + if (!isValid) { + throw new BadRequestException(`Post-migration schema validation failed for: ${migration.name}`); + } + this.logger.log(`Post-migration schema validation passed for: ${migration.name}`); + return true; + } + catch (error) { + this.logger.error(`Post-migration schema validation failed for: ${migration.name}`, error.stack); + throw error; + } } - } - - /** - * Performs pre-migration validation checks - */ - // Fix: prefix unused param with _ — stub method, real implementation pending - private async performPreMigrationValidation(_migration: MigrationConfig): Promise { - // Check if required tables/columns exist before running migration - // This is a simplified version - in practice, you'd check for dependencies - - // For now, just return true - return true; - } - - /** - * Performs post-migration validation checks - */ - // Fix: prefix unused param with _ — stub method, real implementation pending - private async performPostMigrationValidation(_migration: MigrationConfig): Promise { - // Check if the expected schema changes were applied correctly - // This would involve checking if tables/columns exist as expected after migration - - // For now, just return true - return true; - } - - /** - * Validates the current database schema against expected state - */ - async validateCurrentSchema(): Promise { - this.logger.log('Validating current database schema...'); - - try { - // This would involve comparing the current schema with expected schema definitions - // For now, just return true - return true; - } catch (error) { - this.logger.error('Current schema validation failed', error.stack); - return false; + /** + * Performs pre-migration validation checks + */ + // Fix: prefix unused param with _ — stub method, real implementation pending + private async performPreMigrationValidation(_migration: MigrationConfig): Promise { + // Check if required tables/columns exist before running migration + // This is a simplified version - in practice, you'd check for dependencies + // For now, just return true + return true; } - } - - /** - * Checks for potential breaking changes in a migration - */ - async checkForBreakingChanges(migration: MigrationConfig): Promise { - this.logger.log(`Checking for breaking changes in migration: ${migration.name}`); - - const breakingChanges: string[] = []; - - // Analyze the migration to detect potential breaking changes - // This is a simplified implementation - in practice, you'd parse the SQL operations - // and check for things like dropping columns/tables, changing data types, etc. - - // For now, return an empty array - return breakingChanges; - } - - /** - * Creates a backup of the current schema before migration - */ - async backupSchemaBeforeMigration(migration: MigrationConfig): Promise { - this.logger.log(`Creating schema backup before migration: ${migration.name}`); - - try { - // In a real implementation, this would create a backup of the schema - // For now, return a placeholder - return `backup_${migration.name}_${new Date().toISOString()}`; - } catch (error) { - this.logger.error( - `Failed to create schema backup for migration: ${migration.name}`, - error.stack, - ); - return null; + /** + * Performs post-migration validation checks + */ + // Fix: prefix unused param with _ — stub method, real implementation pending + private async performPostMigrationValidation(_migration: MigrationConfig): Promise { + // Check if the expected schema changes were applied correctly + // This would involve checking if tables/columns exist as expected after migration + // For now, just return true + return true; } - } - - /** - * Validates migration dependencies - */ - async validateMigrationDependencies( - migration: MigrationConfig, - appliedMigrations: string[], - ): Promise { - if (!migration.dependencies || migration.dependencies.length === 0) { - return true; + /** + * Validates the current database schema against expected state + */ + async validateCurrentSchema(): Promise { + this.logger.log('Validating current database schema...'); + try { + // This would involve comparing the current schema with expected schema definitions + // For now, just return true + return true; + } + catch (error) { + this.logger.error('Current schema validation failed', error.stack); + return false; + } } - - const missingDependencies = migration.dependencies.filter( - (dep) => !appliedMigrations.includes(dep), - ); - - if (missingDependencies.length > 0) { - this.logger.error( - `Missing dependencies for migration ${migration.name}: ${missingDependencies.join(', ')}`, - ); - return false; + /** + * Checks for potential breaking changes in a migration + */ + async checkForBreakingChanges(migration: MigrationConfig): Promise { + this.logger.log(`Checking for breaking changes in migration: ${migration.name}`); + const breakingChanges: string[] = []; + // Analyze the migration to detect potential breaking changes + // This is a simplified implementation - in practice, you'd parse the SQL operations + // and check for things like dropping columns/tables, changing data types, etc. + // For now, return an empty array + return breakingChanges; + } + /** + * Creates a backup of the current schema before migration + */ + async backupSchemaBeforeMigration(migration: MigrationConfig): Promise { + this.logger.log(`Creating schema backup before migration: ${migration.name}`); + try { + // In a real implementation, this would create a backup of the schema + // For now, return a placeholder + return `backup_${migration.name}_${new Date().toISOString()}`; + } + catch (error) { + this.logger.error(`Failed to create schema backup for migration: ${migration.name}`, error.stack); + return null; + } + } + /** + * Validates migration dependencies + */ + async validateMigrationDependencies(migration: MigrationConfig, appliedMigrations: string[]): Promise { + if (!migration.dependencies || migration.dependencies.length === 0) { + return true; + } + const missingDependencies = migration.dependencies.filter((dep) => !appliedMigrations.includes(dep)); + if (missingDependencies.length > 0) { + this.logger.error(`Missing dependencies for migration ${migration.name}: ${missingDependencies.join(', ')}`); + return false; + } + return true; } - - return true; - } } diff --git a/src/moderation/analytics/moderation-analytics.service.ts b/src/moderation/analytics/moderation-analytics.service.ts index 1bb0cfd5..78cbc8e1 100644 --- a/src/moderation/analytics/moderation-analytics.service.ts +++ b/src/moderation/analytics/moderation-analytics.service.ts @@ -2,20 +2,16 @@ import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { ModerationEvent } from './moderation-event.entity'; - @Injectable() export class ModerationAnalyticsService { - constructor( + constructor( @InjectRepository(ModerationEvent) - private readonly eventRepo: Repository, - ) {} - - async logModerationEvent(content: string, score: number, status: string) { - const event = this.eventRepo.create({ content, score, status }); - await this.eventRepo.save(event); - } - - async getAnalytics() { - return this.eventRepo.find({ order: { timestamp: 'DESC' } }); - } + private readonly eventRepo: Repository) { } + async logModerationEvent(content: string, score: number, status: string) { + const event = this.eventRepo.create({ content, score, status }); + await this.eventRepo.save(event); + } + async getAnalytics() { + return this.eventRepo.find({ order: { timestamp: 'DESC' } }); + } } diff --git a/src/moderation/analytics/moderation-event.entity.ts b/src/moderation/analytics/moderation-event.entity.ts index 9cfa5a9d..1e77aa46 100644 --- a/src/moderation/analytics/moderation-event.entity.ts +++ b/src/moderation/analytics/moderation-event.entity.ts @@ -1,19 +1,14 @@ import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn } from 'typeorm'; - @Entity() export class ModerationEvent { - @PrimaryGeneratedColumn() - id: number; - - @Column('text') - content: string; - - @Column('float') - score: number; - - @Column() - status: string; - - @CreateDateColumn() - timestamp: Date; + @PrimaryGeneratedColumn() + id: number; + @Column('text') + content: string; + @Column('float') + score: number; + @Column() + status: string; + @CreateDateColumn() + timestamp: Date; } diff --git a/src/moderation/auto/auto-moderation.service.ts b/src/moderation/auto/auto-moderation.service.ts index 687cf58c..004fc380 100644 --- a/src/moderation/auto/auto-moderation.service.ts +++ b/src/moderation/auto/auto-moderation.service.ts @@ -1,28 +1,27 @@ import { Injectable } from '@nestjs/common'; import { HfInference } from '@huggingface/inference'; - @Injectable() export class AutoModerationService { - private hf: HfInference; - - constructor() { - this.hf = new HfInference(process.env.HUGGINGFACE_API_KEY); - } - - async analyze(content: string): Promise<{ flagged: boolean; reasons: string[]; score: number }> { - const result = await this.hf.textClassification({ - model: 's-nlp/roberta_toxicity_classifier', // or 'unitary/toxic-bert' - inputs: content, - }); - - // result is an array of { label, score } - const toxicLabel = result.find((r) => r.label.toLowerCase().includes('toxic')); - const score = toxicLabel ? toxicLabel.score : 0; - - return { - flagged: score > 0.7, // threshold can be tuned - reasons: score > 0.7 ? ['AI model detected toxicity'] : [], - score, - }; - } + private hf: HfInference; + constructor() { + this.hf = new HfInference(process.env.HUGGINGFACE_API_KEY); + } + async analyze(content: string): Promise<{ + flagged: boolean; + reasons: string[]; + score: number; + }> { + const result = await this.hf.textClassification({ + model: 's-nlp/roberta_toxicity_classifier', // or 'unitary/toxic-bert' + inputs: content, + }); + // result is an array of { label, score } + const toxicLabel = result.find((r) => r.label.toLowerCase().includes('toxic')); + const score = toxicLabel ? toxicLabel.score : 0; + return { + flagged: score > 0.7, // threshold can be tuned + reasons: score > 0.7 ? ['AI model detected toxicity'] : [], + score, + }; + } } diff --git a/src/moderation/manual/manual-review.service.ts b/src/moderation/manual/manual-review.service.ts index 4dcca2e7..6f3562e8 100644 --- a/src/moderation/manual/manual-review.service.ts +++ b/src/moderation/manual/manual-review.service.ts @@ -2,27 +2,22 @@ import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { ReviewItem } from './review-item.entity'; - @Injectable() export class ManualReviewService { - constructor( + constructor( @InjectRepository(ReviewItem) - private readonly reviewRepo: Repository, - ) {} - - async enqueue(content: string, safetyScore: number) { - const item = this.reviewRepo.create({ content, safetyScore, status: 'pending' }); - await this.reviewRepo.save(item); - } - - async getQueue(): Promise { - return this.reviewRepo.find({ - where: { status: 'pending' }, - order: { safetyScore: 'DESC', createdAt: 'ASC' }, // prioritize high risk, then oldest - }); - } - - async markReviewed(id: number) { - await this.reviewRepo.update(id, { status: 'reviewed' }); - } + private readonly reviewRepo: Repository) { } + async enqueue(content: string, safetyScore: number) { + const item = this.reviewRepo.create({ content, safetyScore, status: 'pending' }); + await this.reviewRepo.save(item); + } + async getQueue(): Promise { + return this.reviewRepo.find({ + where: { status: 'pending' }, + order: { safetyScore: 'DESC', createdAt: 'ASC' }, // prioritize high risk, then oldest + }); + } + async markReviewed(id: number) { + await this.reviewRepo.update(id, { status: 'reviewed' }); + } } diff --git a/src/moderation/manual/review-item.entity.ts b/src/moderation/manual/review-item.entity.ts index 8501bb6e..2f515828 100644 --- a/src/moderation/manual/review-item.entity.ts +++ b/src/moderation/manual/review-item.entity.ts @@ -1,19 +1,14 @@ import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn } from 'typeorm'; - @Entity() export class ReviewItem { - @PrimaryGeneratedColumn() - id: number; - - @Column('text') - content: string; - - @Column('float') - safetyScore: number; - - @Column({ default: 'pending' }) - status: 'pending' | 'reviewed'; - - @CreateDateColumn() - createdAt: Date; + @PrimaryGeneratedColumn() + id: number; + @Column('text') + content: string; + @Column('float') + safetyScore: number; + @Column({ default: 'pending' }) + status: 'pending' | 'reviewed'; + @CreateDateColumn() + createdAt: Date; } diff --git a/src/moderation/moderation.module.ts b/src/moderation/moderation.module.ts index b3aa6ef2..ab1046b4 100644 --- a/src/moderation/moderation.module.ts +++ b/src/moderation/moderation.module.ts @@ -7,16 +7,16 @@ import { ContentSafetyService } from './safety/content-safety.service'; import { ModerationAnalyticsService } from './analytics/moderation-analytics.service'; import { ReviewItem } from './manual/review-item.entity'; import { ModerationEvent } from './analytics/moderation-event.entity'; - @Module({ - imports: [TypeOrmModule.forFeature([ReviewItem, ModerationEvent])], - providers: [ - ModerationService, - AutoModerationService, - ManualReviewService, - ContentSafetyService, - ModerationAnalyticsService, - ], - exports: [ModerationService], + imports: [TypeOrmModule.forFeature([ReviewItem, ModerationEvent])], + providers: [ + ModerationService, + AutoModerationService, + ManualReviewService, + ContentSafetyService, + ModerationAnalyticsService, + ], + exports: [ModerationService], }) -export class ModerationModule {} +export class ModerationModule { +} diff --git a/src/moderation/moderation.service.ts b/src/moderation/moderation.service.ts index 271caea9..e59a73ca 100644 --- a/src/moderation/moderation.service.ts +++ b/src/moderation/moderation.service.ts @@ -3,27 +3,18 @@ import { AutoModerationService } from './auto/auto-moderation.service'; import { ManualReviewService } from './manual/manual-review.service'; import { ContentSafetyService } from './safety/content-safety.service'; import { ModerationAnalyticsService } from './analytics/moderation-analytics.service'; - @Injectable() export class ModerationService { - constructor( - private readonly autoModeration: AutoModerationService, - private readonly manualReview: ManualReviewService, - private readonly safetyService: ContentSafetyService, - private readonly analytics: ModerationAnalyticsService, - ) {} - - async moderateContent(content: string) { - const autoResult = await this.autoModeration.analyze(content); - const safetyScore = this.safetyService.scoreContent(content); - - if (autoResult.flagged || safetyScore > 0.7) { - await this.manualReview.enqueue(content, safetyScore); - this.analytics.logModerationEvent(content, safetyScore, 'flagged'); - return { status: 'flagged', safetyScore }; + constructor(private readonly autoModeration: AutoModerationService, private readonly manualReview: ManualReviewService, private readonly safetyService: ContentSafetyService, private readonly analytics: ModerationAnalyticsService) { } + async moderateContent(content: string) { + const autoResult = await this.autoModeration.analyze(content); + const safetyScore = this.safetyService.scoreContent(content); + if (autoResult.flagged || safetyScore > 0.7) { + await this.manualReview.enqueue(content, safetyScore); + this.analytics.logModerationEvent(content, safetyScore, 'flagged'); + return { status: 'flagged', safetyScore }; + } + this.analytics.logModerationEvent(content, safetyScore, 'approved'); + return { status: 'approved', safetyScore }; } - - this.analytics.logModerationEvent(content, safetyScore, 'approved'); - return { status: 'approved', safetyScore }; - } } diff --git a/src/moderation/safety/content-safety.service.ts b/src/moderation/safety/content-safety.service.ts index 9b71ccce..bf5cc484 100644 --- a/src/moderation/safety/content-safety.service.ts +++ b/src/moderation/safety/content-safety.service.ts @@ -1,12 +1,13 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class ContentSafetyService { - scoreContent(content: string): number { - // Simple scoring logic (replace with ML model later) - let score = 0; - if (/violence|hate|explicit/i.test(content)) score += 0.8; - if (/spam|scam/i.test(content)) score += 0.5; - return Math.min(score, 1); - } + scoreContent(content: string): number { + // Simple scoring logic (replace with ML model later) + let score = 0; + if (/violence|hate|explicit/i.test(content)) + score += 0.8; + if (/spam|scam/i.test(content)) + score += 0.5; + return Math.min(score, 1); + } } diff --git a/src/monitoring/alerting/alerting.service.ts b/src/monitoring/alerting/alerting.service.ts index c80787a7..662b204c 100644 --- a/src/monitoring/alerting/alerting.service.ts +++ b/src/monitoring/alerting/alerting.service.ts @@ -1,11 +1,9 @@ import { Injectable, Logger } from '@nestjs/common'; - @Injectable() export class AlertingService { - private readonly logger = new Logger(AlertingService.name); - - sendAlert(type: string, message: string, severity: 'INFO' | 'WARNING' | 'CRITICAL') { - this.logger.warn(`[ALERT][${severity}] ${type}: ${message}`); - // Here you would integrate with external services like Slack, Email, or PagerDuty - } + private readonly logger = new Logger(AlertingService.name); + sendAlert(type: string, message: string, severity: 'INFO' | 'WARNING' | 'CRITICAL') { + this.logger.warn(`[ALERT][${severity}] ${type}: ${message}`); + // Here you would integrate with external services like Slack, Email, or PagerDuty + } } diff --git a/src/monitoring/logging/typeorm-logger.ts b/src/monitoring/logging/typeorm-logger.ts index c82d2995..52d5d626 100644 --- a/src/monitoring/logging/typeorm-logger.ts +++ b/src/monitoring/logging/typeorm-logger.ts @@ -1,50 +1,39 @@ import { Logger, QueryRunner } from 'typeorm'; import { MetricsCollectionService } from '../metrics/metrics-collection.service'; - export class TypeOrmMonitoringLogger implements Logger { - constructor(private readonly metricsService: MetricsCollectionService) {} - - logQuery(_query: string, _parameters?: any[], _queryRunner?: QueryRunner) { - // Optional: console.log(`[Query]: ${query}`); - } - - logQueryError( - error: string | Error, - query: string, - _parameters?: any[], - _queryRunner?: QueryRunner, - ) { - console.error(`[Query Error]: ${error}`, query); - } - - logQuerySlow(time: number, query: string, _parameters?: any[], _queryRunner?: QueryRunner) { - console.warn(`[Slow Query]: ${time}ms - ${query}`); - const table = this.extractTable(query); - // time is in milliseconds, convert to seconds - this.metricsService.recordDbQuery('slow_query', table, time / 1000); - } - - logSchemaBuild(_message: string, _queryRunner?: QueryRunner) { - // console.log(`[Schema Build]: ${message}`); - } - logMigration(_message: string, _queryRunner?: QueryRunner) { - // console.log(`[Migration]: ${message}`); - } - log(level: 'log' | 'info' | 'warn', _message: any, _queryRunner?: QueryRunner) { - switch (level) { - case 'log': - case 'info': - // console.log(`[TypeORM]: ${message}`); - break; - case 'warn': - // console.warn(`[TypeORM]: ${message}`); - break; + constructor(private readonly metricsService: MetricsCollectionService) { } + logQuery(_query: string, _parameters?: unknown[], _queryRunner?: QueryRunner) { + // Optional: console.log(`[Query]: ${query}`); + } + logQueryError(error: string | Error, query: string, _parameters?: unknown[], _queryRunner?: QueryRunner) { + console.error(`[Query Error]: ${error}`, query); + } + logQuerySlow(time: number, query: string, _parameters?: unknown[], _queryRunner?: QueryRunner) { + console.warn(`[Slow Query]: ${time}ms - ${query}`); + const table = this.extractTable(query); + // time is in milliseconds, convert to seconds + this.metricsService.recordDbQuery('slow_query', table, time / 1000); + } + logSchemaBuild(_message: string, _queryRunner?: QueryRunner) { + // console.log(`[Schema Build]: ${message}`); + } + logMigration(_message: string, _queryRunner?: QueryRunner) { + // console.log(`[Migration]: ${message}`); + } + log(level: 'log' | 'info' | 'warn', _message: unknown, _queryRunner?: QueryRunner) { + switch (level) { + case 'log': + case 'info': + // console.log(`[TypeORM]: ${message}`); + break; + case 'warn': + // console.warn(`[TypeORM]: ${message}`); + break; + } + } + private extractTable(query: string): string { + // Simple regex to extract table name. Not perfect but useful for metrics. + const match = query.match(/(?:FROM|UPDATE|INSERT INTO|DELETE FROM)\s+"?([a-zA-Z0-9_]+)"?/i); + return match ? match[1] : 'unknown'; } - } - - private extractTable(query: string): string { - // Simple regex to extract table name. Not perfect but useful for metrics. - const match = query.match(/(?:FROM|UPDATE|INSERT INTO|DELETE FROM)\s+"?([a-zA-Z0-9_]+)"?/i); - return match ? match[1] : 'unknown'; - } } diff --git a/src/monitoring/metrics/metrics-collection.service.ts b/src/monitoring/metrics/metrics-collection.service.ts index c143d274..cf07cb13 100644 --- a/src/monitoring/metrics/metrics-collection.service.ts +++ b/src/monitoring/metrics/metrics-collection.service.ts @@ -1,153 +1,129 @@ import { Injectable, OnModuleInit } from '@nestjs/common'; import { Registry, collectDefaultMetrics, Histogram, Gauge, Counter } from 'prom-client'; - @Injectable() export class MetricsCollectionService implements OnModuleInit { - private registry: Registry; - public httpRequestDuration: Histogram; - public dbQueryDuration: Histogram; - public activeConnections: Gauge; - public userRegistrations: Counter; - public assessmentCompletions: Counter; - public learningPathProgress: Gauge; - public cacheHitRate: Gauge; - public queueProcessingTime: Histogram; - public emailCampaignsSent: Counter; - public backupOperations: Counter; - - constructor() { - this.registry = new Registry(); - - // HTTP Request Duration - this.httpRequestDuration = new Histogram({ - name: 'http_request_duration_seconds', - help: 'Duration of HTTP requests in seconds', - labelNames: ['method', 'route', 'status_code'], - buckets: [0.1, 0.3, 0.5, 1, 1.5, 2, 5], - registers: [this.registry], - }); - - // Database Query Duration - this.dbQueryDuration = new Histogram({ - name: 'db_query_duration_seconds', - help: 'Duration of database queries in seconds', - labelNames: ['query_type', 'table'], - buckets: [0.01, 0.05, 0.1, 0.5, 1, 2], - registers: [this.registry], - }); - - // Active Connections (Example of custom gauge) - this.activeConnections = new Gauge({ - name: 'active_connections_count', - help: 'Number of active connections', - registers: [this.registry], - }); - - // User Registrations Counter - this.userRegistrations = new Counter({ - name: 'user_registrations_total', - help: 'Total number of user registrations', - labelNames: ['user_type', 'source'], - registers: [this.registry], - }); - - // Assessment Completions Counter - this.assessmentCompletions = new Counter({ - name: 'assessment_completions_total', - help: 'Total number of assessment completions', - labelNames: ['assessment_type', 'difficulty'], - registers: [this.registry], - }); - - // Learning Path Progress Gauge - this.learningPathProgress = new Gauge({ - name: 'learning_path_progress_percentage', - help: 'Average learning path progress percentage', - labelNames: ['path_id', 'user_id'], - registers: [this.registry], - }); - - // Cache Hit Rate Gauge - this.cacheHitRate = new Gauge({ - name: 'cache_hit_rate_percentage', - help: 'Cache hit rate percentage', - labelNames: ['cache_type'], - registers: [this.registry], - }); - - // Queue Processing Time Histogram - this.queueProcessingTime = new Histogram({ - name: 'queue_processing_duration_seconds', - help: 'Duration of queue job processing in seconds', - labelNames: ['queue_name', 'job_type'], - buckets: [0.1, 0.5, 1, 2, 5, 10, 30], - registers: [this.registry], - }); - - // Email Campaigns Sent Counter - this.emailCampaignsSent = new Counter({ - name: 'email_campaigns_sent_total', - help: 'Total number of email campaigns sent', - labelNames: ['campaign_type', 'status'], - registers: [this.registry], - }); - - // Backup Operations Counter - this.backupOperations = new Counter({ - name: 'backup_operations_total', - help: 'Total number of backup operations', - labelNames: ['operation_type', 'status'], - registers: [this.registry], - }); - } - - onModuleInit() { - // Collect default system metrics (CPU, Memory, Event Loop, etc.) - collectDefaultMetrics({ register: this.registry }); - } - - getRegistry(): Registry { - return this.registry; - } - - async getMetrics(): Promise { - return this.registry.metrics(); - } - - recordHttpRequest(method: string, route: string, statusCode: number, duration: number) { - this.httpRequestDuration.observe({ method, route, status_code: statusCode }, duration); - } - - recordDbQuery(queryType: string, table: string, duration: number) { - this.dbQueryDuration.observe({ query_type: queryType, table }, duration); - } - - // Custom business metrics methods - recordUserRegistration(userType: string, source: string) { - this.userRegistrations.inc({ user_type: userType, source }); - } - - recordAssessmentCompletion(assessmentType: string, difficulty: string) { - this.assessmentCompletions.inc({ assessment_type: assessmentType, difficulty }); - } - - updateLearningPathProgress(pathId: string, userId: string, progress: number) { - this.learningPathProgress.set({ path_id: pathId, user_id: userId }, progress); - } - - updateCacheHitRate(cacheType: string, hitRate: number) { - this.cacheHitRate.set({ cache_type: cacheType }, hitRate); - } - - recordQueueProcessingTime(queueName: string, jobType: string, duration: number) { - this.queueProcessingTime.observe({ queue_name: queueName, job_type: jobType }, duration); - } - - recordEmailCampaignSent(campaignType: string, status: string) { - this.emailCampaignsSent.inc({ campaign_type: campaignType, status }); - } - - recordBackupOperation(operationType: string, status: string) { - this.backupOperations.inc({ operation_type: operationType, status }); - } + private registry: Registry; + public httpRequestDuration: Histogram; + public dbQueryDuration: Histogram; + public activeConnections: Gauge; + public userRegistrations: Counter; + public assessmentCompletions: Counter; + public learningPathProgress: Gauge; + public cacheHitRate: Gauge; + public queueProcessingTime: Histogram; + public emailCampaignsSent: Counter; + public backupOperations: Counter; + constructor() { + this.registry = new Registry(); + // HTTP Request Duration + this.httpRequestDuration = new Histogram({ + name: 'http_request_duration_seconds', + help: 'Duration of HTTP requests in seconds', + labelNames: ['method', 'route', 'status_code'], + buckets: [0.1, 0.3, 0.5, 1, 1.5, 2, 5], + registers: [this.registry], + }); + // Database Query Duration + this.dbQueryDuration = new Histogram({ + name: 'db_query_duration_seconds', + help: 'Duration of database queries in seconds', + labelNames: ['query_type', 'table'], + buckets: [0.01, 0.05, 0.1, 0.5, 1, 2], + registers: [this.registry], + }); + // Active Connections (Example of custom gauge) + this.activeConnections = new Gauge({ + name: 'active_connections_count', + help: 'Number of active connections', + registers: [this.registry], + }); + // User Registrations Counter + this.userRegistrations = new Counter({ + name: 'user_registrations_total', + help: 'Total number of user registrations', + labelNames: ['user_type', 'source'], + registers: [this.registry], + }); + // Assessment Completions Counter + this.assessmentCompletions = new Counter({ + name: 'assessment_completions_total', + help: 'Total number of assessment completions', + labelNames: ['assessment_type', 'difficulty'], + registers: [this.registry], + }); + // Learning Path Progress Gauge + this.learningPathProgress = new Gauge({ + name: 'learning_path_progress_percentage', + help: 'Average learning path progress percentage', + labelNames: ['path_id', 'user_id'], + registers: [this.registry], + }); + // Cache Hit Rate Gauge + this.cacheHitRate = new Gauge({ + name: 'cache_hit_rate_percentage', + help: 'Cache hit rate percentage', + labelNames: ['cache_type'], + registers: [this.registry], + }); + // Queue Processing Time Histogram + this.queueProcessingTime = new Histogram({ + name: 'queue_processing_duration_seconds', + help: 'Duration of queue job processing in seconds', + labelNames: ['queue_name', 'job_type'], + buckets: [0.1, 0.5, 1, 2, 5, 10, 30], + registers: [this.registry], + }); + // Email Campaigns Sent Counter + this.emailCampaignsSent = new Counter({ + name: 'email_campaigns_sent_total', + help: 'Total number of email campaigns sent', + labelNames: ['campaign_type', 'status'], + registers: [this.registry], + }); + // Backup Operations Counter + this.backupOperations = new Counter({ + name: 'backup_operations_total', + help: 'Total number of backup operations', + labelNames: ['operation_type', 'status'], + registers: [this.registry], + }); + } + onModuleInit() { + // Collect default system metrics (CPU, Memory, Event Loop, etc.) + collectDefaultMetrics({ register: this.registry }); + } + getRegistry(): Registry { + return this.registry; + } + async getMetrics(): Promise { + return this.registry.metrics(); + } + recordHttpRequest(method: string, route: string, statusCode: number, duration: number) { + this.httpRequestDuration.observe({ method, route, status_code: statusCode }, duration); + } + recordDbQuery(queryType: string, table: string, duration: number) { + this.dbQueryDuration.observe({ query_type: queryType, table }, duration); + } + // Custom business metrics methods + recordUserRegistration(userType: string, source: string) { + this.userRegistrations.inc({ user_type: userType, source }); + } + recordAssessmentCompletion(assessmentType: string, difficulty: string) { + this.assessmentCompletions.inc({ assessment_type: assessmentType, difficulty }); + } + updateLearningPathProgress(pathId: string, userId: string, progress: number) { + this.learningPathProgress.set({ path_id: pathId, user_id: userId }, progress); + } + updateCacheHitRate(cacheType: string, hitRate: number) { + this.cacheHitRate.set({ cache_type: cacheType }, hitRate); + } + recordQueueProcessingTime(queueName: string, jobType: string, duration: number) { + this.queueProcessingTime.observe({ queue_name: queueName, job_type: jobType }, duration); + } + recordEmailCampaignSent(campaignType: string, status: string) { + this.emailCampaignsSent.inc({ campaign_type: campaignType, status }); + } + recordBackupOperation(operationType: string, status: string) { + this.backupOperations.inc({ operation_type: operationType, status }); + } } diff --git a/src/monitoring/monitoring.controller.ts b/src/monitoring/monitoring.controller.ts index 233f6638..63ffd62b 100644 --- a/src/monitoring/monitoring.controller.ts +++ b/src/monitoring/monitoring.controller.ts @@ -2,125 +2,115 @@ import { Controller, Get, Query, Res } from '@nestjs/common'; import { MetricsCollectionService } from './metrics/metrics-collection.service'; import { Response } from 'express'; import { ScheduledTaskMonitoringService } from './scheduled-task-monitoring.service'; - @Controller('metrics') export class MonitoringController { - constructor( - private readonly metricsService: MetricsCollectionService, - private readonly scheduledTaskMonitoringService: ScheduledTaskMonitoringService, - ) {} - - @Get() - async getMetrics(@Res() res: Response) { - const metrics = await this.metricsService.getMetrics(); - res.set('Content-Type', this.metricsService.getRegistry().contentType); - res.send(metrics); - } - - @Get('unified') - async getUnifiedMetrics( - @Query('format') format?: string, - @Query('include') include?: string, - @Query('exclude') exclude?: string, - ) { - const includeTypes = include?.split(',').map((s) => s.trim()) || []; - const excludeTypes = exclude?.split(',').map((s) => s.trim()) || []; - - // Get base Prometheus metrics - const prometheusMetrics = await this.metricsService.getMetrics(); - - // Get scheduled tasks dashboard - const scheduledTasksMetrics = this.scheduledTaskMonitoringService.getDashboard(); - - // Aggregate metrics from different sources - const unifiedMetrics = { - prometheus: prometheusMetrics, - scheduledTasks: scheduledTasksMetrics, - timestamp: new Date().toISOString(), - metadata: { - totalMetrics: prometheusMetrics.split('\n').filter((line) => line && !line.startsWith('#')) - .length, - includeTypes, - excludeTypes, - }, - }; - - // Return in requested format - if (format === 'json') { - return unifiedMetrics; + constructor(private readonly metricsService: MetricsCollectionService, private readonly scheduledTaskMonitoringService: ScheduledTaskMonitoringService) { } + @Get() + async getMetrics( + @Res() + res: Response) { + const metrics = await this.metricsService.getMetrics(); + res.set('Content-Type', this.metricsService.getRegistry().contentType); + res.send(metrics); } - - // Default to Prometheus format - const metrics = await this.metricsService.getMetrics(); - return metrics; - } - - @Get('health') - async getMetricsHealth() { - return { - status: 'healthy', - timestamp: new Date().toISOString(), - services: { - metricsCollection: 'active', - scheduledTasks: 'active', - }, - registry: { - metricsCount: (await this.metricsService.getMetrics()) - .split('\n') - .filter((line) => line && !line.startsWith('#')).length, - }, - }; - } - - @Get('custom') - async getCustomMetrics(@Query('type') type?: string) { - const customMetrics = { - user_registrations: { - name: 'user_registrations_total', - help: 'Total number of user registrations', - type: 'counter', - }, - assessment_completions: { - name: 'assessment_completions_total', - help: 'Total number of assessment completions', - type: 'counter', - }, - learning_path_progress: { - name: 'learning_path_progress_percentage', - help: 'Average learning path progress percentage', - type: 'gauge', - }, - cache_hit_rate: { - name: 'cache_hit_rate_percentage', - help: 'Cache hit rate percentage', - type: 'gauge', - }, - queue_processing_time: { - name: 'queue_processing_duration_seconds', - help: 'Duration of queue job processing in seconds', - type: 'histogram', - }, - email_campaigns_sent: { - name: 'email_campaigns_sent_total', - help: 'Total number of email campaigns sent', - type: 'counter', - }, - backup_operations: { - name: 'backup_operations_total', - help: 'Total number of backup operations', - type: 'counter', - }, - }; - - if (type) { - return customMetrics[type] || { error: 'Metric type not found' }; + @Get('unified') + async getUnifiedMetrics( + @Query('format') + format?: string, + @Query('include') + include?: string, + @Query('exclude') + exclude?: string) { + const includeTypes = include?.split(',').map((s) => s.trim()) || []; + const excludeTypes = exclude?.split(',').map((s) => s.trim()) || []; + // Get base Prometheus metrics + const prometheusMetrics = await this.metricsService.getMetrics(); + // Get scheduled tasks dashboard + const scheduledTasksMetrics = this.scheduledTaskMonitoringService.getDashboard(); + // Aggregate metrics from different sources + const unifiedMetrics = { + prometheus: prometheusMetrics, + scheduledTasks: scheduledTasksMetrics, + timestamp: new Date().toISOString(), + metadata: { + totalMetrics: prometheusMetrics.split('\n').filter((line) => line && !line.startsWith('#')) + .length, + includeTypes, + excludeTypes, + }, + }; + // Return in requested format + if (format === 'json') { + return unifiedMetrics; + } + // Default to Prometheus format + const metrics = await this.metricsService.getMetrics(); + return metrics; + } + @Get('health') + async getMetricsHealth() { + return { + status: 'healthy', + timestamp: new Date().toISOString(), + services: { + metricsCollection: 'active', + scheduledTasks: 'active', + }, + registry: { + metricsCount: (await this.metricsService.getMetrics()) + .split('\n') + .filter((line) => line && !line.startsWith('#')).length, + }, + }; + } + @Get('custom') + async getCustomMetrics( + @Query('type') + type?: string) { + const customMetrics = { + user_registrations: { + name: 'user_registrations_total', + help: 'Total number of user registrations', + type: 'counter', + }, + assessment_completions: { + name: 'assessment_completions_total', + help: 'Total number of assessment completions', + type: 'counter', + }, + learning_path_progress: { + name: 'learning_path_progress_percentage', + help: 'Average learning path progress percentage', + type: 'gauge', + }, + cache_hit_rate: { + name: 'cache_hit_rate_percentage', + help: 'Cache hit rate percentage', + type: 'gauge', + }, + queue_processing_time: { + name: 'queue_processing_duration_seconds', + help: 'Duration of queue job processing in seconds', + type: 'histogram', + }, + email_campaigns_sent: { + name: 'email_campaigns_sent_total', + help: 'Total number of email campaigns sent', + type: 'counter', + }, + backup_operations: { + name: 'backup_operations_total', + help: 'Total number of backup operations', + type: 'counter', + }, + }; + if (type) { + return customMetrics[type] || { error: 'Metric type not found' }; + } + return customMetrics; + } + @Get('scheduled-tasks/dashboard') + getScheduledTasksDashboard() { + return this.scheduledTaskMonitoringService.getDashboard(); } - - return customMetrics; - } - - @Get('scheduled-tasks/dashboard') - getScheduledTasksDashboard() { - return this.scheduledTaskMonitoringService.getDashboard(); - } } diff --git a/src/monitoring/monitoring.module.ts b/src/monitoring/monitoring.module.ts index 2e6ba1cb..811e0b70 100644 --- a/src/monitoring/monitoring.module.ts +++ b/src/monitoring/monitoring.module.ts @@ -7,18 +7,18 @@ import { PerformanceAnalysisService } from './performance/performance-analysis.s import { OptimizationService } from './optimization/optimization.service'; import { AlertingService } from './alerting/alerting.service'; import { ScheduledTaskMonitoringService } from './scheduled-task-monitoring.service'; - @Module({ - imports: [ScheduleModule.forRoot()], - controllers: [MonitoringController], - providers: [ - MonitoringService, - MetricsCollectionService, - PerformanceAnalysisService, - OptimizationService, - AlertingService, - ScheduledTaskMonitoringService, - ], - exports: [MetricsCollectionService, AlertingService, ScheduledTaskMonitoringService], + imports: [ScheduleModule.forRoot()], + controllers: [MonitoringController], + providers: [ + MonitoringService, + MetricsCollectionService, + PerformanceAnalysisService, + OptimizationService, + AlertingService, + ScheduledTaskMonitoringService, + ], + exports: [MetricsCollectionService, AlertingService, ScheduledTaskMonitoringService], }) -export class MonitoringModule {} +export class MonitoringModule { +} diff --git a/src/monitoring/monitoring.service.ts b/src/monitoring/monitoring.service.ts index ac3d4ccc..7510ebbc 100644 --- a/src/monitoring/monitoring.service.ts +++ b/src/monitoring/monitoring.service.ts @@ -3,42 +3,25 @@ import { Cron, CronExpression } from '@nestjs/schedule'; import { PerformanceAnalysisService } from './performance/performance-analysis.service'; import { OptimizationService } from './optimization/optimization.service'; import { AlertingService } from './alerting/alerting.service'; - @Injectable() export class MonitoringService { - private readonly logger = new Logger(MonitoringService.name); - - constructor( - private readonly analysisService: PerformanceAnalysisService, - private readonly optimizationService: OptimizationService, - private readonly alertingService: AlertingService, - ) {} - - @Cron(CronExpression.EVERY_MINUTE) - async handleCron() { - this.logger.debug('Running system performance analysis...'); - const analysis = await this.analysisService.analyze(); - - // Check for alerts - if (analysis.cpuLoad > 80) { - this.alertingService.sendAlert( - 'CPU_HIGH', - `CPU Load is at ${analysis.cpuLoad.toFixed(2)}%`, - 'WARNING', - ); + private readonly logger = new Logger(MonitoringService.name); + constructor(private readonly analysisService: PerformanceAnalysisService, private readonly optimizationService: OptimizationService, private readonly alertingService: AlertingService) { } + @Cron(CronExpression.EVERY_MINUTE) + async handleCron() { + this.logger.debug('Running system performance analysis...'); + const analysis = await this.analysisService.analyze(); + // Check for alerts + if (analysis.cpuLoad > 80) { + this.alertingService.sendAlert('CPU_HIGH', `CPU Load is at ${analysis.cpuLoad.toFixed(2)}%`, 'WARNING'); + } + if (analysis.memoryUsage > 90) { + this.alertingService.sendAlert('MEMORY_HIGH', `Memory Usage is at ${analysis.memoryUsage.toFixed(2)}%`, 'WARNING'); + } + // Get optimization recommendations + const recommendations = this.optimizationService.getOptimizationRecommendations(analysis); + if (recommendations.length > 0) { + this.logger.log(`Optimization Recommendations: ${JSON.stringify(recommendations)}`); + } } - if (analysis.memoryUsage > 90) { - this.alertingService.sendAlert( - 'MEMORY_HIGH', - `Memory Usage is at ${analysis.memoryUsage.toFixed(2)}%`, - 'WARNING', - ); - } - - // Get optimization recommendations - const recommendations = this.optimizationService.getOptimizationRecommendations(analysis); - if (recommendations.length > 0) { - this.logger.log(`Optimization Recommendations: ${JSON.stringify(recommendations)}`); - } - } } diff --git a/src/monitoring/optimization/optimization.service.ts b/src/monitoring/optimization/optimization.service.ts index 82f75e0b..2b3589dd 100644 --- a/src/monitoring/optimization/optimization.service.ts +++ b/src/monitoring/optimization/optimization.service.ts @@ -1,35 +1,21 @@ import { Injectable, Logger } from '@nestjs/common'; - @Injectable() export class OptimizationService { - private readonly logger = new Logger(OptimizationService.name); - - getOptimizationRecommendations(analysisResult: any): string[] { - const recommendations: string[] = []; - - if (analysisResult.cpuLoad && analysisResult.cpuLoad > 80) { - recommendations.push( - 'High CPU usage detected. Consider horizontal scaling or optimizing CPU-intensive tasks.', - ); + private readonly logger = new Logger(OptimizationService.name); + getOptimizationRecommendations(analysisResult: unknown): string[] { + const recommendations: string[] = []; + if (analysisResult.cpuLoad && analysisResult.cpuLoad > 80) { + recommendations.push('High CPU usage detected. Consider horizontal scaling or optimizing CPU-intensive tasks.'); + } + if (analysisResult.memoryUsage && analysisResult.memoryUsage > 85) { + recommendations.push('High Memory usage detected. Check for memory leaks or increase heap size.'); + } + if (analysisResult.slowQueries && analysisResult.slowQueries.length > 0) { + recommendations.push(`Detected ${analysisResult.slowQueries.length} slow database queries. Consider adding indexes or optimizing query structure.`); + analysisResult.slowQueries.forEach((query: unknown) => { + recommendations.push(`- Optimize query on table '${query.table}' (Avg duration: ${query.duration}s)`); + }); + } + return recommendations; } - - if (analysisResult.memoryUsage && analysisResult.memoryUsage > 85) { - recommendations.push( - 'High Memory usage detected. Check for memory leaks or increase heap size.', - ); - } - - if (analysisResult.slowQueries && analysisResult.slowQueries.length > 0) { - recommendations.push( - `Detected ${analysisResult.slowQueries.length} slow database queries. Consider adding indexes or optimizing query structure.`, - ); - analysisResult.slowQueries.forEach((query: any) => { - recommendations.push( - `- Optimize query on table '${query.table}' (Avg duration: ${query.duration}s)`, - ); - }); - } - - return recommendations; - } } diff --git a/src/monitoring/performance/performance-analysis.service.ts b/src/monitoring/performance/performance-analysis.service.ts index 97316c60..1985d60e 100644 --- a/src/monitoring/performance/performance-analysis.service.ts +++ b/src/monitoring/performance/performance-analysis.service.ts @@ -1,49 +1,40 @@ import { Injectable, Logger } from '@nestjs/common'; import { MetricsCollectionService } from '../metrics/metrics-collection.service'; import * as os from 'os'; - @Injectable() export class PerformanceAnalysisService { - private readonly logger = new Logger(PerformanceAnalysisService.name); - - constructor(private readonly metricsService: MetricsCollectionService) {} - - async analyze(): Promise { - const metrics = await this.metricsService.getRegistry().getMetricsAsJSON(); - - // System Analysis - const cpus = os.cpus(); - const loadAvg = os.loadavg(); // [1min, 5min, 15min] - const totalMem = os.totalmem(); - const freeMem = os.freemem(); - const memoryUsagePercent = ((totalMem - freeMem) / totalMem) * 100; - - // CPU Load approximation (Load Average / Number of CPUs) - // On Windows, loadavg returns [0, 0, 0], so we might need a fallback or just report 0. - const cpuLoadPercent = cpus.length > 0 ? (loadAvg[0] / cpus.length) * 100 : 0; - - // Analyze Slow Queries from our custom Histogram - // We iterate through metrics to find db_query_duration_seconds - const slowQueries = []; - const dbMetric = metrics.find((m) => m.name === 'db_query_duration_seconds'); - - if (dbMetric && (dbMetric as any).values) { - // Check for queries falling into buckets > 1 second - // Histogram values are flattened. We look for bucket with le="2" or le="+Inf" and compare counts. - // For this simple implementation, we'll just check if we have recorded any high duration queries recently. - // In a real system, we'd query Prometheus. Here we check the metric state. - // This is a simplified check - // We can also check specific tracked slow queries if we stored them separately. + private readonly logger = new Logger(PerformanceAnalysisService.name); + constructor(private readonly metricsService: MetricsCollectionService) { } + async analyze(): Promise { + const metrics = await this.metricsService.getRegistry().getMetricsAsJSON(); + // System Analysis + const cpus = os.cpus(); + const loadAvg = os.loadavg(); // [1min, 5min, 15min] + const totalMem = os.totalmem(); + const freeMem = os.freemem(); + const memoryUsagePercent = ((totalMem - freeMem) / totalMem) * 100; + // CPU Load approximation (Load Average / Number of CPUs) + // On Windows, loadavg returns [0, 0, 0], so we might need a fallback or just report 0. + const cpuLoadPercent = cpus.length > 0 ? (loadAvg[0] / cpus.length) * 100 : 0; + // Analyze Slow Queries from our custom Histogram + // We iterate through metrics to find db_query_duration_seconds + const slowQueries = []; + const dbMetric = metrics.find((m) => m.name === 'db_query_duration_seconds'); + if (dbMetric && (dbMetric as unknown).values) { + // Check for queries falling into buckets > 1 second + // Histogram values are flattened. We look for bucket with le="2" or le="+Inf" and compare counts. + // For this simple implementation, we'll just check if we have recorded any high duration queries recently. + // In a real system, we'd query Prometheus. Here we check the metric state. + // This is a simplified check + // We can also check specific tracked slow queries if we stored them separately. + } + // Mock detection of slow queries for demonstration if metrics are empty + // In production, this would parse the histogram buckets + return { + timestamp: new Date(), + cpuLoad: cpuLoadPercent, + memoryUsage: memoryUsagePercent, + slowQueries, + }; } - - // Mock detection of slow queries for demonstration if metrics are empty - // In production, this would parse the histogram buckets - - return { - timestamp: new Date(), - cpuLoad: cpuLoadPercent, - memoryUsage: memoryUsagePercent, - slowQueries, - }; - } } diff --git a/src/monitoring/scheduled-task-monitoring.service.ts b/src/monitoring/scheduled-task-monitoring.service.ts index 5dcffda4..6a1c8377 100644 --- a/src/monitoring/scheduled-task-monitoring.service.ts +++ b/src/monitoring/scheduled-task-monitoring.service.ts @@ -1,259 +1,184 @@ import { Injectable, Logger } from '@nestjs/common'; import { Cron, CronExpression } from '@nestjs/schedule'; import { AlertingService } from './alerting/alerting.service'; - export type ScheduledTaskStatus = 'RUNNING' | 'SUCCESS' | 'FAILED' | 'TIMED_OUT'; - export interface ScheduledTaskConfig { - expectedIntervalMs: number; - timeoutMs: number; - maxRetries?: number; - missedExecutionGraceMs?: number; + expectedIntervalMs: number; + timeoutMs: number; + maxRetries?: number; + missedExecutionGraceMs?: number; } - export interface ScheduledTaskExecution { - executionId: string; - taskName: string; - status: ScheduledTaskStatus; - startedAt: Date; - endedAt?: Date; - durationMs?: number; - errorMessage?: string; - attempt?: number; - maxRetries?: number; - metadata?: Record; + executionId: string; + taskName: string; + status: ScheduledTaskStatus; + startedAt: Date; + endedAt?: Date; + durationMs?: number; + errorMessage?: string; + attempt?: number; + maxRetries?: number; + metadata?: Record; } - @Injectable() export class ScheduledTaskMonitoringService { - private readonly logger = new Logger(ScheduledTaskMonitoringService.name); - private readonly historyLimitPerTask = 50; - - private readonly taskConfigs = new Map(); - private readonly activeExecutions = new Map(); - private readonly executionHistory = new Map(); - private readonly retryStats = new Map(); - private readonly lastMissedAlertAt = new Map(); - private readonly taskRegisteredAt = new Map(); - - constructor(private readonly alertingService: AlertingService) {} - - registerTask(taskName: string, config: ScheduledTaskConfig): void { - if (!this.taskRegisteredAt.has(taskName)) { - this.taskRegisteredAt.set(taskName, new Date()); + private readonly logger = new Logger(ScheduledTaskMonitoringService.name); + private readonly historyLimitPerTask = 50; + private readonly taskConfigs = new Map(); + private readonly activeExecutions = new Map(); + private readonly executionHistory = new Map(); + private readonly retryStats = new Map(); + private readonly lastMissedAlertAt = new Map(); + private readonly taskRegisteredAt = new Map(); + constructor(private readonly alertingService: AlertingService) { } + registerTask(taskName: string, config: ScheduledTaskConfig): void { + if (!this.taskRegisteredAt.has(taskName)) { + this.taskRegisteredAt.set(taskName, new Date()); + } + this.taskConfigs.set(taskName, { + ...config, + maxRetries: config.maxRetries ?? 0, + missedExecutionGraceMs: config.missedExecutionGraceMs ?? 60000, + }); } - - this.taskConfigs.set(taskName, { - ...config, - maxRetries: config.maxRetries ?? 0, - missedExecutionGraceMs: config.missedExecutionGraceMs ?? 60000, - }); - } - - startExecution( - taskName: string, - config: ScheduledTaskConfig, - metadata: Record = {}, - ): string { - this.registerTask(taskName, config); - - const executionId = `${taskName}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`; - const execution: ScheduledTaskExecution = { - executionId, - taskName, - status: 'RUNNING', - startedAt: new Date(), - maxRetries: this.taskConfigs.get(taskName)?.maxRetries, - metadata, - }; - - this.activeExecutions.set(executionId, execution); - this.pushHistory(taskName, execution); - this.logger.debug(`Started scheduled task execution: ${taskName} (${executionId})`); - - return executionId; - } - - markSuccess(executionId: string, metadata: Record = {}): void { - const execution = this.activeExecutions.get(executionId); - if (!execution) { - return; + startExecution(taskName: string, config: ScheduledTaskConfig, metadata: Record = {}): string { + this.registerTask(taskName, config); + const executionId = `${taskName}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`; + const execution: ScheduledTaskExecution = { + executionId, + taskName, + status: 'RUNNING', + startedAt: new Date(), + maxRetries: this.taskConfigs.get(taskName)?.maxRetries, + metadata, + }; + this.activeExecutions.set(executionId, execution); + this.pushHistory(taskName, execution); + this.logger.debug(`Started scheduled task execution: ${taskName} (${executionId})`); + return executionId; } - - execution.endedAt = new Date(); - execution.durationMs = execution.endedAt.getTime() - execution.startedAt.getTime(); - execution.status = 'SUCCESS'; - execution.metadata = { - ...(execution.metadata || {}), - ...metadata, - }; - - this.activeExecutions.delete(executionId); - - this.alertingService.sendAlert( - 'SCHEDULED_TASK_SUCCESS', - `Task ${execution.taskName} completed successfully in ${execution.durationMs}ms`, - 'INFO', - ); - } - - markFailure( - executionId: string, - error: Error | string, - metadata: Record = {}, - ): void { - const execution = this.activeExecutions.get(executionId); - if (!execution) { - return; + markSuccess(executionId: string, metadata: Record = {}): void { + const execution = this.activeExecutions.get(executionId); + if (!execution) { + return; + } + execution.endedAt = new Date(); + execution.durationMs = execution.endedAt.getTime() - execution.startedAt.getTime(); + execution.status = 'SUCCESS'; + execution.metadata = { + ...(execution.metadata || {}), + ...metadata, + }; + this.activeExecutions.delete(executionId); + this.alertingService.sendAlert('SCHEDULED_TASK_SUCCESS', `Task ${execution.taskName} completed successfully in ${execution.durationMs}ms`, 'INFO'); } - - const errorMessage = typeof error === 'string' ? error : error.message; - - execution.endedAt = new Date(); - execution.durationMs = execution.endedAt.getTime() - execution.startedAt.getTime(); - execution.status = 'FAILED'; - execution.errorMessage = errorMessage; - execution.metadata = { - ...(execution.metadata || {}), - ...metadata, - }; - - this.activeExecutions.delete(executionId); - - this.alertingService.sendAlert( - 'SCHEDULED_TASK_FAILURE', - `Task ${execution.taskName} failed: ${errorMessage}`, - 'CRITICAL', - ); - } - - recordRetry(taskName: string, attempt: number, maxRetries: number, reason?: string): void { - const current = this.retryStats.get(taskName) || { totalRetries: 0 }; - current.totalRetries += 1; - current.lastRetryAt = new Date(); - this.retryStats.set(taskName, current); - - this.alertingService.sendAlert( - 'SCHEDULED_TASK_RETRY', - `Task ${taskName} retry attempt ${attempt}/${maxRetries}${reason ? `: ${reason}` : ''}`, - 'WARNING', - ); - } - - @Cron(CronExpression.EVERY_MINUTE) - monitorScheduledTasks(): void { - const now = new Date(); - - for (const [executionId, execution] of this.activeExecutions.entries()) { - const config = this.taskConfigs.get(execution.taskName); - if (!config) { - continue; - } - - const elapsedMs = now.getTime() - execution.startedAt.getTime(); - if (elapsedMs > config.timeoutMs) { - execution.endedAt = now; - execution.durationMs = elapsedMs; - execution.status = 'TIMED_OUT'; - execution.errorMessage = `Execution exceeded timeout of ${config.timeoutMs}ms`; - + markFailure(executionId: string, error: Error | string, metadata: Record = {}): void { + const execution = this.activeExecutions.get(executionId); + if (!execution) { + return; + } + const errorMessage = typeof error === 'string' ? error : error.message; + execution.endedAt = new Date(); + execution.durationMs = execution.endedAt.getTime() - execution.startedAt.getTime(); + execution.status = 'FAILED'; + execution.errorMessage = errorMessage; + execution.metadata = { + ...(execution.metadata || {}), + ...metadata, + }; this.activeExecutions.delete(executionId); - - this.alertingService.sendAlert( - 'SCHEDULED_TASK_TIMEOUT', - `Task ${execution.taskName} exceeded timeout (${elapsedMs}ms)`, - 'CRITICAL', - ); - } + this.alertingService.sendAlert('SCHEDULED_TASK_FAILURE', `Task ${execution.taskName} failed: ${errorMessage}`, 'CRITICAL'); + } + recordRetry(taskName: string, attempt: number, maxRetries: number, reason?: string): void { + const current = this.retryStats.get(taskName) || { totalRetries: 0 }; + current.totalRetries += 1; + current.lastRetryAt = new Date(); + this.retryStats.set(taskName, current); + this.alertingService.sendAlert('SCHEDULED_TASK_RETRY', `Task ${taskName} retry attempt ${attempt}/${maxRetries}${reason ? `: ${reason}` : ''}`, 'WARNING'); + } + @Cron(CronExpression.EVERY_MINUTE) + monitorScheduledTasks(): void { + const now = new Date(); + for (const [executionId, execution] of this.activeExecutions.entries()) { + const config = this.taskConfigs.get(execution.taskName); + if (!config) { + continue; + } + const elapsedMs = now.getTime() - execution.startedAt.getTime(); + if (elapsedMs > config.timeoutMs) { + execution.endedAt = now; + execution.durationMs = elapsedMs; + execution.status = 'TIMED_OUT'; + execution.errorMessage = `Execution exceeded timeout of ${config.timeoutMs}ms`; + this.activeExecutions.delete(executionId); + this.alertingService.sendAlert('SCHEDULED_TASK_TIMEOUT', `Task ${execution.taskName} exceeded timeout (${elapsedMs}ms)`, 'CRITICAL'); + } + } + for (const [taskName, config] of this.taskConfigs.entries()) { + const history = this.executionHistory.get(taskName) || []; + const latest = history[history.length - 1]; + const baseline = latest?.startedAt || this.taskRegisteredAt.get(taskName); + if (!baseline) { + continue; + } + const elapsedSinceStart = now.getTime() - baseline.getTime(); + const threshold = config.expectedIntervalMs + (config.missedExecutionGraceMs || 0); + const isOverdue = elapsedSinceStart > threshold; + const alreadyAlertedAt = this.lastMissedAlertAt.get(taskName); + const shouldAlertAgain = !alreadyAlertedAt || now.getTime() - alreadyAlertedAt.getTime() > config.expectedIntervalMs; + if (isOverdue && shouldAlertAgain) { + this.lastMissedAlertAt.set(taskName, now); + this.alertingService.sendAlert('SCHEDULED_TASK_MISSED', `Task ${taskName} appears delayed/missed. Last start was ${baseline.toISOString()}`, 'CRITICAL'); + } + } } - - for (const [taskName, config] of this.taskConfigs.entries()) { - const history = this.executionHistory.get(taskName) || []; - const latest = history[history.length - 1]; - - const baseline = latest?.startedAt || this.taskRegisteredAt.get(taskName); - if (!baseline) { - continue; - } - - const elapsedSinceStart = now.getTime() - baseline.getTime(); - const threshold = config.expectedIntervalMs + (config.missedExecutionGraceMs || 0); - const isOverdue = elapsedSinceStart > threshold; - const alreadyAlertedAt = this.lastMissedAlertAt.get(taskName); - const shouldAlertAgain = - !alreadyAlertedAt || now.getTime() - alreadyAlertedAt.getTime() > config.expectedIntervalMs; - - if (isOverdue && shouldAlertAgain) { - this.lastMissedAlertAt.set(taskName, now); - this.alertingService.sendAlert( - 'SCHEDULED_TASK_MISSED', - `Task ${taskName} appears delayed/missed. Last start was ${baseline.toISOString()}`, - 'CRITICAL', - ); - } + getDashboard() { + const now = new Date(); + const tasks = Array.from(this.taskConfigs.entries()).map(([taskName, config]) => { + const history = this.executionHistory.get(taskName) || []; + const lastExecution = history[history.length - 1] || null; + const lastSuccess = [...history].reverse().find((entry) => entry.status === 'SUCCESS') || null; + const lastFailure = [...history] + .reverse() + .find((entry) => entry.status === 'FAILED' || entry.status === 'TIMED_OUT') || null; + const activeCount = Array.from(this.activeExecutions.values()).filter((entry) => entry.taskName === taskName).length; + const threshold = config.expectedIntervalMs + (config.missedExecutionGraceMs || 0); + const missed = !!lastExecution && + now.getTime() - lastExecution.startedAt.getTime() > threshold && + activeCount === 0; + const retryStats = this.retryStats.get(taskName) || { totalRetries: 0 }; + return { + taskName, + config, + activeCount, + missed, + lastExecution, + lastSuccessAt: lastSuccess?.endedAt || null, + lastFailureAt: lastFailure?.endedAt || null, + retry: retryStats, + recentHistory: history.slice(-10), + }; + }); + return { + generatedAt: now, + summary: { + taskCount: tasks.length, + activeExecutions: this.activeExecutions.size, + tasksWithMissedExecutions: tasks.filter((task) => task.missed).length, + tasksWithRecentFailures: tasks.filter((task) => task.lastExecution?.status === 'FAILED' || task.lastExecution?.status === 'TIMED_OUT').length, + }, + tasks, + }; } - } - - getDashboard() { - const now = new Date(); - - const tasks = Array.from(this.taskConfigs.entries()).map(([taskName, config]) => { - const history = this.executionHistory.get(taskName) || []; - const lastExecution = history[history.length - 1] || null; - const lastSuccess = - [...history].reverse().find((entry) => entry.status === 'SUCCESS') || null; - const lastFailure = - [...history] - .reverse() - .find((entry) => entry.status === 'FAILED' || entry.status === 'TIMED_OUT') || null; - const activeCount = Array.from(this.activeExecutions.values()).filter( - (entry) => entry.taskName === taskName, - ).length; - - const threshold = config.expectedIntervalMs + (config.missedExecutionGraceMs || 0); - const missed = - !!lastExecution && - now.getTime() - lastExecution.startedAt.getTime() > threshold && - activeCount === 0; - - const retryStats = this.retryStats.get(taskName) || { totalRetries: 0 }; - - return { - taskName, - config, - activeCount, - missed, - lastExecution, - lastSuccessAt: lastSuccess?.endedAt || null, - lastFailureAt: lastFailure?.endedAt || null, - retry: retryStats, - recentHistory: history.slice(-10), - }; - }); - - return { - generatedAt: now, - summary: { - taskCount: tasks.length, - activeExecutions: this.activeExecutions.size, - tasksWithMissedExecutions: tasks.filter((task) => task.missed).length, - tasksWithRecentFailures: tasks.filter( - (task) => - task.lastExecution?.status === 'FAILED' || task.lastExecution?.status === 'TIMED_OUT', - ).length, - }, - tasks, - }; - } - - private pushHistory(taskName: string, execution: ScheduledTaskExecution): void { - const existing = this.executionHistory.get(taskName) || []; - existing.push(execution); - - if (existing.length > this.historyLimitPerTask) { - existing.splice(0, existing.length - this.historyLimitPerTask); + private pushHistory(taskName: string, execution: ScheduledTaskExecution): void { + const existing = this.executionHistory.get(taskName) || []; + existing.push(execution); + if (existing.length > this.historyLimitPerTask) { + existing.splice(0, existing.length - this.historyLimitPerTask); + } + this.executionHistory.set(taskName, existing); } - - this.executionHistory.set(taskName, existing); - } } diff --git a/src/notifications/dto/notification.dto.ts b/src/notifications/dto/notification.dto.ts index 00c108fd..636223f7 100644 --- a/src/notifications/dto/notification.dto.ts +++ b/src/notifications/dto/notification.dto.ts @@ -1,79 +1,55 @@ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; -import { - IsEnum, - IsNotEmpty, - IsOptional, - IsString, - IsUUID, - IsBoolean, - IsObject, -} from 'class-validator'; +import { IsEnum, IsNotEmpty, IsOptional, IsString, IsUUID, IsBoolean, IsObject, } from 'class-validator'; import { NotificationType, NotificationPriority } from '../entities/notification.entity'; - export class CreateNotificationDto { - @ApiProperty({ description: 'The ID of the user the notification is for' }) - @IsUUID() - @IsNotEmpty() - userId: string; - - @ApiProperty({ description: 'The title of the notification' }) - @IsString() - @IsNotEmpty() - title: string; - - @ApiProperty({ description: 'The content of the notification' }) - @IsString() - @IsNotEmpty() - content: string; - - @ApiProperty({ enum: NotificationType, description: 'The type of the notification' }) - @IsEnum(NotificationType) - @IsOptional() - type?: NotificationType; - - @ApiProperty({ enum: NotificationPriority, description: 'The priority of the notification' }) - @IsEnum(NotificationPriority) - @IsOptional() - priority?: NotificationPriority; - - @ApiPropertyOptional({ description: 'Additional metadata for the notification' }) - @IsObject() - @IsOptional() - metadata?: Record; + @ApiProperty({ description: 'The ID of the user the notification is for' }) + @IsUUID() + @IsNotEmpty() + userId: string; + @ApiProperty({ description: 'The title of the notification' }) + @IsString() + @IsNotEmpty() + title: string; + @ApiProperty({ description: 'The content of the notification' }) + @IsString() + @IsNotEmpty() + content: string; + @ApiProperty({ enum: NotificationType, description: 'The type of the notification' }) + @IsEnum(NotificationType) + @IsOptional() + type?: NotificationType; + @ApiProperty({ enum: NotificationPriority, description: 'The priority of the notification' }) + @IsEnum(NotificationPriority) + @IsOptional() + priority?: NotificationPriority; + @ApiPropertyOptional({ description: 'Additional metadata for the notification' }) + @IsObject() + @IsOptional() + metadata?: Record; } - export class UpdateNotificationDto { - @ApiPropertyOptional({ description: 'Mark as read or unread' }) - @IsBoolean() - @IsOptional() - isRead?: boolean; + @ApiPropertyOptional({ description: 'Mark as read or unread' }) + @IsBoolean() + @IsOptional() + isRead?: boolean; } - export class NotificationResponseDto { - @ApiProperty() - id: string; - - @ApiProperty() - userId: string; - - @ApiProperty() - title: string; - - @ApiProperty() - content: string; - - @ApiProperty({ enum: NotificationType }) - type: NotificationType; - - @ApiProperty({ enum: NotificationPriority }) - priority: NotificationPriority; - - @ApiProperty() - isRead: boolean; - - @ApiPropertyOptional() - readAt?: Date; - - @ApiProperty() - createdAt: Date; + @ApiProperty() + id: string; + @ApiProperty() + userId: string; + @ApiProperty() + title: string; + @ApiProperty() + content: string; + @ApiProperty({ enum: NotificationType }) + type: NotificationType; + @ApiProperty({ enum: NotificationPriority }) + priority: NotificationPriority; + @ApiProperty() + isRead: boolean; + @ApiPropertyOptional() + readAt?: Date; + @ApiProperty() + createdAt: Date; } diff --git a/src/notifications/email/email.processor.ts b/src/notifications/email/email.processor.ts index 9d3e1715..91a1258d 100644 --- a/src/notifications/email/email.processor.ts +++ b/src/notifications/email/email.processor.ts @@ -3,37 +3,31 @@ import { Logger } from '@nestjs/common'; import { Job } from 'bull'; import { EmailService, EmailOptions } from './email.service'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; - @Processor(QUEUE_NAMES.EMAIL) export class EmailProcessor { - private readonly logger = new Logger(EmailProcessor.name); - - constructor(private readonly emailService: EmailService) {} - - @Process(JOB_NAMES.SEND_EMAIL) - async handleSendEmail(job: Job) { - this.logger.log(`Processing email job ${job.id} for ${job.data.to}`); - - try { - await this.emailService.sendEmail(job.data); - this.logger.log(`Email job ${job.id} completed successfully`); - } catch (error) { - this.logger.error(`Email job ${job.id} failed: ${error.message}`, error.stack); - - // Check if job should be retried using built-in Bull retry settings - const maxAttempts = job.opts.attempts || 3; - - if (job.attemptsMade >= maxAttempts) { - // Send to dead letter queue - this.logger.error( - `Job ${job.id} moved to dead letter queue after ${job.attemptsMade} attempts. Final error: ${error.message}`, - ); - // In production, you would push to a dead letter queue here: - // await this.deadLetterQueue.add('failed-email', { ...job.data, error: error.message }); - } else { - // Re-throw to let Bull handle the retry with exponential backoff - throw error; - } + private readonly logger = new Logger(EmailProcessor.name); + constructor(private readonly emailService: EmailService) { } + @Process(JOB_NAMES.SEND_EMAIL) + async handleSendEmail(job: Job) { + this.logger.log(`Processing email job ${job.id} for ${job.data.to}`); + try { + await this.emailService.sendEmail(job.data); + this.logger.log(`Email job ${job.id} completed successfully`); + } + catch (error) { + this.logger.error(`Email job ${job.id} failed: ${error.message}`, error.stack); + // Check if job should be retried using built-in Bull retry settings + const maxAttempts = job.opts.attempts || 3; + if (job.attemptsMade >= maxAttempts) { + // Send to dead letter queue + this.logger.error(`Job ${job.id} moved to dead letter queue after ${job.attemptsMade} attempts. Final error: ${error.message}`); + // In production, you would push to a dead letter queue here: + // await this.deadLetterQueue.add('failed-email', { ...job.data, error: error.message }); + } + else { + // Re-throw to let Bull handle the retry with exponential backoff + throw error; + } + } } - } } diff --git a/src/notifications/email/email.service.spec.ts b/src/notifications/email/email.service.spec.ts index d7f8b90c..b1712d70 100644 --- a/src/notifications/email/email.service.spec.ts +++ b/src/notifications/email/email.service.spec.ts @@ -2,92 +2,73 @@ import { Test, TestingModule } from '@nestjs/testing'; import { ConfigService } from '@nestjs/config'; import { getQueueToken } from '@nestjs/bull'; import { EmailService } from './email.service'; - describe('EmailService', () => { - let service: EmailService; - let mockQueue: any; - - beforeEach(async () => { - mockQueue = { - add: jest.fn().mockResolvedValue({}), - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - EmailService, - { - provide: ConfigService, - useValue: { - get: jest.fn((key: string) => { - const config = { - SMTP_HOST: 'smtp.test.com', - SMTP_PORT: 587, - SMTP_SECURE: false, - SMTP_USER: 'test@test.com', - SMTP_PASS: 'password', - EMAIL_FROM: 'noreply@teachlink.io', - EMAIL_FROM_NAME: 'TeachLink', - APP_URL: 'http://localhost:3000', - }; - return config[key]; - }), - }, - }, - { - provide: getQueueToken('email'), - useValue: mockQueue, - }, - ], - }).compile(); - - service = module.get(EmailService); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('sendVerificationEmail', () => { - it('should queue verification email', async () => { - const email = 'test@example.com'; - const token = 'test-token-123'; - - await service.sendVerificationEmail(email, token); - - expect(mockQueue.add).toHaveBeenCalledWith( - 'send-email', - expect.objectContaining({ - to: email, - subject: 'Verify Your Email - TeachLink', - template: 'verification', - context: expect.objectContaining({ - verificationUrl: expect.stringContaining(token), - }), - }), - expect.any(Object), - ); + let service: EmailService; + let mockQueue: unknown; + beforeEach(async () => { + mockQueue = { + add: jest.fn().mockResolvedValue({}), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + EmailService, + { + provide: ConfigService, + useValue: { + get: jest.fn((key: string) => { + const config = { + SMTP_HOST: 'smtp.test.com', + SMTP_PORT: 587, + SMTP_SECURE: false, + SMTP_USER: 'test@test.com', + SMTP_PASS: 'password', + EMAIL_FROM: 'noreply@teachlink.io', + EMAIL_FROM_NAME: 'TeachLink', + APP_URL: 'http://localhost:3000', + }; + return config[key]; + }), + }, + }, + { + provide: getQueueToken('email'), + useValue: mockQueue, + }, + ], + }).compile(); + service = module.get(EmailService); }); - }); - - describe('sendPasswordResetEmail', () => { - it('should queue password reset email', async () => { - const email = 'test@example.com'; - const token = 'reset-token-456'; - - await service.sendPasswordResetEmail(email, token); - - expect(mockQueue.add).toHaveBeenCalledWith( - 'send-email', - expect.objectContaining({ - to: email, - subject: 'Reset Your Password - TeachLink', - template: 'reset-password', - context: expect.objectContaining({ - resetUrl: expect.stringContaining(token), - }), - }), - expect.any(Object), - ); + it('should be defined', () => { + expect(service).toBeDefined(); + }); + describe('sendVerificationEmail', () => { + it('should queue verification email', async () => { + const email = 'test@example.com'; + const token = 'test-token-123'; + await service.sendVerificationEmail(email, token); + expect(mockQueue.add).toHaveBeenCalledWith('send-email', expect.objectContaining({ + to: email, + subject: 'Verify Your Email - TeachLink', + template: 'verification', + context: expect.objectContaining({ + verificationUrl: expect.stringContaining(token), + }), + }), expect.any(Object)); + }); + }); + describe('sendPasswordResetEmail', () => { + it('should queue password reset email', async () => { + const email = 'test@example.com'; + const token = 'reset-token-456'; + await service.sendPasswordResetEmail(email, token); + expect(mockQueue.add).toHaveBeenCalledWith('send-email', expect.objectContaining({ + to: email, + subject: 'Reset Your Password - TeachLink', + template: 'reset-password', + context: expect.objectContaining({ + resetUrl: expect.stringContaining(token), + }), + }), expect.any(Object)); + }); }); - }); }); diff --git a/src/notifications/email/email.service.ts b/src/notifications/email/email.service.ts index e34bae0f..9fad84bd 100644 --- a/src/notifications/email/email.service.ts +++ b/src/notifications/email/email.service.ts @@ -7,146 +7,118 @@ import * as path from 'path'; import { InjectQueue } from '@nestjs/bull'; import { Queue } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; - export interface EmailOptions { - to: string; - subject: string; - template: string; - context: Record; + to: string; + subject: string; + template: string; + context: Record; } - @Injectable() export class EmailService { - private readonly logger = new Logger(EmailService.name); - private transporter: nodemailer.Transporter; - private templatesCache: Map = new Map(); - - constructor( - private readonly configService: ConfigService, - @InjectQueue(QUEUE_NAMES.EMAIL) private readonly emailQueue: Queue, - ) { - this.initializeTransporter(); - } - - private initializeTransporter() { - this.transporter = nodemailer.createTransport({ - host: this.configService.get('SMTP_HOST'), - port: this.configService.get('SMTP_PORT'), - secure: this.configService.get('SMTP_SECURE') || false, - auth: { - user: this.configService.get('SMTP_USER'), - pass: this.configService.get('SMTP_PASS'), - }, - }); - } - - async sendVerificationEmail(email: string, token: string): Promise { - const appUrl = this.configService.get('APP_URL') || 'http://localhost:3000'; - - await this.emailQueue.add( - JOB_NAMES.SEND_EMAIL, - { - to: email, - subject: 'Verify Your Email - TeachLink', - template: 'verification', - context: { - verificationUrl: `${appUrl}/auth/verify-email?token=${token}`, - appUrl, - }, - }, - { - attempts: 5, - backoff: { - type: 'exponential', - delay: 2000, - }, - }, - ); - - this.logger.log(`Verification email queued for ${email}`); - } - - async sendPasswordResetEmail(email: string, token: string): Promise { - const appUrl = this.configService.get('APP_URL') || 'http://localhost:3000'; - - await this.emailQueue.add( - JOB_NAMES.SEND_EMAIL, - { - to: email, - subject: 'Reset Your Password - TeachLink', - template: 'reset-password', - context: { - resetUrl: `${appUrl}/auth/reset-password?token=${token}`, - appUrl, - }, - }, - { - attempts: 5, - backoff: { - type: 'exponential', - delay: 2000, - }, - }, - ); - - this.logger.log(`Password reset email queued for ${email}`); - } - - async sendEmail(options: EmailOptions): Promise { - try { - const template = await this.getTemplate(options.template); - const html = template(options.context); - - const mailOptions = { - from: `${this.configService.get('EMAIL_FROM_NAME')} <${this.configService.get('EMAIL_FROM')}>`, - to: options.to, - subject: options.subject, - html, - }; - - await this.transporter.sendMail(mailOptions); - this.logger.log(`Email sent successfully to ${options.to}`); - } catch (error) { - this.logger.error(`Failed to send email to ${options.to}`, error.stack); - - // Check if we should retry or send to dead letter queue - const maxAttempts = 5; - const attemptNumber = 1; - - if (attemptNumber >= maxAttempts) { - this.logger.error( - `Email to ${options.to} moved to dead letter queue. Error: ${error.message}`, - ); - // Here you would push to a dead letter queue: - // await this.deadLetterQueue.add('failed-email', { ...options, error: error.message }); - } else { - throw error; - } + private readonly logger = new Logger(EmailService.name); + private transporter: nodemailer.Transporter; + private templatesCache: Map = new Map(); + constructor(private readonly configService: ConfigService, + @InjectQueue(QUEUE_NAMES.EMAIL) + private readonly emailQueue: Queue) { + this.initializeTransporter(); } - } - - private async getTemplate(templateName: string): Promise { - if (this.templatesCache.has(templateName)) { - return this.templatesCache.get(templateName); + private initializeTransporter() { + this.transporter = nodemailer.createTransport({ + host: this.configService.get('SMTP_HOST'), + port: this.configService.get('SMTP_PORT'), + secure: this.configService.get('SMTP_SECURE') || false, + auth: { + user: this.configService.get('SMTP_USER'), + pass: this.configService.get('SMTP_PASS'), + }, + }); } - - const templatePath = path.join(__dirname, 'templates', `${templateName}.hbs`); - - const templateSource = fs.readFileSync(templatePath, 'utf-8'); - const template = handlebars.compile(templateSource); - - this.templatesCache.set(templateName, template); - return template; - } - - async verifyConnection(): Promise { - try { - await this.transporter.verify(); - this.logger.log('Email service connection verified'); - return true; - } catch (error) { - this.logger.error('Email service connection failed', error.stack); - return false; + async sendVerificationEmail(email: string, token: string): Promise { + const appUrl = this.configService.get('APP_URL') || 'http://localhost:3000'; + await this.emailQueue.add(JOB_NAMES.SEND_EMAIL, { + to: email, + subject: 'Verify Your Email - TeachLink', + template: 'verification', + context: { + verificationUrl: `${appUrl}/auth/verify-email?token=${token}`, + appUrl, + }, + }, { + attempts: 5, + backoff: { + type: 'exponential', + delay: 2000, + }, + }); + this.logger.log(`Verification email queued for ${email}`); + } + async sendPasswordResetEmail(email: string, token: string): Promise { + const appUrl = this.configService.get('APP_URL') || 'http://localhost:3000'; + await this.emailQueue.add(JOB_NAMES.SEND_EMAIL, { + to: email, + subject: 'Reset Your Password - TeachLink', + template: 'reset-password', + context: { + resetUrl: `${appUrl}/auth/reset-password?token=${token}`, + appUrl, + }, + }, { + attempts: 5, + backoff: { + type: 'exponential', + delay: 2000, + }, + }); + this.logger.log(`Password reset email queued for ${email}`); + } + async sendEmail(options: EmailOptions): Promise { + try { + const template = await this.getTemplate(options.template); + const html = template(options.context); + const mailOptions = { + from: `${this.configService.get('EMAIL_FROM_NAME')} <${this.configService.get('EMAIL_FROM')}>`, + to: options.to, + subject: options.subject, + html, + }; + await this.transporter.sendMail(mailOptions); + this.logger.log(`Email sent successfully to ${options.to}`); + } + catch (error) { + this.logger.error(`Failed to send email to ${options.to}`, error.stack); + // Check if we should retry or send to dead letter queue + const maxAttempts = 5; + const attemptNumber = 1; + if (attemptNumber >= maxAttempts) { + this.logger.error(`Email to ${options.to} moved to dead letter queue. Error: ${error.message}`); + // Here you would push to a dead letter queue: + // await this.deadLetterQueue.add('failed-email', { ...options, error: error.message }); + } + else { + throw error; + } + } + } + private async getTemplate(templateName: string): Promise { + if (this.templatesCache.has(templateName)) { + return this.templatesCache.get(templateName); + } + const templatePath = path.join(__dirname, 'templates', `${templateName}.hbs`); + const templateSource = fs.readFileSync(templatePath, 'utf-8'); + const template = handlebars.compile(templateSource); + this.templatesCache.set(templateName, template); + return template; + } + async verifyConnection(): Promise { + try { + await this.transporter.verify(); + this.logger.log('Email service connection verified'); + return true; + } + catch (error) { + this.logger.error('Email service connection failed', error.stack); + return false; + } } - } } diff --git a/src/notifications/entities/notification-preferences.entity.ts b/src/notifications/entities/notification-preferences.entity.ts index a56ddfdc..01fb3c98 100644 --- a/src/notifications/entities/notification-preferences.entity.ts +++ b/src/notifications/entities/notification-preferences.entity.ts @@ -1,50 +1,30 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - OneToOne, - JoinColumn, - CreateDateColumn, - UpdateDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, OneToOne, JoinColumn, CreateDateColumn, UpdateDateColumn, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; - @Entity('notification_preferences') export class NotificationPreferences { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - userId: string; - - @OneToOne(() => User, { onDelete: 'CASCADE' }) - @JoinColumn() - user: User; - - @Column({ default: true }) - emailEnabled: boolean; - - @Column({ default: true }) - pushEnabled: boolean; - - @Column({ default: true }) - inAppEnabled: boolean; - - @Column({ default: false }) - smsEnabled: boolean; - - @Column({ type: 'jsonb', nullable: true }) - topicSubscriptions: Record; - - @Column({ type: 'varchar', default: '09:00' }) - quietTimeStart: string; - - @Column({ type: 'varchar', default: '21:00' }) - quietTimeEnd: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + userId: string; + @OneToOne(() => User, { onDelete: 'CASCADE' }) + @JoinColumn() + user: User; + @Column({ default: true }) + emailEnabled: boolean; + @Column({ default: true }) + pushEnabled: boolean; + @Column({ default: true }) + inAppEnabled: boolean; + @Column({ default: false }) + smsEnabled: boolean; + @Column({ type: 'jsonb', nullable: true }) + topicSubscriptions: Record; + @Column({ type: 'varchar', default: '09:00' }) + quietTimeStart: string; + @Column({ type: 'varchar', default: '21:00' }) + quietTimeEnd: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/notifications/entities/notification.entity.ts b/src/notifications/entities/notification.entity.ts index d99d27a6..a1d89136 100644 --- a/src/notifications/entities/notification.entity.ts +++ b/src/notifications/entities/notification.entity.ts @@ -1,77 +1,53 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, Index, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; - export enum NotificationType { - EMAIL = 'email', - PUSH = 'push', - IN_APP = 'in_app', - SMS = 'sms', + EMAIL = 'email', + PUSH = 'push', + IN_APP = 'in_app', + SMS = 'sms' } - export enum NotificationPriority { - LOW = 'low', - MEDIUM = 'medium', - HIGH = 'high', - URGENT = 'urgent', + LOW = 'low', + MEDIUM = 'medium', + HIGH = 'high', + URGENT = 'urgent' } - @Entity('notifications') export class Notification { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - userId: string; - - @ManyToOne(() => User, { onDelete: 'CASCADE' }) - user: User; - - @Column() - title: string; - - @Column('text') - content: string; - - @Column({ - type: 'enum', - enum: NotificationType, - default: NotificationType.IN_APP, - }) - type: NotificationType; - - @Column({ - type: 'enum', - enum: NotificationPriority, - default: NotificationPriority.MEDIUM, - }) - priority: NotificationPriority; - - @Column({ default: false }) - isRead: boolean; - - @Column({ type: 'jsonb', nullable: true }) - metadata: Record; - - @Column({ nullable: true }) - readAt: Date; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - @Index() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + userId: string; + @ManyToOne(() => User, { onDelete: 'CASCADE' }) + user: User; + @Column() + title: string; + @Column('text') + content: string; + @Column({ + type: 'enum', + enum: NotificationType, + default: NotificationType.IN_APP, + }) + type: NotificationType; + @Column({ + type: 'enum', + enum: NotificationPriority, + default: NotificationPriority.MEDIUM, + }) + priority: NotificationPriority; + @Column({ default: false }) + isRead: boolean; + @Column({ type: 'jsonb', nullable: true }) + metadata: Record; + @Column({ nullable: true }) + readAt: Date; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + @Index() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/notifications/notification-templates.service.ts b/src/notifications/notification-templates.service.ts index 1a19225e..33a464e0 100644 --- a/src/notifications/notification-templates.service.ts +++ b/src/notifications/notification-templates.service.ts @@ -1,67 +1,57 @@ import { Injectable, Logger } from '@nestjs/common'; import { NotificationType } from './entities/notification.entity'; - export interface NotificationTemplate { - title: string; - content: string; + title: string; + content: string; } - @Injectable() export class NotificationTemplatesService { - private readonly logger = new Logger(NotificationTemplatesService.name); - private readonly templates: Map NotificationTemplate> = new Map(); - - constructor() { - this.registerTemplates(); - } - - private registerTemplates() { - this.templates.set('COURSE_ENROLLMENT', (data: any) => ({ - title: 'Course Enrollment', - content: `Welcome to the course ${data.courseName}! You have been successfully enrolled.`, - })); - - this.templates.set('PAYMENT_RECEIVED', (data: any) => ({ - title: 'Payment Confirmed', - content: `A payment of ${data.amount} ${data.currency} for the course ${data.courseName} was successfully processed.`, - })); - - this.templates.set('NEW_MESSAGE', (data: any) => ({ - title: 'New Message', - content: `You have received a new message from ${data.senderName}.`, - })); - - this.templates.set('COURSE_REPLY', (data: any) => ({ - title: 'Course Discussion Reply', - content: `${data.senderName} replied to your comment in ${data.courseName}.`, - })); - } - - renderTemplate(type: string, data: any): NotificationTemplate { - const templateFn = this.templates.get(type); - if (!templateFn) { - this.logger.warn(`Template callback not found for type: ${type}`); - return { - title: 'Notification', - content: JSON.stringify(data), - }; + private readonly logger = new Logger(NotificationTemplatesService.name); + private readonly templates: Map NotificationTemplate> = new Map(); + constructor() { + this.registerTemplates(); } - - return templateFn(data); - } - - formatForType(n: NotificationTemplate, type: NotificationType): string { - switch (type) { - case NotificationType.EMAIL: - return `

${n.title}

${n.content}

`; - case NotificationType.PUSH: - return n.content; - case NotificationType.IN_APP: - return n.content; - case NotificationType.SMS: - return `${n.title}: ${n.content}`; - default: - return n.content; + private registerTemplates() { + this.templates.set('COURSE_ENROLLMENT', (data: unknown) => ({ + title: 'Course Enrollment', + content: `Welcome to the course ${data.courseName}! You have been successfully enrolled.`, + })); + this.templates.set('PAYMENT_RECEIVED', (data: unknown) => ({ + title: 'Payment Confirmed', + content: `A payment of ${data.amount} ${data.currency} for the course ${data.courseName} was successfully processed.`, + })); + this.templates.set('NEW_MESSAGE', (data: unknown) => ({ + title: 'New Message', + content: `You have received a new message from ${data.senderName}.`, + })); + this.templates.set('COURSE_REPLY', (data: unknown) => ({ + title: 'Course Discussion Reply', + content: `${data.senderName} replied to your comment in ${data.courseName}.`, + })); + } + renderTemplate(type: string, data: unknown): NotificationTemplate { + const templateFn = this.templates.get(type); + if (!templateFn) { + this.logger.warn(`Template callback not found for type: ${type}`); + return { + title: 'Notification', + content: JSON.stringify(data), + }; + } + return templateFn(data); + } + formatForType(n: NotificationTemplate, type: NotificationType): string { + switch (type) { + case NotificationType.EMAIL: + return `

${n.title}

${n.content}

`; + case NotificationType.PUSH: + return n.content; + case NotificationType.IN_APP: + return n.content; + case NotificationType.SMS: + return `${n.title}: ${n.content}`; + default: + return n.content; + } } - } } diff --git a/src/notifications/notifications.controller.ts b/src/notifications/notifications.controller.ts index b04ad4cd..96b3665b 100644 --- a/src/notifications/notifications.controller.ts +++ b/src/notifications/notifications.controller.ts @@ -1,80 +1,78 @@ -import { - Controller, - Get, - Patch, - Delete, - Param, - Body, - UseGuards, - Query, - HttpCode, - HttpStatus, -} from '@nestjs/common'; +import { Controller, Get, Patch, Delete, Param, Body, UseGuards, Query, HttpCode, HttpStatus, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiBearerAuth, ApiParam, ApiResponse } from '@nestjs/swagger'; import { NotificationsService } from './notifications.service'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { CurrentUser } from '../auth/decorators/current-user.decorator'; import { NotificationResponseDto } from './dto/notification.dto'; import { NotificationPreferences } from './entities/notification-preferences.entity'; - @ApiTags('Notifications') @ApiBearerAuth() @UseGuards(JwtAuthGuard) @Controller('notifications') export class NotificationsController { - constructor(private readonly notificationsService: NotificationsService) {} - - @Get() - @ApiOperation({ summary: 'Get all notifications for current user' }) - @ApiResponse({ status: 200, type: [NotificationResponseDto] }) - async getMyNotifications( - @CurrentUser('id') userId: string, - @Query('isRead') isRead?: boolean, - @Query('limit') limit?: number, - @Query('offset') offset?: number, - ) { - const [data, total] = await this.notificationsService.findAllForUser(userId, { - isRead, - limit, - offset, - }); - return { data, total }; - } - - @Patch(':id/read') - @ApiOperation({ summary: 'Mark a notification as read' }) - @ApiParam({ name: 'id', description: 'Notification ID' }) - async markAsRead(@Param('id') id: string, @CurrentUser('id') userId: string) { - return this.notificationsService.markAsRead(id, userId); - } - - @Patch('read-all') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Mark all notifications as read' }) - async markAllAsRead(@CurrentUser('id') userId: string) { - await this.notificationsService.markAllAsRead(userId); - } - - @Delete(':id') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete a notification' }) - @ApiParam({ name: 'id', description: 'Notification ID' }) - async deleteNotification(@Param('id') id: string, @CurrentUser('id') userId: string) { - await this.notificationsService.remove(id, userId); - } - - @Get('preferences') - @ApiOperation({ summary: 'Get current user notification preferences' }) - async getPreferences(@CurrentUser('id') userId: string) { - return this.notificationsService.getPreferences(userId); - } - - @Patch('preferences') - @ApiOperation({ summary: 'Update notification preferences' }) - async updatePreferences( - @CurrentUser('id') userId: string, - @Body() preferencesDto: Partial, - ) { - return this.notificationsService.updatePreferences(userId, preferencesDto); - } + constructor(private readonly notificationsService: NotificationsService) { } + @Get() + @ApiOperation({ summary: 'Get all notifications for current user' }) + @ApiResponse({ status: 200, type: [NotificationResponseDto] }) + async getMyNotifications( + @CurrentUser('id') + userId: string, + @Query('isRead') + isRead?: boolean, + @Query('limit') + limit?: number, + @Query('offset') + offset?: number) { + const [data, total] = await this.notificationsService.findAllForUser(userId, { + isRead, + limit, + offset, + }); + return { data, total }; + } + @Patch(':id/read') + @ApiOperation({ summary: 'Mark a notification as read' }) + @ApiParam({ name: 'id', description: 'Notification ID' }) + async markAsRead( + @Param('id') + id: string, + @CurrentUser('id') + userId: string) { + return this.notificationsService.markAsRead(id, userId); + } + @Patch('read-all') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Mark all notifications as read' }) + async markAllAsRead( + @CurrentUser('id') + userId: string) { + await this.notificationsService.markAllAsRead(userId); + } + @Delete(':id') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete a notification' }) + @ApiParam({ name: 'id', description: 'Notification ID' }) + async deleteNotification( + @Param('id') + id: string, + @CurrentUser('id') + userId: string) { + await this.notificationsService.remove(id, userId); + } + @Get('preferences') + @ApiOperation({ summary: 'Get current user notification preferences' }) + async getPreferences( + @CurrentUser('id') + userId: string) { + return this.notificationsService.getPreferences(userId); + } + @Patch('preferences') + @ApiOperation({ summary: 'Update notification preferences' }) + async updatePreferences( + @CurrentUser('id') + userId: string, + @Body() + preferencesDto: Partial) { + return this.notificationsService.updatePreferences(userId, preferencesDto); + } } diff --git a/src/notifications/notifications.gateway.ts b/src/notifications/notifications.gateway.ts index b17e6a71..4e2c8908 100644 --- a/src/notifications/notifications.gateway.ts +++ b/src/notifications/notifications.gateway.ts @@ -1,12 +1,4 @@ -import { - WebSocketGateway, - WebSocketServer, - SubscribeMessage, - OnGatewayConnection, - OnGatewayDisconnect, - ConnectedSocket, - MessageBody, -} from '@nestjs/websockets'; +import { WebSocketGateway, WebSocketServer, SubscribeMessage, OnGatewayConnection, OnGatewayDisconnect, ConnectedSocket, MessageBody, } from '@nestjs/websockets'; import { Server, Socket } from 'socket.io'; import { UseGuards, Logger } from '@nestjs/common'; import { WsJwtAuthGuard } from '../auth/guards/ws-jwt-auth.guard'; @@ -14,98 +6,88 @@ import { Notification } from './entities/notification.entity'; import { wsManager } from '../common/utils/websocket.utils'; import { WsThrottlerGuard } from '../common/guards/ws-throttler.guard'; import { NOTIFICATION_GATEWAY_EVENTS } from '../collaboration/constants/collaboration-events.constants'; - @WebSocketGateway({ - cors: { - origin: '*', - }, - namespace: 'notifications', + cors: { + origin: '*', + }, + namespace: 'notifications', }) @UseGuards(WsThrottlerGuard) export class NotificationsGateway implements OnGatewayConnection, OnGatewayDisconnect { - @WebSocketServer() - server: Server; - - private readonly logger = new Logger(NotificationsGateway.name); - private userSockets: Map> = new Map(); - - async handleConnection(client: Socket) { - // Optionally we can get userId here from token if required, but currently it handles subscribe via message. - // However, for connection limit we can temporarily register with client id if no token or if handled at subscribe. - // Let's enforce global connection limits here. - if (wsManager.getTotalConnections() >= 5000) { - // Same as MAX_GLOBAL_CONNECTIONS - client.emit('error', { message: 'Server is at maximum capacity' }); - client.disconnect(true); - return; + @WebSocketServer() + server: Server; + private readonly logger = new Logger(NotificationsGateway.name); + private userSockets: Map> = new Map(); + async handleConnection(client: Socket) { + // Optionally we can get userId here from token if required, but currently it handles subscribe via message. + // However, for connection limit we can temporarily register with client id if no token or if handled at subscribe. + // Let's enforce global connection limits here. + if (wsManager.getTotalConnections() >= 5000) { + // Same as MAX_GLOBAL_CONNECTIONS + client.emit('error', { message: 'Server is at maximum capacity' }); + client.disconnect(true); + return; + } + this.logger.log(`Client connected: ${client.id}`); + } + handleDisconnect(client: Socket) { + wsManager.cleanupSocket(client); + this.logger.log(`Client disconnected: ${client.id}`); + this.removeSocketId(client.id); + } + @UseGuards(WsJwtAuthGuard) + @SubscribeMessage(NOTIFICATION_GATEWAY_EVENTS.SUBSCRIBE) + async handleSubscribe( + @ConnectedSocket() + client: Socket, + @MessageBody() + data: { + userId: string; + }) { + const user = (client as unknown).user; + const userId = user?.id || data.userId; + if (!userId) { + this.logger.warn(`User ID not found for client: ${client.id}`); + return; + } + const registered = wsManager.registerConnection(userId, client); + if (!registered) { + return { status: 'error', message: 'Connection limit reached' }; + } + this.addSocketId(userId, client.id); + client.join(`user:${userId}`); + this.logger.log(`User ${userId} subscribed to notifications via socket ${client.id}`); + return { status: 'subscribed', userId }; } - this.logger.log(`Client connected: ${client.id}`); - } - - handleDisconnect(client: Socket) { - wsManager.cleanupSocket(client); - this.logger.log(`Client disconnected: ${client.id}`); - this.removeSocketId(client.id); - } - - @UseGuards(WsJwtAuthGuard) - @SubscribeMessage(NOTIFICATION_GATEWAY_EVENTS.SUBSCRIBE) - async handleSubscribe( - @ConnectedSocket() client: Socket, - @MessageBody() data: { userId: string }, - ) { - const user = (client as any).user; - const userId = user?.id || data.userId; - - if (!userId) { - this.logger.warn(`User ID not found for client: ${client.id}`); - return; + /** + * Send notification to a specific user in real-time + */ + async sendToUser(userId: string, notification: Notification) { + this.server.to(`user:${userId}`).emit(NOTIFICATION_GATEWAY_EVENTS.NOTIFICATION, notification); + this.logger.debug(`Notification sent to user:${userId}`); } - - const registered = wsManager.registerConnection(userId, client); - if (!registered) { - return { status: 'error', message: 'Connection limit reached' }; + /** + * Broadcast notification to all users + */ + async broadcast(notification: Partial) { + this.server.emit(NOTIFICATION_GATEWAY_EVENTS.BROADCAST_NOTIFICATION, notification); + this.logger.debug('Broadcast notification sent'); } - - this.addSocketId(userId, client.id); - client.join(`user:${userId}`); - this.logger.log(`User ${userId} subscribed to notifications via socket ${client.id}`); - - return { status: 'subscribed', userId }; - } - - /** - * Send notification to a specific user in real-time - */ - async sendToUser(userId: string, notification: Notification) { - this.server.to(`user:${userId}`).emit(NOTIFICATION_GATEWAY_EVENTS.NOTIFICATION, notification); - this.logger.debug(`Notification sent to user:${userId}`); - } - - /** - * Broadcast notification to all users - */ - async broadcast(notification: Partial) { - this.server.emit(NOTIFICATION_GATEWAY_EVENTS.BROADCAST_NOTIFICATION, notification); - this.logger.debug('Broadcast notification sent'); - } - - private addSocketId(userId: string, socketId: string) { - if (!this.userSockets.has(userId)) { - this.userSockets.set(userId, new Set()); + private addSocketId(userId: string, socketId: string) { + if (!this.userSockets.has(userId)) { + this.userSockets.set(userId, new Set()); + } + this.userSockets.get(userId)?.add(socketId); } - this.userSockets.get(userId)?.add(socketId); - } - - private removeSocketId(socketId: string) { - for (const [userId, sockets] of this.userSockets.entries()) { - if (sockets.has(socketId)) { - sockets.delete(socketId); - if (sockets.size === 0) { - this.userSockets.delete(userId); + private removeSocketId(socketId: string) { + for (const [userId, sockets] of this.userSockets.entries()) { + if (sockets.has(socketId)) { + sockets.delete(socketId); + if (sockets.size === 0) { + this.userSockets.delete(userId); + } + break; + } } - break; - } } - } } diff --git a/src/notifications/notifications.module.ts b/src/notifications/notifications.module.ts index cd19b321..720fab42 100644 --- a/src/notifications/notifications.module.ts +++ b/src/notifications/notifications.module.ts @@ -13,29 +13,29 @@ import { EmailService } from './email/email.service'; import { EmailProcessor } from './email/email.processor'; import { Notification } from './entities/notification.entity'; import { NotificationPreferences } from './entities/notification-preferences.entity'; - @Global() @Module({ - imports: [ - TypeOrmModule.forFeature([Notification, NotificationPreferences]), - ConfigModule, - JwtModule.register({ - secret: process.env.JWT_SECRET || 'default-secret', - signOptions: { expiresIn: '24h' }, - }), - BullModule.registerQueue({ - name: QUEUE_NAMES.EMAIL, - }), - ], - controllers: [NotificationsController], - providers: [ - NotificationsService, - NotificationsGateway, - NotificationTemplatesService, - PreferencesService, - EmailService, - EmailProcessor, - ], - exports: [NotificationsService, PreferencesService], + imports: [ + TypeOrmModule.forFeature([Notification, NotificationPreferences]), + ConfigModule, + JwtModule.register({ + secret: process.env.JWT_SECRET || 'default-secret', + signOptions: { expiresIn: '24h' }, + }), + BullModule.registerQueue({ + name: QUEUE_NAMES.EMAIL, + }), + ], + controllers: [NotificationsController], + providers: [ + NotificationsService, + NotificationsGateway, + NotificationTemplatesService, + PreferencesService, + EmailService, + EmailProcessor, + ], + exports: [NotificationsService, PreferencesService], }) -export class NotificationsModule {} +export class NotificationsModule { +} diff --git a/src/notifications/notifications.service.ts b/src/notifications/notifications.service.ts index a38c210a..ebc4d1d6 100644 --- a/src/notifications/notifications.service.ts +++ b/src/notifications/notifications.service.ts @@ -3,11 +3,7 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { OnEvent } from '@nestjs/event-emitter'; import { APP_EVENTS } from '../common/constants/event.constants'; -import { - Notification, - NotificationType, - NotificationPriority, -} from './entities/notification.entity'; +import { Notification, NotificationType, NotificationPriority, } from './entities/notification.entity'; import { NotificationPreferences } from './entities/notification-preferences.entity'; import { CreateNotificationDto } from './dto/notification.dto'; import { NotificationsGateway } from './notifications.gateway'; @@ -15,242 +11,188 @@ import { NotificationTemplatesService } from './notification-templates.service'; import { PreferencesService } from './preferences/preferences.service'; import { EmailService } from './email/email.service'; import { sanitizeEmail } from '../common/utils/pii-sanitizer.utils'; - @Injectable() export class NotificationsService { - private readonly logger = new Logger(NotificationsService.name); - - constructor( + private readonly logger = new Logger(NotificationsService.name); + constructor( @InjectRepository(Notification) - private readonly notificationRepository: Repository, - private readonly gateway: NotificationsGateway, - private readonly templatesService: NotificationTemplatesService, - private readonly preferencesService: PreferencesService, - private readonly emailService: EmailService, - ) {} - - async sendVerificationEmail(email: string, token: string): Promise { - try { - await this.emailService.sendVerificationEmail(email, token); - this.logger.log(`Verification email sent to ${sanitizeEmail(email)}`); - } catch (error) { - this.logger.error( - `Failed to send verification email to ${sanitizeEmail(email)}`, - error instanceof Error ? error.stack : String(error), - ); - throw error; + private readonly notificationRepository: Repository, private readonly gateway: NotificationsGateway, private readonly templatesService: NotificationTemplatesService, private readonly preferencesService: PreferencesService, private readonly emailService: EmailService) { } + async sendVerificationEmail(email: string, token: string): Promise { + try { + await this.emailService.sendVerificationEmail(email, token); + this.logger.log(`Verification email sent to ${sanitizeEmail(email)}`); + } + catch (error) { + this.logger.error(`Failed to send verification email to ${sanitizeEmail(email)}`, error instanceof Error ? error.stack : String(error)); + throw error; + } } - } - - async sendPasswordResetEmail(email: string, token: string): Promise { - try { - await this.emailService.sendPasswordResetEmail(email, token); - this.logger.log(`Password reset email sent to ${sanitizeEmail(email)}`); - } catch (error) { - this.logger.error( - `Failed to send password reset email to ${sanitizeEmail(email)}`, - error instanceof Error ? error.stack : String(error), - ); - throw error; + async sendPasswordResetEmail(email: string, token: string): Promise { + try { + await this.emailService.sendPasswordResetEmail(email, token); + this.logger.log(`Password reset email sent to ${sanitizeEmail(email)}`); + } + catch (error) { + this.logger.error(`Failed to send password reset email to ${sanitizeEmail(email)}`, error instanceof Error ? error.stack : String(error)); + throw error; + } } - } - - /** - * Create and send a notification - */ - async create(createNotificationDto: CreateNotificationDto): Promise { - const { userId, title, content, type, priority, metadata } = createNotificationDto; - - const preferences = await this.preferencesService.getPreferences(userId); - const shouldSend = this.shouldSendNotification(type || NotificationType.IN_APP, preferences); - - if (!shouldSend) { - this.logger.debug(`Notification skipped for user ${userId} based on preferences`); + /** + * Create and send a notification + */ + async create(createNotificationDto: CreateNotificationDto): Promise { + const { userId, title, content, type, priority, metadata } = createNotificationDto; + const preferences = await this.preferencesService.getPreferences(userId); + const shouldSend = this.shouldSendNotification(type || NotificationType.IN_APP, preferences); + if (!shouldSend) { + this.logger.debug(`Notification skipped for user ${userId} based on preferences`); + } + const notification = this.notificationRepository.create({ + userId, + title, + content, + type: type || NotificationType.IN_APP, + priority: priority || NotificationPriority.MEDIUM, + metadata, + }); + const savedNotification = await this.notificationRepository.save(notification); + if (shouldSend) { + await this.sendNotification(savedNotification); + } + return savedNotification; } - - const notification = this.notificationRepository.create({ - userId, - title, - content, - type: type || NotificationType.IN_APP, - priority: priority || NotificationPriority.MEDIUM, - metadata, - }); - - const savedNotification = await this.notificationRepository.save(notification); - - if (shouldSend) { - await this.sendNotification(savedNotification); + /** + * Send notification via the specified channel + */ + private async sendNotification(notification: Notification): Promise { + try { + if (notification.type === NotificationType.IN_APP || + notification.type === NotificationType.PUSH) { + await this.gateway.sendToUser(notification.userId, notification); + } + if (notification.type === NotificationType.EMAIL) { + await this.sendEmailNotification(notification); + } + if (notification.type === NotificationType.PUSH) { + await this.sendExternalPushNotification(notification); + } + } + catch (error) { + this.logger.error(`Failed to send notification ${notification.id}`, error instanceof Error ? error.stack : String(error)); + } } - - return savedNotification; - } - - /** - * Send notification via the specified channel - */ - private async sendNotification(notification: Notification): Promise { - try { - if ( - notification.type === NotificationType.IN_APP || - notification.type === NotificationType.PUSH - ) { - await this.gateway.sendToUser(notification.userId, notification); - } - - if (notification.type === NotificationType.EMAIL) { - await this.sendEmailNotification(notification); - } - - if (notification.type === NotificationType.PUSH) { - await this.sendExternalPushNotification(notification); - } - } catch (error) { - this.logger.error( - `Failed to send notification ${notification.id}`, - error instanceof Error ? error.stack : String(error), - ); + private async sendEmailNotification(notification: Notification): Promise { + this.logger.log(`Sending email notification to user ${notification.userId}: ${notification.title}`); + // Integrate with EmailService or MailerService here if notification email delivery is required } - } - - private async sendEmailNotification(notification: Notification): Promise { - this.logger.log( - `Sending email notification to user ${notification.userId}: ${notification.title}`, - ); - // Integrate with EmailService or MailerService here if notification email delivery is required - } - - private async sendExternalPushNotification(notification: Notification): Promise { - this.logger.log( - `Sending external push notification to user ${notification.userId}: ${notification.title}`, - ); - // Integrate with FCM, OneSignal, etc. - } - - /** - * Get all notifications for a user - */ - async findAllForUser( - userId: string, - options: { isRead?: boolean; limit?: number; offset?: number } = {}, - ): Promise<[Notification[], number]> { - const query = this.notificationRepository - .createQueryBuilder('notification') - .where('notification.userId = :userId', { userId }); - - if (options.isRead !== undefined) { - query.andWhere('notification.isRead = :isRead', { - isRead: options.isRead, - }); + private async sendExternalPushNotification(notification: Notification): Promise { + this.logger.log(`Sending external push notification to user ${notification.userId}: ${notification.title}`); + // Integrate with FCM, OneSignal, etc. } - - query - .orderBy('notification.createdAt', 'DESC') - .take(options.limit || 20) - .skip(options.offset || 0); - - return query.getManyAndCount(); - } - - /** - * Mark a notification as read - */ - async markAsRead(id: string, userId: string): Promise { - const notification = await this.notificationRepository.findOne({ - where: { id, userId }, - }); - - if (!notification) { - throw new NotFoundException(`Notification with ID ${id} not found`); + /** + * Get all notifications for a user + */ + async findAllForUser(userId: string, options: { + isRead?: boolean; + limit?: number; + offset?: number; + } = {}): Promise<[ + Notification[], + number + ]> { + const query = this.notificationRepository + .createQueryBuilder('notification') + .where('notification.userId = :userId', { userId }); + if (options.isRead !== undefined) { + query.andWhere('notification.isRead = :isRead', { + isRead: options.isRead, + }); + } + query + .orderBy('notification.createdAt', 'DESC') + .take(options.limit || 20) + .skip(options.offset || 0); + return query.getManyAndCount(); } - - notification.isRead = true; - notification.readAt = new Date(); - return this.notificationRepository.save(notification); - } - - /** - * Mark all notifications as read for a user - */ - async markAllAsRead(userId: string): Promise { - await this.notificationRepository.update( - { userId, isRead: false }, - { isRead: true, readAt: new Date() }, - ); - } - - /** - * Delete a notification - */ - async remove(id: string, userId: string): Promise { - const result = await this.notificationRepository.softDelete({ id, userId }); - - if (result.affected === 0) { - throw new NotFoundException(`Notification with ID ${id} not found`); + /** + * Mark a notification as read + */ + async markAsRead(id: string, userId: string): Promise { + const notification = await this.notificationRepository.findOne({ + where: { id, userId }, + }); + if (!notification) { + throw new NotFoundException(`Notification with ID ${id} not found`); + } + notification.isRead = true; + notification.readAt = new Date(); + return this.notificationRepository.save(notification); } - } - - /** - * Update notification preferences - */ - async updatePreferences( - userId: string, - updateDto: Partial, - ): Promise { - return this.preferencesService.updatePreferences(userId, updateDto); - } - - /** - * Get user preferences - */ - async getPreferences(userId: string): Promise { - return this.preferencesService.getPreferences(userId); - } - - private shouldSendNotification( - type: NotificationType, - preferences: NotificationPreferences, - ): boolean { - switch (type) { - case NotificationType.EMAIL: - return preferences.emailEnabled; - case NotificationType.PUSH: - return preferences.pushEnabled; - case NotificationType.IN_APP: - return preferences.inAppEnabled; - case NotificationType.SMS: - return preferences.smsEnabled; - default: - return true; + /** + * Mark all notifications as read for a user + */ + async markAllAsRead(userId: string): Promise { + await this.notificationRepository.update({ userId, isRead: false }, { isRead: true, readAt: new Date() }); + } + /** + * Delete a notification + */ + async remove(id: string, userId: string): Promise { + const result = await this.notificationRepository.softDelete({ id, userId }); + if (result.affected === 0) { + throw new NotFoundException(`Notification with ID ${id} not found`); + } + } + /** + * Update notification preferences + */ + async updatePreferences(userId: string, updateDto: Partial): Promise { + return this.preferencesService.updatePreferences(userId, updateDto); + } + /** + * Get user preferences + */ + async getPreferences(userId: string): Promise { + return this.preferencesService.getPreferences(userId); + } + private shouldSendNotification(type: NotificationType, preferences: NotificationPreferences): boolean { + switch (type) { + case NotificationType.EMAIL: + return preferences.emailEnabled; + case NotificationType.PUSH: + return preferences.pushEnabled; + case NotificationType.IN_APP: + return preferences.inAppEnabled; + case NotificationType.SMS: + return preferences.smsEnabled; + default: + return true; + } + } + /** + * Event listener for system-wide notifications + */ + @OnEvent(APP_EVENTS.NOTIFICATION_SEND) + async handleSendNotification(payload: CreateNotificationDto): Promise { + await this.create(payload); + } + /** + * Event listener for specific templates + */ + @OnEvent(APP_EVENTS.NOTIFICATION_TEMPLATE_SEND) + async handleSendTemplateNotification(payload: { + userId: string; + templateType: string; + data: unknown; + type?: NotificationType; + }): Promise { + const template = this.templatesService.renderTemplate(payload.templateType, payload.data); + await this.create({ + userId: payload.userId, + title: template.title, + content: template.content, + type: payload.type || NotificationType.IN_APP, + priority: NotificationPriority.MEDIUM, + }); } - } - - /** - * Event listener for system-wide notifications - */ - @OnEvent(APP_EVENTS.NOTIFICATION_SEND) - async handleSendNotification(payload: CreateNotificationDto): Promise { - await this.create(payload); - } - - /** - * Event listener for specific templates - */ - @OnEvent(APP_EVENTS.NOTIFICATION_TEMPLATE_SEND) - async handleSendTemplateNotification(payload: { - userId: string; - templateType: string; - data: any; - type?: NotificationType; - }): Promise { - const template = this.templatesService.renderTemplate(payload.templateType, payload.data); - - await this.create({ - userId: payload.userId, - title: template.title, - content: template.content, - type: payload.type || NotificationType.IN_APP, - priority: NotificationPriority.MEDIUM, - }); - } } diff --git a/src/notifications/preferences/preferences.service.ts b/src/notifications/preferences/preferences.service.ts index 77fab88c..9a9c2b80 100644 --- a/src/notifications/preferences/preferences.service.ts +++ b/src/notifications/preferences/preferences.service.ts @@ -2,61 +2,45 @@ import { Injectable, Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { NotificationPreferences } from '../entities/notification-preferences.entity'; - @Injectable() export class PreferencesService { - private readonly logger = new Logger(PreferencesService.name); - - constructor( + private readonly logger = new Logger(PreferencesService.name); + constructor( @InjectRepository(NotificationPreferences) - private readonly preferencesRepository: Repository, - ) {} - - /** - * Get user preferences or create default if not exists - */ - async getPreferences(userId: string): Promise { - let preferences = await this.preferencesRepository.findOne({ where: { userId } }); - if (!preferences) { - this.logger.debug(`Creating default preferences for user ${userId}`); - preferences = this.preferencesRepository.create({ userId }); - preferences = await this.preferencesRepository.save(preferences); + private readonly preferencesRepository: Repository) { } + /** + * Get user preferences or create default if not exists + */ + async getPreferences(userId: string): Promise { + let preferences = await this.preferencesRepository.findOne({ where: { userId } }); + if (!preferences) { + this.logger.debug(`Creating default preferences for user ${userId}`); + preferences = this.preferencesRepository.create({ userId }); + preferences = await this.preferencesRepository.save(preferences); + } + return preferences; + } + /** + * Update user preferences + */ + async updatePreferences(userId: string, updateDto: Partial): Promise { + const preferences = await this.getPreferences(userId); + Object.assign(preferences, updateDto); + return this.preferencesRepository.save(preferences); + } + /** + * Check if a specific channel is enabled for a user + */ + async isChannelEnabled(userId: string, channel: 'emailEnabled' | 'pushEnabled' | 'inAppEnabled' | 'smsEnabled'): Promise { + const preferences = await this.getPreferences(userId); + return !!preferences[channel]; + } + /** + * Toggle a specific channel for a user + */ + async toggleChannel(userId: string, channel: 'emailEnabled' | 'pushEnabled' | 'inAppEnabled' | 'smsEnabled'): Promise { + const preferences = await this.getPreferences(userId); + preferences[channel] = !preferences[channel]; + await this.preferencesRepository.save(preferences); } - return preferences; - } - - /** - * Update user preferences - */ - async updatePreferences( - userId: string, - updateDto: Partial, - ): Promise { - const preferences = await this.getPreferences(userId); - Object.assign(preferences, updateDto); - return this.preferencesRepository.save(preferences); - } - - /** - * Check if a specific channel is enabled for a user - */ - async isChannelEnabled( - userId: string, - channel: 'emailEnabled' | 'pushEnabled' | 'inAppEnabled' | 'smsEnabled', - ): Promise { - const preferences = await this.getPreferences(userId); - return !!preferences[channel]; - } - - /** - * Toggle a specific channel for a user - */ - async toggleChannel( - userId: string, - channel: 'emailEnabled' | 'pushEnabled' | 'inAppEnabled' | 'smsEnabled', - ): Promise { - const preferences = await this.getPreferences(userId); - preferences[channel] = !preferences[channel]; - await this.preferencesRepository.save(preferences); - } } diff --git a/src/observability/anomaly/anomaly-detection.service.ts b/src/observability/anomaly/anomaly-detection.service.ts index 8cc56c1f..f7c66640 100644 --- a/src/observability/anomaly/anomaly-detection.service.ts +++ b/src/observability/anomaly/anomaly-detection.service.ts @@ -2,369 +2,301 @@ import { Injectable, Logger } from '@nestjs/common'; import { Cron, CronExpression } from '@nestjs/schedule'; import { MetricsAnalysisService } from '../metrics/metrics-analysis.service'; import { AnomalyDetectionResult } from '../interfaces/observability.interfaces'; - /** * Anomaly Detection Service * Automatically detects unusual patterns in metrics and logs */ @Injectable() export class AnomalyDetectionService { - private readonly logger = new Logger(AnomalyDetectionService.name); - private anomalies: AnomalyDetectionResult[] = []; - private readonly MAX_ANOMALIES = 1000; - - // Thresholds for anomaly detection - private readonly thresholds = { - errorRate: 5, // 5% error rate - responseTime: 5000, // 5 seconds - memoryUsage: 90, // 90% memory usage - cpuUsage: 85, // 85% CPU usage - queueBacklog: 1000, // 1000 jobs waiting - failureRate: 10, // 10% failure rate - }; - - constructor(private readonly metricsService: MetricsAnalysisService) {} - - /** - * Detect anomalies in a metric using statistical methods - */ - detectAnomalies(metricName: string, windowSize: number = 100): AnomalyDetectionResult[] { - const metrics = this.metricsService.getMetrics(metricName, windowSize); - - if (metrics.length < 10) { - return []; // Not enough data + private readonly logger = new Logger(AnomalyDetectionService.name); + private anomalies: AnomalyDetectionResult[] = []; + private readonly MAX_ANOMALIES = 1000; + // Thresholds for anomaly detection + private readonly thresholds = { + errorRate: 5, // 5% error rate + responseTime: 5000, // 5 seconds + memoryUsage: 90, // 90% memory usage + cpuUsage: 85, // 85% CPU usage + queueBacklog: 1000, // 1000 jobs waiting + failureRate: 10, // 10% failure rate + }; + constructor(private readonly metricsService: MetricsAnalysisService) { } + /** + * Detect anomalies in a metric using statistical methods + */ + detectAnomalies(metricName: string, windowSize: number = 100): AnomalyDetectionResult[] { + const metrics = this.metricsService.getMetrics(metricName, windowSize); + if (metrics.length < 10) { + return []; // Not enough data + } + const values = metrics.map((m) => m.value); + const mean = values.reduce((a, b) => a + b, 0) / values.length; + const variance = values.reduce((acc, val) => acc + Math.pow(val - mean, 2), 0) / values.length; + const stdDev = Math.sqrt(variance); + const anomalies: AnomalyDetectionResult[] = []; + // Z-score method: values beyond 3 standard deviations are anomalies + const threshold = 3; + metrics.forEach((metric) => { + const zScore = Math.abs((metric.value - mean) / stdDev); + if (zScore > threshold) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: zScore, + threshold, + metric: metricName, + timestamp: metric.timestamp, + details: `Value ${metric.value} is ${zScore.toFixed(2)} standard deviations from mean ${mean.toFixed(2)}`, + }; + anomalies.push(anomaly); + this.recordAnomaly(anomaly); + } + }); + return anomalies; } - - const values = metrics.map((m) => m.value); - const mean = values.reduce((a, b) => a + b, 0) / values.length; - const variance = values.reduce((acc, val) => acc + Math.pow(val - mean, 2), 0) / values.length; - const stdDev = Math.sqrt(variance); - - const anomalies: AnomalyDetectionResult[] = []; - - // Z-score method: values beyond 3 standard deviations are anomalies - const threshold = 3; - - metrics.forEach((metric) => { - const zScore = Math.abs((metric.value - mean) / stdDev); - - if (zScore > threshold) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: zScore, - threshold, - metric: metricName, - timestamp: metric.timestamp, - details: `Value ${metric.value} is ${zScore.toFixed(2)} standard deviations from mean ${mean.toFixed(2)}`, - }; - - anomalies.push(anomaly); - this.recordAnomaly(anomaly); - } - }); - - return anomalies; - } - - /** - * Detect anomalies using moving average - */ - detectAnomaliesMovingAverage( - metricName: string, - windowSize: number = 20, - threshold: number = 2, - ): AnomalyDetectionResult[] { - const metrics = this.metricsService.getMetrics(metricName, windowSize * 2); - - if (metrics.length < windowSize) { - return []; + /** + * Detect anomalies using moving average + */ + detectAnomaliesMovingAverage(metricName: string, windowSize: number = 20, threshold: number = 2): AnomalyDetectionResult[] { + const metrics = this.metricsService.getMetrics(metricName, windowSize * 2); + if (metrics.length < windowSize) { + return []; + } + const anomalies: AnomalyDetectionResult[] = []; + for (let i = windowSize; i < metrics.length; i++) { + const window = metrics.slice(i - windowSize, i); + const avg = window.reduce((a, b) => a + b.value, 0) / window.length; + const current = metrics[i].value; + const deviation = Math.abs(current - avg) / avg; + if (deviation > threshold) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: deviation, + threshold, + metric: metricName, + timestamp: metrics[i].timestamp, + details: `Value ${current} deviates ${(deviation * 100).toFixed(2)}% from moving average ${avg.toFixed(2)}`, + }; + anomalies.push(anomaly); + this.recordAnomaly(anomaly); + } + } + return anomalies; } - - const anomalies: AnomalyDetectionResult[] = []; - - for (let i = windowSize; i < metrics.length; i++) { - const window = metrics.slice(i - windowSize, i); - const avg = window.reduce((a, b) => a + b.value, 0) / window.length; - const current = metrics[i].value; - const deviation = Math.abs(current - avg) / avg; - - if (deviation > threshold) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: deviation, - threshold, - metric: metricName, - timestamp: metrics[i].timestamp, - details: `Value ${current} deviates ${(deviation * 100).toFixed(2)}% from moving average ${avg.toFixed(2)}`, - }; - - anomalies.push(anomaly); - this.recordAnomaly(anomaly); - } + /** + * Check for error rate anomalies + */ + checkErrorRateAnomaly(): AnomalyDetectionResult | null { + const stats = this.metricsService.getMetricStatistics('api.response_time'); + if (!stats) + return null; + // Calculate error rate from response codes + // This is a simplified version - in production, track actual error counts + const errorRate = 0; // Placeholder + if (errorRate > this.thresholds.errorRate) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: errorRate, + threshold: this.thresholds.errorRate, + metric: 'error_rate', + timestamp: new Date(), + details: `Error rate ${errorRate.toFixed(2)}% exceeds threshold ${this.thresholds.errorRate}%`, + }; + this.recordAnomaly(anomaly); + return anomaly; + } + return null; } - - return anomalies; - } - - /** - * Check for error rate anomalies - */ - checkErrorRateAnomaly(): AnomalyDetectionResult | null { - const stats = this.metricsService.getMetricStatistics('api.response_time'); - - if (!stats) return null; - - // Calculate error rate from response codes - // This is a simplified version - in production, track actual error counts - const errorRate = 0; // Placeholder - - if (errorRate > this.thresholds.errorRate) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: errorRate, - threshold: this.thresholds.errorRate, - metric: 'error_rate', - timestamp: new Date(), - details: `Error rate ${errorRate.toFixed(2)}% exceeds threshold ${this.thresholds.errorRate}%`, - }; - - this.recordAnomaly(anomaly); - return anomaly; + /** + * Check for response time anomalies + */ + checkResponseTimeAnomaly(): AnomalyDetectionResult | null { + const stats = this.metricsService.getMetricStatistics('api.response_time'); + if (!stats) + return null; + if (stats.p95 > this.thresholds.responseTime) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: stats.p95, + threshold: this.thresholds.responseTime, + metric: 'api.response_time', + timestamp: new Date(), + details: `P95 response time ${stats.p95.toFixed(2)}ms exceeds threshold ${this.thresholds.responseTime}ms`, + }; + this.recordAnomaly(anomaly); + return anomaly; + } + return null; } - - return null; - } - - /** - * Check for response time anomalies - */ - checkResponseTimeAnomaly(): AnomalyDetectionResult | null { - const stats = this.metricsService.getMetricStatistics('api.response_time'); - - if (!stats) return null; - - if (stats.p95 > this.thresholds.responseTime) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: stats.p95, - threshold: this.thresholds.responseTime, - metric: 'api.response_time', - timestamp: new Date(), - details: `P95 response time ${stats.p95.toFixed(2)}ms exceeds threshold ${this.thresholds.responseTime}ms`, - }; - - this.recordAnomaly(anomaly); - return anomaly; + /** + * Check for memory usage anomalies + */ + checkMemoryAnomaly(): AnomalyDetectionResult | null { + const stats = this.metricsService.getMetricStatistics('system.memory.heap_used'); + if (!stats) + return null; + const memoryUsagePercent = (stats.avg / (1024 * 1024 * 1024)) * 100; // Convert to GB and percentage + if (memoryUsagePercent > this.thresholds.memoryUsage) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: memoryUsagePercent, + threshold: this.thresholds.memoryUsage, + metric: 'system.memory.heap_used', + timestamp: new Date(), + details: `Memory usage ${memoryUsagePercent.toFixed(2)}% exceeds threshold ${this.thresholds.memoryUsage}%`, + }; + this.recordAnomaly(anomaly); + return anomaly; + } + return null; } - - return null; - } - - /** - * Check for memory usage anomalies - */ - checkMemoryAnomaly(): AnomalyDetectionResult | null { - const stats = this.metricsService.getMetricStatistics('system.memory.heap_used'); - - if (!stats) return null; - - const memoryUsagePercent = (stats.avg / (1024 * 1024 * 1024)) * 100; // Convert to GB and percentage - - if (memoryUsagePercent > this.thresholds.memoryUsage) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: memoryUsagePercent, - threshold: this.thresholds.memoryUsage, - metric: 'system.memory.heap_used', - timestamp: new Date(), - details: `Memory usage ${memoryUsagePercent.toFixed(2)}% exceeds threshold ${this.thresholds.memoryUsage}%`, - }; - - this.recordAnomaly(anomaly); - return anomaly; + /** + * Check for sudden spikes in metrics + */ + detectSuddenSpike(metricName: string, spikeThreshold: number = 3): AnomalyDetectionResult | null { + const metrics = this.metricsService.getMetrics(metricName, 10); + if (metrics.length < 2) + return null; + const latest = metrics[metrics.length - 1].value; + const previous = metrics[metrics.length - 2].value; + if (previous === 0) + return null; + const change = (latest - previous) / previous; + if (Math.abs(change) > spikeThreshold) { + const anomaly: AnomalyDetectionResult = { + isAnomaly: true, + score: Math.abs(change), + threshold: spikeThreshold, + metric: metricName, + timestamp: new Date(), + details: `Sudden ${change > 0 ? 'spike' : 'drop'} of ${(Math.abs(change) * 100).toFixed(2)}% detected`, + }; + this.recordAnomaly(anomaly); + return anomaly; + } + return null; } - - return null; - } - - /** - * Check for sudden spikes in metrics - */ - detectSuddenSpike(metricName: string, spikeThreshold: number = 3): AnomalyDetectionResult | null { - const metrics = this.metricsService.getMetrics(metricName, 10); - - if (metrics.length < 2) return null; - - const latest = metrics[metrics.length - 1].value; - const previous = metrics[metrics.length - 2].value; - - if (previous === 0) return null; - - const change = (latest - previous) / previous; - - if (Math.abs(change) > spikeThreshold) { - const anomaly: AnomalyDetectionResult = { - isAnomaly: true, - score: Math.abs(change), - threshold: spikeThreshold, - metric: metricName, - timestamp: new Date(), - details: `Sudden ${change > 0 ? 'spike' : 'drop'} of ${(Math.abs(change) * 100).toFixed(2)}% detected`, - }; - - this.recordAnomaly(anomaly); - return anomaly; + /** + * Record an anomaly + */ + private recordAnomaly(anomaly: AnomalyDetectionResult): void { + this.anomalies.push(anomaly); + // Maintain size limit + if (this.anomalies.length > this.MAX_ANOMALIES) { + this.anomalies.shift(); + } + this.logger.warn(`Anomaly detected: ${anomaly.details}`); + // In production, send alerts + this.sendAlert(anomaly); } - - return null; - } - - /** - * Record an anomaly - */ - private recordAnomaly(anomaly: AnomalyDetectionResult): void { - this.anomalies.push(anomaly); - - // Maintain size limit - if (this.anomalies.length > this.MAX_ANOMALIES) { - this.anomalies.shift(); + /** + * Get all detected anomalies + */ + getAnomalies(limit?: number): AnomalyDetectionResult[] { + return limit ? this.anomalies.slice(-limit) : this.anomalies; } - - this.logger.warn(`Anomaly detected: ${anomaly.details}`); - - // In production, send alerts - this.sendAlert(anomaly); - } - - /** - * Get all detected anomalies - */ - getAnomalies(limit?: number): AnomalyDetectionResult[] { - return limit ? this.anomalies.slice(-limit) : this.anomalies; - } - - /** - * Get anomalies by metric - */ - getAnomaliesByMetric(metricName: string): AnomalyDetectionResult[] { - return this.anomalies.filter((a) => a.metric === metricName); - } - - /** - * Get recent anomalies - */ - getRecentAnomalies(minutes: number = 60): AnomalyDetectionResult[] { - const cutoff = new Date(Date.now() - minutes * 60 * 1000); - return this.anomalies.filter((a) => a.timestamp > cutoff); - } - - /** - * Clear old anomalies - */ - clearOldAnomalies(olderThan: Date): number { - const initialLength = this.anomalies.length; - this.anomalies = this.anomalies.filter((a) => a.timestamp > olderThan); - const cleared = initialLength - this.anomalies.length; - this.logger.log(`Cleared ${cleared} old anomalies`); - return cleared; - } - - /** - * Periodic anomaly detection (runs every 5 minutes) - */ - @Cron(CronExpression.EVERY_5_MINUTES) - async periodicAnomalyDetection(): Promise { - this.logger.debug('Running periodic anomaly detection'); - - try { - // Check various metrics for anomalies - this.checkErrorRateAnomaly(); - this.checkResponseTimeAnomaly(); - this.checkMemoryAnomaly(); - - // Check for spikes in key metrics - const keyMetrics = [ - 'api.response_time', - 'db.query_time', - 'queue.processing_time', - 'user.signups', - 'payment.amount', - ]; - - keyMetrics.forEach((metric) => { - this.detectSuddenSpike(metric); - }); - - // Statistical anomaly detection - keyMetrics.forEach((metric) => { - this.detectAnomalies(metric); - }); - } catch (error) { - this.logger.error('Error during periodic anomaly detection:', error); + /** + * Get anomalies by metric + */ + getAnomaliesByMetric(metricName: string): AnomalyDetectionResult[] { + return this.anomalies.filter((a) => a.metric === metricName); } - } - - /** - * Send alert for anomaly - */ - private async sendAlert(anomaly: AnomalyDetectionResult): Promise { - // In production, integrate with alerting systems: - // - PagerDuty - // - Slack - // - Email - // - SMS - // - Custom webhooks - - // For now, just log - this.logger.warn(`ALERT: ${anomaly.details}`); - } - - /** - * Get anomaly statistics - */ - getAnomalyStatistics() { - const byMetric: Record = {}; - - this.anomalies.forEach((anomaly) => { - byMetric[anomaly.metric] = (byMetric[anomaly.metric] || 0) + 1; - }); - - const recentAnomalies = this.getRecentAnomalies(60); - - return { - total: this.anomalies.length, - recent: recentAnomalies.length, - byMetric, - avgScore: - this.anomalies.length > 0 - ? this.anomalies.reduce((sum, a) => sum + a.score, 0) / this.anomalies.length - : 0, - }; - } - - /** - * Check system health based on anomalies - */ - getSystemHealth(): { - status: 'healthy' | 'warning' | 'critical'; - issues: string[]; - } { - const recentAnomalies = this.getRecentAnomalies(15); // Last 15 minutes - - if (recentAnomalies.length === 0) { - return { status: 'healthy', issues: [] }; + /** + * Get recent anomalies + */ + getRecentAnomalies(minutes: number = 60): AnomalyDetectionResult[] { + const cutoff = new Date(Date.now() - minutes * 60 * 1000); + return this.anomalies.filter((a) => a.timestamp > cutoff); } - - const issues = recentAnomalies.map((a) => a.details); - - if (recentAnomalies.length > 10) { - return { status: 'critical', issues }; + /** + * Clear old anomalies + */ + clearOldAnomalies(olderThan: Date): number { + const initialLength = this.anomalies.length; + this.anomalies = this.anomalies.filter((a) => a.timestamp > olderThan); + const cleared = initialLength - this.anomalies.length; + this.logger.log(`Cleared ${cleared} old anomalies`); + return cleared; + } + /** + * Periodic anomaly detection (runs every 5 minutes) + */ + @Cron(CronExpression.EVERY_5_MINUTES) + async periodicAnomalyDetection(): Promise { + this.logger.debug('Running periodic anomaly detection'); + try { + // Check various metrics for anomalies + this.checkErrorRateAnomaly(); + this.checkResponseTimeAnomaly(); + this.checkMemoryAnomaly(); + // Check for spikes in key metrics + const keyMetrics = [ + 'api.response_time', + 'db.query_time', + 'queue.processing_time', + 'user.signups', + 'payment.amount', + ]; + keyMetrics.forEach((metric) => { + this.detectSuddenSpike(metric); + }); + // Statistical anomaly detection + keyMetrics.forEach((metric) => { + this.detectAnomalies(metric); + }); + } + catch (error) { + this.logger.error('Error during periodic anomaly detection:', error); + } + } + /** + * Send alert for anomaly + */ + private async sendAlert(anomaly: AnomalyDetectionResult): Promise { + // In production, integrate with alerting systems: + // - PagerDuty + // - Slack + // - Email + // - SMS + // - Custom webhooks + // For now, just log + this.logger.warn(`ALERT: ${anomaly.details}`); + } + /** + * Get anomaly statistics + */ + getAnomalyStatistics() { + const byMetric: Record = {}; + this.anomalies.forEach((anomaly) => { + byMetric[anomaly.metric] = (byMetric[anomaly.metric] || 0) + 1; + }); + const recentAnomalies = this.getRecentAnomalies(60); + return { + total: this.anomalies.length, + recent: recentAnomalies.length, + byMetric, + avgScore: this.anomalies.length > 0 + ? this.anomalies.reduce((sum, a) => sum + a.score, 0) / this.anomalies.length + : 0, + }; } - - if (recentAnomalies.length > 5) { - return { status: 'warning', issues }; + /** + * Check system health based on anomalies + */ + getSystemHealth(): { + status: 'healthy' | 'warning' | 'critical'; + issues: string[]; + } { + const recentAnomalies = this.getRecentAnomalies(15); // Last 15 minutes + if (recentAnomalies.length === 0) { + return { status: 'healthy', issues: [] }; + } + const issues = recentAnomalies.map((a) => a.details); + if (recentAnomalies.length > 10) { + return { status: 'critical', issues }; + } + if (recentAnomalies.length > 5) { + return { status: 'warning', issues }; + } + return { status: 'healthy', issues }; } - - return { status: 'healthy', issues }; - } } diff --git a/src/observability/interfaces/observability.interfaces.ts b/src/observability/interfaces/observability.interfaces.ts index 5b6edfd0..9dc72430 100644 --- a/src/observability/interfaces/observability.interfaces.ts +++ b/src/observability/interfaces/observability.interfaces.ts @@ -1,104 +1,93 @@ export interface LogContext { - correlationId: string; - traceId?: string; - spanId?: string; - userId?: string; - requestId?: string; - service: string; - environment: string; - timestamp: Date; - metadata?: Record; + correlationId: string; + traceId?: string; + spanId?: string; + userId?: string; + requestId?: string; + service: string; + environment: string; + timestamp: Date; + metadata?: Record; } - export interface StructuredLog { - level: LogLevel; - message: string; - context: LogContext; - error?: ErrorDetails; - duration?: number; - tags?: string[]; + level: LogLevel; + message: string; + context: LogContext; + error?: ErrorDetails; + duration?: number; + tags?: string[]; } - export interface ErrorDetails { - name: string; - message: string; - stack?: string; - code?: string; - statusCode?: number; + name: string; + message: string; + stack?: string; + code?: string; + statusCode?: number; } - export enum LogLevel { - DEBUG = 'debug', - INFO = 'info', - WARN = 'warn', - ERROR = 'error', - FATAL = 'fatal', + DEBUG = 'debug', + INFO = 'info', + WARN = 'warn', + ERROR = 'error', + FATAL = 'fatal' } - export interface MetricData { - name: string; - value: number; - type: MetricType; - tags?: Record; - timestamp: Date; + name: string; + value: number; + type: MetricType; + tags?: Record; + timestamp: Date; } - export enum MetricType { - COUNTER = 'counter', - GAUGE = 'gauge', - HISTOGRAM = 'histogram', - SUMMARY = 'summary', + COUNTER = 'counter', + GAUGE = 'gauge', + HISTOGRAM = 'histogram', + SUMMARY = 'summary' } - export interface TraceSpan { - traceId: string; - spanId: string; - parentSpanId?: string; - name: string; - startTime: Date; - endTime?: Date; - duration?: number; - status: SpanStatus; - attributes: Record; - events: SpanEvent[]; + traceId: string; + spanId: string; + parentSpanId?: string; + name: string; + startTime: Date; + endTime?: Date; + duration?: number; + status: SpanStatus; + attributes: Record; + events: SpanEvent[]; } - export enum SpanStatus { - OK = 'ok', - ERROR = 'error', - UNSET = 'unset', + OK = 'ok', + ERROR = 'error', + UNSET = 'unset' } - export interface SpanEvent { - name: string; - timestamp: Date; - attributes?: Record; + name: string; + timestamp: Date; + attributes?: Record; } - export interface AnomalyDetectionResult { - isAnomaly: boolean; - score: number; - threshold: number; - metric: string; - timestamp: Date; - details?: string; + isAnomaly: boolean; + score: number; + threshold: number; + metric: string; + timestamp: Date; + details?: string; } - export interface LogQuery { - level?: LogLevel; - service?: string; - correlationId?: string; - userId?: string; - startTime?: Date; - endTime?: Date; - search?: string; - limit?: number; - offset?: number; + level?: LogLevel; + service?: string; + correlationId?: string; + userId?: string; + startTime?: Date; + endTime?: Date; + search?: string; + limit?: number; + offset?: number; } - export interface LogSearchResult { - logs: StructuredLog[]; - total: number; - page: number; - pageSize: number; + logs: StructuredLog[]; + total: number; + page: number; + pageSize: number; } diff --git a/src/observability/logging/log-aggregation.service.ts b/src/observability/logging/log-aggregation.service.ts index e10e0112..5ff01ae7 100644 --- a/src/observability/logging/log-aggregation.service.ts +++ b/src/observability/logging/log-aggregation.service.ts @@ -1,253 +1,207 @@ import { Injectable, Logger } from '@nestjs/common'; -import { - StructuredLog, - LogQuery, - LogSearchResult, - LogLevel, -} from '../interfaces/observability.interfaces'; - +import { StructuredLog, LogQuery, LogSearchResult, LogLevel, } from '../interfaces/observability.interfaces'; /** * Log Aggregation Service * Centralized log storage and search functionality */ @Injectable() export class LogAggregationService { - private readonly logger = new Logger(LogAggregationService.name); - private logs: StructuredLog[] = []; - private readonly MAX_LOGS = 10000; // In-memory limit - - /** - * Store a log entry - */ - async storeLogs(log: StructuredLog): Promise { - this.logs.push(log); - - // Maintain size limit (FIFO) - if (this.logs.length > this.MAX_LOGS) { - this.logs.shift(); + private readonly logger = new Logger(LogAggregationService.name); + private logs: StructuredLog[] = []; + private readonly MAX_LOGS = 10000; // In-memory limit + /** + * Store a log entry + */ + async storeLogs(log: StructuredLog): Promise { + this.logs.push(log); + // Maintain size limit (FIFO) + if (this.logs.length > this.MAX_LOGS) { + this.logs.shift(); + } + // In production, send to external log aggregation service + // Examples: Elasticsearch, CloudWatch, Datadog, etc. + await this.sendToExternalService(log); } - - // In production, send to external log aggregation service - // Examples: Elasticsearch, CloudWatch, Datadog, etc. - await this.sendToExternalService(log); - } - - /** - * Search logs with filters - */ - async searchLogs(query: LogQuery): Promise { - let filteredLogs = [...this.logs]; - - // Apply filters - if (query.level) { - filteredLogs = filteredLogs.filter((log) => log.level === query.level); + /** + * Search logs with filters + */ + async searchLogs(query: LogQuery): Promise { + let filteredLogs = [...this.logs]; + // Apply filters + if (query.level) { + filteredLogs = filteredLogs.filter((log) => log.level === query.level); + } + if (query.service) { + filteredLogs = filteredLogs.filter((log) => log.context.service === query.service); + } + if (query.correlationId) { + filteredLogs = filteredLogs.filter((log) => log.context.correlationId === query.correlationId); + } + if (query.userId) { + filteredLogs = filteredLogs.filter((log) => log.context.userId === query.userId); + } + if (query.startTime) { + const startTime = query.startTime; + filteredLogs = filteredLogs.filter((log) => log.context.timestamp >= startTime); + } + if (query.endTime) { + const endTime = query.endTime; + filteredLogs = filteredLogs.filter((log) => log.context.timestamp <= endTime); + } + if (query.search) { + const searchLower = query.search.toLowerCase(); + filteredLogs = filteredLogs.filter((log) => log.message.toLowerCase().includes(searchLower) || + JSON.stringify(log.context.metadata).toLowerCase().includes(searchLower)); + } + // Sort by timestamp (newest first) + filteredLogs.sort((a, b) => b.context.timestamp.getTime() - a.context.timestamp.getTime()); + // Pagination + const limit = query.limit || 50; + const offset = query.offset || 0; + const paginatedLogs = filteredLogs.slice(offset, offset + limit); + return { + logs: paginatedLogs, + total: filteredLogs.length, + page: Math.floor(offset / limit) + 1, + pageSize: limit, + }; } - - if (query.service) { - filteredLogs = filteredLogs.filter((log) => log.context.service === query.service); + /** + * Get logs by correlation ID (trace all related logs) + */ + async getLogsByCorrelationId(correlationId: string): Promise { + return this.logs.filter((log) => log.context.correlationId === correlationId); } - - if (query.correlationId) { - filteredLogs = filteredLogs.filter( - (log) => log.context.correlationId === query.correlationId, - ); + /** + * Get logs by trace ID + */ + async getLogsByTraceId(traceId: string): Promise { + return this.logs.filter((log) => log.context.traceId === traceId); } - - if (query.userId) { - filteredLogs = filteredLogs.filter((log) => log.context.userId === query.userId); + /** + * Get error logs + */ + async getErrorLogs(limit: number = 100): Promise { + return this.logs + .filter((log) => log.level === LogLevel.ERROR || log.level === LogLevel.FATAL) + .slice(-limit) + .reverse(); } - - if (query.startTime) { - const startTime = query.startTime; - filteredLogs = filteredLogs.filter((log) => log.context.timestamp >= startTime); + /** + * Get logs by user + */ + async getLogsByUser(userId: string, limit: number = 100): Promise { + return this.logs + .filter((log) => log.context.userId === userId) + .slice(-limit) + .reverse(); } - - if (query.endTime) { - const endTime = query.endTime; - filteredLogs = filteredLogs.filter((log) => log.context.timestamp <= endTime); + /** + * Get log statistics + */ + async getLogStatistics(timeRange?: { + start: Date; + end: Date; + }) { + let logsToAnalyze = this.logs; + if (timeRange) { + logsToAnalyze = this.logs.filter((log) => log.context.timestamp >= timeRange.start && log.context.timestamp <= timeRange.end); + } + const stats = { + total: logsToAnalyze.length, + byLevel: { + debug: 0, + info: 0, + warn: 0, + error: 0, + fatal: 0, + }, + byService: {} as Record, + errorRate: 0, + avgDuration: 0, + }; + let totalDuration = 0; + let durationCount = 0; + logsToAnalyze.forEach((log) => { + // Count by level + stats.byLevel[log.level]++; + // Count by service + const service = log.context.service; + stats.byService[service] = (stats.byService[service] || 0) + 1; + // Calculate duration stats + if (log.duration) { + totalDuration += log.duration; + durationCount++; + } + }); + // Calculate error rate + const errorCount = stats.byLevel.error + stats.byLevel.fatal; + stats.errorRate = stats.total > 0 ? (errorCount / stats.total) * 100 : 0; + // Calculate average duration + stats.avgDuration = durationCount > 0 ? totalDuration / durationCount : 0; + return stats; } - - if (query.search) { - const searchLower = query.search.toLowerCase(); - filteredLogs = filteredLogs.filter( - (log) => - log.message.toLowerCase().includes(searchLower) || - JSON.stringify(log.context.metadata).toLowerCase().includes(searchLower), - ); + /** + * Clear old logs + */ + async clearOldLogs(olderThan: Date): Promise { + const initialCount = this.logs.length; + this.logs = this.logs.filter((log) => log.context.timestamp > olderThan); + const removed = initialCount - this.logs.length; + this.logger.log(`Cleared ${removed} old logs`); + return removed; } - - // Sort by timestamp (newest first) - filteredLogs.sort((a, b) => b.context.timestamp.getTime() - a.context.timestamp.getTime()); - - // Pagination - const limit = query.limit || 50; - const offset = query.offset || 0; - const paginatedLogs = filteredLogs.slice(offset, offset + limit); - - return { - logs: paginatedLogs, - total: filteredLogs.length, - page: Math.floor(offset / limit) + 1, - pageSize: limit, - }; - } - - /** - * Get logs by correlation ID (trace all related logs) - */ - async getLogsByCorrelationId(correlationId: string): Promise { - return this.logs.filter((log) => log.context.correlationId === correlationId); - } - - /** - * Get logs by trace ID - */ - async getLogsByTraceId(traceId: string): Promise { - return this.logs.filter((log) => log.context.traceId === traceId); - } - - /** - * Get error logs - */ - async getErrorLogs(limit: number = 100): Promise { - return this.logs - .filter((log) => log.level === LogLevel.ERROR || log.level === LogLevel.FATAL) - .slice(-limit) - .reverse(); - } - - /** - * Get logs by user - */ - async getLogsByUser(userId: string, limit: number = 100): Promise { - return this.logs - .filter((log) => log.context.userId === userId) - .slice(-limit) - .reverse(); - } - - /** - * Get log statistics - */ - async getLogStatistics(timeRange?: { start: Date; end: Date }) { - let logsToAnalyze = this.logs; - - if (timeRange) { - logsToAnalyze = this.logs.filter( - (log) => log.context.timestamp >= timeRange.start && log.context.timestamp <= timeRange.end, - ); + /** + * Export logs for analysis + */ + async exportLogs(query: LogQuery): Promise { + const result = await this.searchLogs(query); + return JSON.stringify(result.logs, null, 2); } - - const stats = { - total: logsToAnalyze.length, - byLevel: { - debug: 0, - info: 0, - warn: 0, - error: 0, - fatal: 0, - }, - byService: {} as Record, - errorRate: 0, - avgDuration: 0, - }; - - let totalDuration = 0; - let durationCount = 0; - - logsToAnalyze.forEach((log) => { - // Count by level - stats.byLevel[log.level]++; - - // Count by service - const service = log.context.service; - stats.byService[service] = (stats.byService[service] || 0) + 1; - - // Calculate duration stats - if (log.duration) { - totalDuration += log.duration; - durationCount++; - } - }); - - // Calculate error rate - const errorCount = stats.byLevel.error + stats.byLevel.fatal; - stats.errorRate = stats.total > 0 ? (errorCount / stats.total) * 100 : 0; - - // Calculate average duration - stats.avgDuration = durationCount > 0 ? totalDuration / durationCount : 0; - - return stats; - } - - /** - * Clear old logs - */ - async clearOldLogs(olderThan: Date): Promise { - const initialCount = this.logs.length; - this.logs = this.logs.filter((log) => log.context.timestamp > olderThan); - const removed = initialCount - this.logs.length; - this.logger.log(`Cleared ${removed} old logs`); - return removed; - } - - /** - * Export logs for analysis - */ - async exportLogs(query: LogQuery): Promise { - const result = await this.searchLogs(query); - return JSON.stringify(result.logs, null, 2); - } - - /** - * Send logs to external service (placeholder) - */ - private async sendToExternalService(_log: StructuredLog): Promise { - // In production, implement integration with: - // - Elasticsearch - // - AWS CloudWatch - // - Datadog - // - Splunk - // - Grafana Loki - // etc. - - // Example: Elasticsearch - // await this.elasticsearchClient.index({ - // index: 'logs', - // body: _log, - // }); - - // For now, just log to console in development - if (process.env.NODE_ENV === 'development') { - // Already logged by StructuredLoggerService + /** + * Send logs to external service (placeholder) + */ + private async sendToExternalService(_log: StructuredLog): Promise { + // In production, implement integration with: + // - Elasticsearch + // - AWS CloudWatch + // - Datadog + // - Splunk + // - Grafana Loki + // etc. + // Example: Elasticsearch + // await this.elasticsearchClient.index({ + // index: 'logs', + // body: _log, + // }); + // For now, just log to console in development + if (process.env.NODE_ENV === 'development') { + // Already logged by StructuredLoggerService + } + } + /** + * Get recent logs + */ + async getRecentLogs(limit: number = 100): Promise { + return this.logs.slice(-limit).reverse(); + } + /** + * Count logs by criteria + */ + async countLogs(query: Partial): Promise { + const result = await this.searchLogs(query as LogQuery); + return result.total; + } + /** + * Get unique correlation IDs + */ + async getUniqueCorrelationIds(limit: number = 100): Promise { + const correlationIds = new Set(); + this.logs.slice(-limit * 10).forEach((log) => { + if (log.context.correlationId) { + correlationIds.add(log.context.correlationId); + } + }); + return Array.from(correlationIds).slice(0, limit); } - } - - /** - * Get recent logs - */ - async getRecentLogs(limit: number = 100): Promise { - return this.logs.slice(-limit).reverse(); - } - - /** - * Count logs by criteria - */ - async countLogs(query: Partial): Promise { - const result = await this.searchLogs(query as LogQuery); - return result.total; - } - - /** - * Get unique correlation IDs - */ - async getUniqueCorrelationIds(limit: number = 100): Promise { - const correlationIds = new Set(); - this.logs.slice(-limit * 10).forEach((log) => { - if (log.context.correlationId) { - correlationIds.add(log.context.correlationId); - } - }); - return Array.from(correlationIds).slice(0, limit); - } } diff --git a/src/observability/logging/structured-logger.service.ts b/src/observability/logging/structured-logger.service.ts index 01f7c581..8f91f32c 100644 --- a/src/observability/logging/structured-logger.service.ts +++ b/src/observability/logging/structured-logger.service.ts @@ -1,259 +1,212 @@ import { Injectable, Logger, LoggerService, Scope } from '@nestjs/common'; import { v4 as uuidv4 } from 'uuid'; -import { - LogContext, - StructuredLog, - LogLevel, - ErrorDetails, -} from '../interfaces/observability.interfaces'; - +import { LogContext, StructuredLog, LogLevel, ErrorDetails, } from '../interfaces/observability.interfaces'; /** * Structured Logger Service * Provides structured logging with correlation IDs and context */ @Injectable({ scope: Scope.TRANSIENT }) export class StructuredLoggerService implements LoggerService { - private readonly nestLogger = new Logger(StructuredLoggerService.name); - private context: Partial = {}; - private serviceName: string = 'teachlink'; - - constructor() { - this.context = { - service: this.serviceName, - environment: process.env.NODE_ENV || 'development', - }; - } - - /** - * Set context for all subsequent logs - */ - setContext(context: Partial): void { - this.context = { ...this.context, ...context }; - } - - /** - * Set correlation ID for request tracking - */ - setCorrelationId(correlationId: string): void { - this.context.correlationId = correlationId; - } - - /** - * Set trace information - */ - setTraceInfo(traceId: string, spanId: string): void { - this.context.traceId = traceId; - this.context.spanId = spanId; - } - - /** - * Set user ID for user-specific logging - */ - setUserId(userId: string): void { - this.context.userId = userId; - } - - /** - * Log debug message - */ - debug(message: string, metadata?: Record): void { - this.log(LogLevel.DEBUG, message, metadata); - } - - /** - * Log info message - */ - log(message: string, metadata?: Record): void; - log(level: LogLevel, message: string, metadata?: Record): void; - log( - messageOrLevel: string | LogLevel, - messageOrMetadata?: string | Record, - metadata?: Record, - ): void { - let level: LogLevel; - let message: string; - let meta: Record | undefined; - - if (typeof messageOrLevel === 'string' && !messageOrMetadata) { - // log(message) - level = LogLevel.INFO; - message = messageOrLevel; - meta = undefined; - } else if (typeof messageOrLevel === 'string' && typeof messageOrMetadata === 'object') { - // log(message, metadata) - level = LogLevel.INFO; - message = messageOrLevel; - meta = messageOrMetadata; - } else { - // log(level, message, metadata) - level = messageOrLevel as LogLevel; - message = messageOrMetadata as string; - meta = metadata; + private readonly nestLogger = new Logger(StructuredLoggerService.name); + private context: Partial = {}; + private serviceName: string = 'teachlink'; + constructor() { + this.context = { + service: this.serviceName, + environment: process.env.NODE_ENV || 'development', + }; } - - this.writeLog(level, message, meta); - } - - /** - * Log warning message - */ - warn(message: string, metadata?: Record): void { - this.log(LogLevel.WARN, message, metadata); - } - - /** - * Log error message - */ - error(message: string, trace?: string, metadata?: Record): void; - error(message: string, error?: Error, metadata?: Record): void; - error(message: string, traceOrError?: string | Error, metadata?: Record): void { - let errorDetails: ErrorDetails | undefined; - - if (traceOrError instanceof Error) { - errorDetails = { - name: traceOrError.name, - message: traceOrError.message, - stack: traceOrError.stack, - }; - } else if (typeof traceOrError === 'string') { - errorDetails = { - name: 'Error', - message, - stack: traceOrError, - }; + /** + * Set context for all subsequent logs + */ + setContext(context: Partial): void { + this.context = { ...this.context, ...context }; } - - this.writeLog(LogLevel.ERROR, message, metadata, errorDetails); - } - - /** - * Log fatal error - */ - fatal(message: string, error?: Error, metadata?: Record): void { - const errorDetails: ErrorDetails | undefined = error - ? { - name: error.name, - message: error.message, - stack: error.stack, + /** + * Set correlation ID for request tracking + */ + setCorrelationId(correlationId: string): void { + this.context.correlationId = correlationId; + } + /** + * Set trace information + */ + setTraceInfo(traceId: string, spanId: string): void { + this.context.traceId = traceId; + this.context.spanId = spanId; + } + /** + * Set user ID for user-specific logging + */ + setUserId(userId: string): void { + this.context.userId = userId; + } + /** + * Log debug message + */ + debug(message: string, metadata?: Record): void { + this.log(LogLevel.DEBUG, message, metadata); + } + /** + * Log info message + */ + log(message: string, metadata?: Record): void; + log(level: LogLevel, message: string, metadata?: Record): void; + log(messageOrLevel: string | LogLevel, messageOrMetadata?: string | Record, metadata?: Record): void { + let level: LogLevel; + let message: string; + let meta: Record | undefined; + if (typeof messageOrLevel === 'string' && !messageOrMetadata) { + // log(message) + level = LogLevel.INFO; + message = messageOrLevel; + meta = undefined; + } + else if (typeof messageOrLevel === 'string' && typeof messageOrMetadata === 'object') { + // log(message, metadata) + level = LogLevel.INFO; + message = messageOrLevel; + meta = messageOrMetadata; + } + else { + // log(level, message, metadata) + level = messageOrLevel as LogLevel; + message = messageOrMetadata as string; + meta = metadata; + } + this.writeLog(level, message, meta); + } + /** + * Log warning message + */ + warn(message: string, metadata?: Record): void { + this.log(LogLevel.WARN, message, metadata); + } + /** + * Log error message + */ + error(message: string, trace?: string, metadata?: Record): void; + error(message: string, error?: Error, metadata?: Record): void; + error(message: string, traceOrError?: string | Error, metadata?: Record): void { + let errorDetails: ErrorDetails | undefined; + if (traceOrError instanceof Error) { + errorDetails = { + name: traceOrError.name, + message: traceOrError.message, + stack: traceOrError.stack, + }; } - : undefined; - - this.writeLog(LogLevel.FATAL, message, metadata, errorDetails); - } - - /** - * Log with timing information - */ - logWithTiming( - level: LogLevel, - message: string, - startTime: Date, - metadata?: Record, - ): void { - const duration = Date.now() - startTime.getTime(); - this.writeLog(level, message, { ...metadata, duration }); - } - - /** - * Write structured log - */ - private writeLog( - level: LogLevel, - message: string, - metadata?: Record, - error?: ErrorDetails, - ): void { - const logContext: LogContext = { - correlationId: this.context.correlationId || uuidv4(), - traceId: this.context.traceId, - spanId: this.context.spanId, - userId: this.context.userId, - requestId: this.context.requestId, - service: this.context.service || this.serviceName, - environment: this.context.environment || 'development', - timestamp: new Date(), - metadata, - }; - - const structuredLog: StructuredLog = { - level, - message, - context: logContext, - error, - }; - - // Output as JSON for log aggregation systems - const logOutput = JSON.stringify(structuredLog); - - // Route to NestJS Logger based on level — avoids raw console statements - switch (level) { - case LogLevel.DEBUG: - this.nestLogger.debug(logOutput); - break; - case LogLevel.INFO: - this.nestLogger.log(logOutput); - break; - case LogLevel.WARN: - this.nestLogger.warn(logOutput); - break; - case LogLevel.ERROR: - this.nestLogger.error(logOutput); - break; - case LogLevel.FATAL: - this.nestLogger.fatal(logOutput); - break; + else if (typeof traceOrError === 'string') { + errorDetails = { + name: 'Error', + message, + stack: traceOrError, + }; + } + this.writeLog(LogLevel.ERROR, message, metadata, errorDetails); + } + /** + * Log fatal error + */ + fatal(message: string, error?: Error, metadata?: Record): void { + const errorDetails: ErrorDetails | undefined = error + ? { + name: error.name, + message: error.message, + stack: error.stack, + } + : undefined; + this.writeLog(LogLevel.FATAL, message, metadata, errorDetails); + } + /** + * Log with timing information + */ + logWithTiming(level: LogLevel, message: string, startTime: Date, metadata?: Record): void { + const duration = Date.now() - startTime.getTime(); + this.writeLog(level, message, { ...metadata, duration }); + } + /** + * Write structured log + */ + private writeLog(level: LogLevel, message: string, metadata?: Record, error?: ErrorDetails): void { + const logContext: LogContext = { + correlationId: this.context.correlationId || uuidv4(), + traceId: this.context.traceId, + spanId: this.context.spanId, + userId: this.context.userId, + requestId: this.context.requestId, + service: this.context.service || this.serviceName, + environment: this.context.environment || 'development', + timestamp: new Date(), + metadata, + }; + const structuredLog: StructuredLog = { + level, + message, + context: logContext, + error, + }; + // Output as JSON for log aggregation systems + const logOutput = JSON.stringify(structuredLog); + // Route to NestJS Logger based on level — avoids raw console statements + switch (level) { + case LogLevel.DEBUG: + this.nestLogger.debug(logOutput); + break; + case LogLevel.INFO: + this.nestLogger.log(logOutput); + break; + case LogLevel.WARN: + this.nestLogger.warn(logOutput); + break; + case LogLevel.ERROR: + this.nestLogger.error(logOutput); + break; + case LogLevel.FATAL: + this.nestLogger.fatal(logOutput); + break; + } + } + /** + * Create child logger with additional context + */ + child(additionalContext: Partial): StructuredLoggerService { + const childLogger = new StructuredLoggerService(); + childLogger.setContext({ ...this.context, ...additionalContext }); + return childLogger; + } + /** + * Log HTTP request + */ + logRequest(method: string, url: string, statusCode: number, duration: number, metadata?: Record): void { + this.log(LogLevel.INFO, `${method} ${url} ${statusCode}`, { + ...metadata, + method, + url, + statusCode, + duration, + type: 'http_request', + }); + } + /** + * Log database query + */ + logQuery(query: string, duration: number, metadata?: Record): void { + this.log(LogLevel.DEBUG, 'Database query executed', { + ...metadata, + query, + duration, + type: 'database_query', + }); + } + /** + * Log business event + */ + logBusinessEvent(eventName: string, eventData: Record): void { + this.log(LogLevel.INFO, `Business event: ${eventName}`, { + ...eventData, + eventName, + type: 'business_event', + }); } - } - - /** - * Create child logger with additional context - */ - child(additionalContext: Partial): StructuredLoggerService { - const childLogger = new StructuredLoggerService(); - childLogger.setContext({ ...this.context, ...additionalContext }); - return childLogger; - } - - /** - * Log HTTP request - */ - logRequest( - method: string, - url: string, - statusCode: number, - duration: number, - metadata?: Record, - ): void { - this.log(LogLevel.INFO, `${method} ${url} ${statusCode}`, { - ...metadata, - method, - url, - statusCode, - duration, - type: 'http_request', - }); - } - - /** - * Log database query - */ - logQuery(query: string, duration: number, metadata?: Record): void { - this.log(LogLevel.DEBUG, 'Database query executed', { - ...metadata, - query, - duration, - type: 'database_query', - }); - } - - /** - * Log business event - */ - logBusinessEvent(eventName: string, eventData: Record): void { - this.log(LogLevel.INFO, `Business event: ${eventName}`, { - ...eventData, - eventName, - type: 'business_event', - }); - } } diff --git a/src/observability/metrics/metrics-analysis.service.ts b/src/observability/metrics/metrics-analysis.service.ts index 00dd7d01..bccdb190 100644 --- a/src/observability/metrics/metrics-analysis.service.ts +++ b/src/observability/metrics/metrics-analysis.service.ts @@ -1,345 +1,299 @@ import { Injectable, Logger } from '@nestjs/common'; import { MetricData, MetricType } from '../interfaces/observability.interfaces'; - /** * Metrics Analysis Service * Collects and analyzes custom metrics for business and performance insights */ @Injectable() export class MetricsAnalysisService { - private readonly logger = new Logger(MetricsAnalysisService.name); - private metrics: Map = new Map(); - private readonly MAX_METRICS_PER_NAME = 1000; - - /** - * Record a counter metric - */ - incrementCounter(name: string, value: number = 1, tags?: Record): void { - this.recordMetric({ - name, - value, - type: MetricType.COUNTER, - tags, - timestamp: new Date(), - }); - } - - /** - * Record a gauge metric - */ - recordGauge(name: string, value: number, tags?: Record): void { - this.recordMetric({ - name, - value, - type: MetricType.GAUGE, - tags, - timestamp: new Date(), - }); - } - - /** - * Record a histogram metric - */ - recordHistogram(name: string, value: number, tags?: Record): void { - this.recordMetric({ - name, - value, - type: MetricType.HISTOGRAM, - tags, - timestamp: new Date(), - }); - } - - /** - * Record a summary metric - */ - recordSummary(name: string, value: number, tags?: Record): void { - this.recordMetric({ - name, - value, - type: MetricType.SUMMARY, - tags, - timestamp: new Date(), - }); - } - - /** - * Record a metric - */ - private recordMetric(metric: MetricData): void { - if (!this.metrics.has(metric.name)) { - this.metrics.set(metric.name, []); + private readonly logger = new Logger(MetricsAnalysisService.name); + private metrics: Map = new Map(); + private readonly MAX_METRICS_PER_NAME = 1000; + /** + * Record a counter metric + */ + incrementCounter(name: string, value: number = 1, tags?: Record): void { + this.recordMetric({ + name, + value, + type: MetricType.COUNTER, + tags, + timestamp: new Date(), + }); } - - // Safe retrieval with fallback — avoids non-null assertion - const metricList = this.metrics.get(metric.name) ?? []; - metricList.push(metric); - - // Maintain size limit (FIFO) - if (metricList.length > this.MAX_METRICS_PER_NAME) { - metricList.shift(); + /** + * Record a gauge metric + */ + recordGauge(name: string, value: number, tags?: Record): void { + this.recordMetric({ + name, + value, + type: MetricType.GAUGE, + tags, + timestamp: new Date(), + }); } - - // Ensure the (possibly new) array is always stored back in the map - this.metrics.set(metric.name, metricList); - - this.logger.debug(`Recorded metric: ${metric.name} = ${metric.value}`); - } - - /** - * Get metrics by name - */ - getMetrics(name: string, limit?: number): MetricData[] { - const metrics = this.metrics.get(name) || []; - return limit ? metrics.slice(-limit) : metrics; - } - - /** - * Get all metric names - */ - getMetricNames(): string[] { - return Array.from(this.metrics.keys()); - } - - /** - * Calculate metric statistics - */ - getMetricStatistics(name: string, timeRange?: { start: Date; end: Date }) { - let metrics = this.metrics.get(name) || []; - - if (timeRange) { - metrics = metrics.filter( - (m) => m.timestamp >= timeRange.start && m.timestamp <= timeRange.end, - ); + /** + * Record a histogram metric + */ + recordHistogram(name: string, value: number, tags?: Record): void { + this.recordMetric({ + name, + value, + type: MetricType.HISTOGRAM, + tags, + timestamp: new Date(), + }); } - - if (metrics.length === 0) { - return null; + /** + * Record a summary metric + */ + recordSummary(name: string, value: number, tags?: Record): void { + this.recordMetric({ + name, + value, + type: MetricType.SUMMARY, + tags, + timestamp: new Date(), + }); } - - const values = metrics.map((m) => m.value); - const sum = values.reduce((a, b) => a + b, 0); - const avg = sum / values.length; - const min = Math.min(...values); - const max = Math.max(...values); - - // Calculate percentiles - const sortedValues = [...values].sort((a, b) => a - b); - const p50 = this.percentile(sortedValues, 50); - const p95 = this.percentile(sortedValues, 95); - const p99 = this.percentile(sortedValues, 99); - - // Calculate standard deviation - const variance = values.reduce((acc, val) => acc + Math.pow(val - avg, 2), 0) / values.length; - const stdDev = Math.sqrt(variance); - - return { - name, - count: metrics.length, - sum, - avg, - min, - max, - p50, - p95, - p99, - stdDev, - type: metrics[0].type, - }; - } - - /** - * Calculate percentile - */ - private percentile(sortedValues: number[], percentile: number): number { - const index = (percentile / 100) * (sortedValues.length - 1); - const lower = Math.floor(index); - const upper = Math.ceil(index); - const weight = index - lower; - - if (lower === upper) { - return sortedValues[lower]; + /** + * Record a metric + */ + private recordMetric(metric: MetricData): void { + if (!this.metrics.has(metric.name)) { + this.metrics.set(metric.name, []); + } + // Safe retrieval with fallback — avoids non-null assertion + const metricList = this.metrics.get(metric.name) ?? []; + metricList.push(metric); + // Maintain size limit (FIFO) + if (metricList.length > this.MAX_METRICS_PER_NAME) { + metricList.shift(); + } + // Ensure the (possibly new) array is always stored back in the map + this.metrics.set(metric.name, metricList); + this.logger.debug(`Recorded metric: ${metric.name} = ${metric.value}`); } - - return sortedValues[lower] * (1 - weight) + sortedValues[upper] * weight; - } - - /** - * Get metrics trend - */ - getMetricTrend(name: string, windowSize: number = 10): 'up' | 'down' | 'stable' { - const metrics = this.metrics.get(name) || []; - if (metrics.length < windowSize * 2) { - return 'stable'; + /** + * Get metrics by name + */ + getMetrics(name: string, limit?: number): MetricData[] { + const metrics = this.metrics.get(name) || []; + return limit ? metrics.slice(-limit) : metrics; + } + /** + * Get all metric names + */ + getMetricNames(): string[] { + return Array.from(this.metrics.keys()); + } + /** + * Calculate metric statistics + */ + getMetricStatistics(name: string, timeRange?: { + start: Date; + end: Date; + }) { + let metrics = this.metrics.get(name) || []; + if (timeRange) { + metrics = metrics.filter((m) => m.timestamp >= timeRange.start && m.timestamp <= timeRange.end); + } + if (metrics.length === 0) { + return null; + } + const values = metrics.map((m) => m.value); + const sum = values.reduce((a, b) => a + b, 0); + const avg = sum / values.length; + const min = Math.min(...values); + const max = Math.max(...values); + // Calculate percentiles + const sortedValues = [...values].sort((a, b) => a - b); + const p50 = this.percentile(sortedValues, 50); + const p95 = this.percentile(sortedValues, 95); + const p99 = this.percentile(sortedValues, 99); + // Calculate standard deviation + const variance = values.reduce((acc, val) => acc + Math.pow(val - avg, 2), 0) / values.length; + const stdDev = Math.sqrt(variance); + return { + name, + count: metrics.length, + sum, + avg, + min, + max, + p50, + p95, + p99, + stdDev, + type: metrics[0].type, + }; + } + /** + * Calculate percentile + */ + private percentile(sortedValues: number[], percentile: number): number { + const index = (percentile / 100) * (sortedValues.length - 1); + const lower = Math.floor(index); + const upper = Math.ceil(index); + const weight = index - lower; + if (lower === upper) { + return sortedValues[lower]; + } + return sortedValues[lower] * (1 - weight) + sortedValues[upper] * weight; + } + /** + * Get metrics trend + */ + getMetricTrend(name: string, windowSize: number = 10): 'up' | 'down' | 'stable' { + const metrics = this.metrics.get(name) || []; + if (metrics.length < windowSize * 2) { + return 'stable'; + } + const recent = metrics.slice(-windowSize); + const previous = metrics.slice(-windowSize * 2, -windowSize); + const recentAvg = recent.reduce((a, b) => a + b.value, 0) / recent.length; + const previousAvg = previous.reduce((a, b) => a + b.value, 0) / previous.length; + const change = ((recentAvg - previousAvg) / previousAvg) * 100; + if (change > 10) + return 'up'; + if (change < -10) + return 'down'; + return 'stable'; + } + /** + * Get all metrics statistics + */ + getAllMetricsStatistics() { + const stats: Record = {}; + this.metrics.forEach((_, name) => { + stats[name] = this.getMetricStatistics(name); + }); + return stats; + } + /** + * Clear old metrics + */ + clearOldMetrics(olderThan: Date): number { + let cleared = 0; + this.metrics.forEach((metricList, name) => { + const initialLength = metricList.length; + const filtered = metricList.filter((m) => m.timestamp > olderThan); + this.metrics.set(name, filtered); + cleared += initialLength - filtered.length; + }); + this.logger.log(`Cleared ${cleared} old metrics`); + return cleared; + } + /** + * Business Metrics - Track user signups + */ + trackUserSignup(_userId: string, source?: string): void { + this.incrementCounter('user.signups', 1, { source: source || 'direct' }); + } + /** + * Business Metrics - Track course enrollment + */ + trackCourseEnrollment(courseId: string, _userId: string): void { + this.incrementCounter('course.enrollments', 1, { courseId }); + } + /** + * Business Metrics - Track payment + */ + trackPayment(amount: number, currency: string = 'USD'): void { + this.recordHistogram('payment.amount', amount, { currency }); + this.incrementCounter('payment.count', 1, { currency }); + } + /** + * Performance Metrics - Track API response time + */ + trackApiResponseTime(endpoint: string, duration: number, statusCode: number): void { + this.recordHistogram('api.response_time', duration, { + endpoint, + status: statusCode.toString(), + }); + } + /** + * Performance Metrics - Track database query time + */ + trackDatabaseQueryTime(query: string, duration: number): void { + this.recordHistogram('db.query_time', duration, { + query: query.substring(0, 50), // Truncate for tag + }); + } + /** + * Performance Metrics - Track cache hit/miss + */ + trackCacheHit(key: string, hit: boolean): void { + this.incrementCounter('cache.requests', 1, { + result: hit ? 'hit' : 'miss', + }); + } + /** + * Performance Metrics - Track queue processing time + */ + trackQueueProcessingTime(jobName: string, duration: number): void { + this.recordHistogram('queue.processing_time', duration, { jobName }); + } + /** + * System Metrics - Track memory usage + */ + trackMemoryUsage(): void { + const usage = process.memoryUsage(); + this.recordGauge('system.memory.heap_used', usage.heapUsed); + this.recordGauge('system.memory.heap_total', usage.heapTotal); + this.recordGauge('system.memory.rss', usage.rss); + } + /** + * System Metrics - Track CPU usage + */ + trackCpuUsage(usage: number): void { + this.recordGauge('system.cpu.usage', usage); + } + /** + * Get dashboard metrics + */ + getDashboardMetrics() { + return { + business: { + signups: this.getMetricStatistics('user.signups'), + enrollments: this.getMetricStatistics('course.enrollments'), + payments: this.getMetricStatistics('payment.amount'), + }, + performance: { + apiResponseTime: this.getMetricStatistics('api.response_time'), + dbQueryTime: this.getMetricStatistics('db.query_time'), + queueProcessingTime: this.getMetricStatistics('queue.processing_time'), + }, + system: { + memoryUsage: this.getMetricStatistics('system.memory.heap_used'), + cpuUsage: this.getMetricStatistics('system.cpu.usage'), + }, + cache: { + requests: this.getMetricStatistics('cache.requests'), + }, + }; + } + /** + * Export metrics in Prometheus format + */ + exportPrometheusMetrics(): string { + const lines: string[] = []; + this.metrics.forEach((metricList, name) => { + if (metricList.length === 0) + return; + const latestMetric = metricList[metricList.length - 1]; + const sanitizedName = name.replace(/\./g, '_'); + // Add metric type + lines.push(`# TYPE ${sanitizedName} ${latestMetric.type}`); + // Add metric value + const tags = latestMetric.tags + ? Object.entries(latestMetric.tags) + .map(([k, v]) => `${k}="${v}"`) + .join(',') + : ''; + lines.push(`${sanitizedName}${tags ? `{${tags}}` : ''} ${latestMetric.value}`); + }); + return lines.join('\n'); } - - const recent = metrics.slice(-windowSize); - const previous = metrics.slice(-windowSize * 2, -windowSize); - - const recentAvg = recent.reduce((a, b) => a + b.value, 0) / recent.length; - const previousAvg = previous.reduce((a, b) => a + b.value, 0) / previous.length; - - const change = ((recentAvg - previousAvg) / previousAvg) * 100; - - if (change > 10) return 'up'; - if (change < -10) return 'down'; - return 'stable'; - } - - /** - * Get all metrics statistics - */ - getAllMetricsStatistics() { - const stats: Record = {}; - - this.metrics.forEach((_, name) => { - stats[name] = this.getMetricStatistics(name); - }); - - return stats; - } - - /** - * Clear old metrics - */ - clearOldMetrics(olderThan: Date): number { - let cleared = 0; - - this.metrics.forEach((metricList, name) => { - const initialLength = metricList.length; - const filtered = metricList.filter((m) => m.timestamp > olderThan); - this.metrics.set(name, filtered); - cleared += initialLength - filtered.length; - }); - - this.logger.log(`Cleared ${cleared} old metrics`); - return cleared; - } - - /** - * Business Metrics - Track user signups - */ - trackUserSignup(_userId: string, source?: string): void { - this.incrementCounter('user.signups', 1, { source: source || 'direct' }); - } - - /** - * Business Metrics - Track course enrollment - */ - trackCourseEnrollment(courseId: string, _userId: string): void { - this.incrementCounter('course.enrollments', 1, { courseId }); - } - - /** - * Business Metrics - Track payment - */ - trackPayment(amount: number, currency: string = 'USD'): void { - this.recordHistogram('payment.amount', amount, { currency }); - this.incrementCounter('payment.count', 1, { currency }); - } - - /** - * Performance Metrics - Track API response time - */ - trackApiResponseTime(endpoint: string, duration: number, statusCode: number): void { - this.recordHistogram('api.response_time', duration, { - endpoint, - status: statusCode.toString(), - }); - } - - /** - * Performance Metrics - Track database query time - */ - trackDatabaseQueryTime(query: string, duration: number): void { - this.recordHistogram('db.query_time', duration, { - query: query.substring(0, 50), // Truncate for tag - }); - } - - /** - * Performance Metrics - Track cache hit/miss - */ - trackCacheHit(key: string, hit: boolean): void { - this.incrementCounter('cache.requests', 1, { - result: hit ? 'hit' : 'miss', - }); - } - - /** - * Performance Metrics - Track queue processing time - */ - trackQueueProcessingTime(jobName: string, duration: number): void { - this.recordHistogram('queue.processing_time', duration, { jobName }); - } - - /** - * System Metrics - Track memory usage - */ - trackMemoryUsage(): void { - const usage = process.memoryUsage(); - this.recordGauge('system.memory.heap_used', usage.heapUsed); - this.recordGauge('system.memory.heap_total', usage.heapTotal); - this.recordGauge('system.memory.rss', usage.rss); - } - - /** - * System Metrics - Track CPU usage - */ - trackCpuUsage(usage: number): void { - this.recordGauge('system.cpu.usage', usage); - } - - /** - * Get dashboard metrics - */ - getDashboardMetrics() { - return { - business: { - signups: this.getMetricStatistics('user.signups'), - enrollments: this.getMetricStatistics('course.enrollments'), - payments: this.getMetricStatistics('payment.amount'), - }, - performance: { - apiResponseTime: this.getMetricStatistics('api.response_time'), - dbQueryTime: this.getMetricStatistics('db.query_time'), - queueProcessingTime: this.getMetricStatistics('queue.processing_time'), - }, - system: { - memoryUsage: this.getMetricStatistics('system.memory.heap_used'), - cpuUsage: this.getMetricStatistics('system.cpu.usage'), - }, - cache: { - requests: this.getMetricStatistics('cache.requests'), - }, - }; - } - - /** - * Export metrics in Prometheus format - */ - exportPrometheusMetrics(): string { - const lines: string[] = []; - - this.metrics.forEach((metricList, name) => { - if (metricList.length === 0) return; - - const latestMetric = metricList[metricList.length - 1]; - const sanitizedName = name.replace(/\./g, '_'); - - // Add metric type - lines.push(`# TYPE ${sanitizedName} ${latestMetric.type}`); - - // Add metric value - const tags = latestMetric.tags - ? Object.entries(latestMetric.tags) - .map(([k, v]) => `${k}="${v}"`) - .join(',') - : ''; - - lines.push(`${sanitizedName}${tags ? `{${tags}}` : ''} ${latestMetric.value}`); - }); - - return lines.join('\n'); - } } diff --git a/src/observability/observability.controller.ts b/src/observability/observability.controller.ts index c5daabe9..bc0b8b40 100644 --- a/src/observability/observability.controller.ts +++ b/src/observability/observability.controller.ts @@ -5,202 +5,198 @@ import { DistributedTracingService } from './tracing/distributed-tracing.service import { MetricsAnalysisService } from './metrics/metrics-analysis.service'; import { AnomalyDetectionService } from './anomaly/anomaly-detection.service'; import { LogQuery } from './interfaces/observability.interfaces'; - /** * Observability Controller * Provides REST API for observability features */ @Controller('observability') export class ObservabilityController { - constructor( - private readonly observabilityService: ObservabilityService, - private readonly logAggregation: LogAggregationService, - private readonly tracing: DistributedTracingService, - private readonly metrics: MetricsAnalysisService, - private readonly anomalyDetection: AnomalyDetectionService, - ) {} - - /** - * Get observability dashboard - */ - @Get('dashboard') - async getDashboard(): Promise { - return this.observabilityService.getObservabilityDashboard(); - } - - /** - * Get system overview - */ - @Get('overview') - async getOverview(): Promise { - return this.observabilityService.getSystemOverview(); - } - - /** - * Search logs - */ - @Post('logs/search') - async searchLogs(@Body() query: LogQuery): Promise { - return this.logAggregation.searchLogs(query); - } - - /** - * Get logs by correlation ID - */ - @Get('logs/correlation/:id') - async getLogsByCorrelation(@Query('id') id: string): Promise { - return this.logAggregation.getLogsByCorrelationId(id); - } - - /** - * Get error logs - */ - @Get('logs/errors') - async getErrorLogs(@Query('limit') limit?: number): Promise { - return this.logAggregation.getErrorLogs(limit ? parseInt(limit.toString()) : 100); - } - - /** - * Get log statistics - */ - @Get('logs/statistics') - async getLogStatistics( - @Query('startTime') startTime?: string, - @Query('endTime') endTime?: string, - ): Promise { - const timeRange = - startTime && endTime ? { start: new Date(startTime), end: new Date(endTime) } : undefined; - - return this.logAggregation.getLogStatistics(timeRange); - } - - /** - * Get recent logs - */ - @Get('logs/recent') - async getRecentLogs(@Query('limit') limit?: number): Promise { - return this.logAggregation.getRecentLogs(limit ? parseInt(limit.toString()) : 100); - } - - /** - * Get trace by ID - */ - @Get('traces/:id') - async getTrace(@Query('id') id: string): Promise { - return this.tracing.getTraceById(id); - } - - /** - * Get all traces - */ - @Get('traces') - async getAllTraces(): Promise { - return this.tracing.getAllSpans(); - } - - /** - * Get trace statistics - */ - @Get('traces/statistics') - async getTraceStatistics(): Promise { - return this.tracing.getTraceStatistics(); - } - - /** - * Get metrics - */ - @Get('metrics/:name') - async getMetrics(@Query('name') name: string, @Query('limit') limit?: number): Promise { - return this.metrics.getMetrics(name, limit ? parseInt(limit.toString()) : undefined); - } - - /** - * Get metric statistics - */ - @Get('metrics/:name/statistics') - async getMetricStatistics( - @Query('name') name: string, - @Query('startTime') startTime?: string, - @Query('endTime') endTime?: string, - ): Promise { - const timeRange = - startTime && endTime ? { start: new Date(startTime), end: new Date(endTime) } : undefined; - - return this.metrics.getMetricStatistics(name, timeRange); - } - - /** - * Get all metrics - */ - @Get('metrics') - async getAllMetrics(): Promise { - return { - names: this.metrics.getMetricNames(), - statistics: this.metrics.getAllMetricsStatistics(), - }; - } - - /** - * Get dashboard metrics - */ - @Get('metrics/dashboard') - async getDashboardMetrics(): Promise { - return this.metrics.getDashboardMetrics(); - } - - /** - * Export Prometheus metrics - */ - @Get('metrics/export/prometheus') - async exportPrometheusMetrics(): Promise { - return this.metrics.exportPrometheusMetrics(); - } - - /** - * Get anomalies - */ - @Get('anomalies') - async getAnomalies(@Query('limit') limit?: number): Promise { - return this.anomalyDetection.getAnomalies(limit ? parseInt(limit.toString()) : undefined); - } - - /** - * Get anomalies by metric - */ - @Get('anomalies/metric/:name') - async getAnomaliesByMetric(@Query('name') name: string): Promise { - return this.anomalyDetection.getAnomaliesByMetric(name); - } - - /** - * Get recent anomalies - */ - @Get('anomalies/recent') - async getRecentAnomalies(@Query('minutes') minutes?: number): Promise { - return this.anomalyDetection.getRecentAnomalies(minutes ? parseInt(minutes.toString()) : 60); - } - - /** - * Get anomaly statistics - */ - @Get('anomalies/statistics') - async getAnomalyStatistics(): Promise { - return this.anomalyDetection.getAnomalyStatistics(); - } - - /** - * Get system health - */ - @Get('health') - async getSystemHealth(): Promise { - return this.anomalyDetection.getSystemHealth(); - } - - /** - * Detect anomalies in a metric - */ - @Post('anomalies/detect') - async detectAnomalies(@Body() body: { metricName: string; windowSize?: number }): Promise { - return this.anomalyDetection.detectAnomalies(body.metricName, body.windowSize); - } + constructor(private readonly observabilityService: ObservabilityService, private readonly logAggregation: LogAggregationService, private readonly tracing: DistributedTracingService, private readonly metrics: MetricsAnalysisService, private readonly anomalyDetection: AnomalyDetectionService) { } + /** + * Get observability dashboard + */ + @Get('dashboard') + async getDashboard(): Promise { + return this.observabilityService.getObservabilityDashboard(); + } + /** + * Get system overview + */ + @Get('overview') + async getOverview(): Promise { + return this.observabilityService.getSystemOverview(); + } + /** + * Search logs + */ + @Post('logs/search') + async searchLogs( + @Body() + query: LogQuery): Promise { + return this.logAggregation.searchLogs(query); + } + /** + * Get logs by correlation ID + */ + @Get('logs/correlation/:id') + async getLogsByCorrelation( + @Query('id') + id: string): Promise { + return this.logAggregation.getLogsByCorrelationId(id); + } + /** + * Get error logs + */ + @Get('logs/errors') + async getErrorLogs( + @Query('limit') + limit?: number): Promise { + return this.logAggregation.getErrorLogs(limit ? parseInt(limit.toString()) : 100); + } + /** + * Get log statistics + */ + @Get('logs/statistics') + async getLogStatistics( + @Query('startTime') + startTime?: string, + @Query('endTime') + endTime?: string): Promise { + const timeRange = startTime && endTime ? { start: new Date(startTime), end: new Date(endTime) } : undefined; + return this.logAggregation.getLogStatistics(timeRange); + } + /** + * Get recent logs + */ + @Get('logs/recent') + async getRecentLogs( + @Query('limit') + limit?: number): Promise { + return this.logAggregation.getRecentLogs(limit ? parseInt(limit.toString()) : 100); + } + /** + * Get trace by ID + */ + @Get('traces/:id') + async getTrace( + @Query('id') + id: string): Promise { + return this.tracing.getTraceById(id); + } + /** + * Get all traces + */ + @Get('traces') + async getAllTraces(): Promise { + return this.tracing.getAllSpans(); + } + /** + * Get trace statistics + */ + @Get('traces/statistics') + async getTraceStatistics(): Promise { + return this.tracing.getTraceStatistics(); + } + /** + * Get metrics + */ + @Get('metrics/:name') + async getMetrics( + @Query('name') + name: string, + @Query('limit') + limit?: number): Promise { + return this.metrics.getMetrics(name, limit ? parseInt(limit.toString()) : undefined); + } + /** + * Get metric statistics + */ + @Get('metrics/:name/statistics') + async getMetricStatistics( + @Query('name') + name: string, + @Query('startTime') + startTime?: string, + @Query('endTime') + endTime?: string): Promise { + const timeRange = startTime && endTime ? { start: new Date(startTime), end: new Date(endTime) } : undefined; + return this.metrics.getMetricStatistics(name, timeRange); + } + /** + * Get all metrics + */ + @Get('metrics') + async getAllMetrics(): Promise { + return { + names: this.metrics.getMetricNames(), + statistics: this.metrics.getAllMetricsStatistics(), + }; + } + /** + * Get dashboard metrics + */ + @Get('metrics/dashboard') + async getDashboardMetrics(): Promise { + return this.metrics.getDashboardMetrics(); + } + /** + * Export Prometheus metrics + */ + @Get('metrics/export/prometheus') + async exportPrometheusMetrics(): Promise { + return this.metrics.exportPrometheusMetrics(); + } + /** + * Get anomalies + */ + @Get('anomalies') + async getAnomalies( + @Query('limit') + limit?: number): Promise { + return this.anomalyDetection.getAnomalies(limit ? parseInt(limit.toString()) : undefined); + } + /** + * Get anomalies by metric + */ + @Get('anomalies/metric/:name') + async getAnomaliesByMetric( + @Query('name') + name: string): Promise { + return this.anomalyDetection.getAnomaliesByMetric(name); + } + /** + * Get recent anomalies + */ + @Get('anomalies/recent') + async getRecentAnomalies( + @Query('minutes') + minutes?: number): Promise { + return this.anomalyDetection.getRecentAnomalies(minutes ? parseInt(minutes.toString()) : 60); + } + /** + * Get anomaly statistics + */ + @Get('anomalies/statistics') + async getAnomalyStatistics(): Promise { + return this.anomalyDetection.getAnomalyStatistics(); + } + /** + * Get system health + */ + @Get('health') + async getSystemHealth(): Promise { + return this.anomalyDetection.getSystemHealth(); + } + /** + * Detect anomalies in a metric + */ + @Post('anomalies/detect') + async detectAnomalies( + @Body() + body: { + metricName: string; + windowSize?: number; + }): Promise { + return this.anomalyDetection.detectAnomalies(body.metricName, body.windowSize); + } } diff --git a/src/observability/observability.module.ts b/src/observability/observability.module.ts index b713d8ad..18e40dfc 100644 --- a/src/observability/observability.module.ts +++ b/src/observability/observability.module.ts @@ -7,30 +7,30 @@ import { LogAggregationService } from './logging/log-aggregation.service'; import { DistributedTracingService } from './tracing/distributed-tracing.service'; import { MetricsAnalysisService } from './metrics/metrics-analysis.service'; import { AnomalyDetectionService } from './anomaly/anomaly-detection.service'; - /** * Observability Module * Comprehensive logging, tracing, metrics, and anomaly detection */ @Global() @Module({ - imports: [ScheduleModule.forRoot()], - controllers: [ObservabilityController], - providers: [ - ObservabilityService, - StructuredLoggerService, - LogAggregationService, - DistributedTracingService, - MetricsAnalysisService, - AnomalyDetectionService, - ], - exports: [ - ObservabilityService, - StructuredLoggerService, - LogAggregationService, - DistributedTracingService, - MetricsAnalysisService, - AnomalyDetectionService, - ], + imports: [ScheduleModule.forRoot()], + controllers: [ObservabilityController], + providers: [ + ObservabilityService, + StructuredLoggerService, + LogAggregationService, + DistributedTracingService, + MetricsAnalysisService, + AnomalyDetectionService, + ], + exports: [ + ObservabilityService, + StructuredLoggerService, + LogAggregationService, + DistributedTracingService, + MetricsAnalysisService, + AnomalyDetectionService, + ], }) -export class ObservabilityModule {} +export class ObservabilityModule { +} diff --git a/src/observability/observability.service.ts b/src/observability/observability.service.ts index 84d7fa20..eee394a7 100644 --- a/src/observability/observability.service.ts +++ b/src/observability/observability.service.ts @@ -4,163 +4,127 @@ import { LogAggregationService } from './logging/log-aggregation.service'; import { DistributedTracingService } from './tracing/distributed-tracing.service'; import { MetricsAnalysisService } from './metrics/metrics-analysis.service'; import { AnomalyDetectionService } from './anomaly/anomaly-detection.service'; - /** * Observability Service * Central service for all observability features */ @Injectable() export class ObservabilityService { - private readonly logger = new Logger(ObservabilityService.name); - - constructor( - private readonly structuredLogger: StructuredLoggerService, - private readonly logAggregation: LogAggregationService, - private readonly tracing: DistributedTracingService, - private readonly metrics: MetricsAnalysisService, - private readonly anomalyDetection: AnomalyDetectionService, - ) {} - - /** - * Get comprehensive observability dashboard - */ - async getObservabilityDashboard() { - const [logStats, traceStats, metricsStats, anomalyStats, systemHealth] = await Promise.all([ - this.logAggregation.getLogStatistics(), - this.tracing.getTraceStatistics(), - this.metrics.getAllMetricsStatistics(), - this.anomalyDetection.getAnomalyStatistics(), - this.anomalyDetection.getSystemHealth(), - ]); - - return { - logs: logStats, - traces: traceStats, - metrics: metricsStats, - anomalies: anomalyStats, - health: systemHealth, - timestamp: new Date(), - }; - } - - /** - * Get logger instance - */ - getLogger(): StructuredLoggerService { - return this.structuredLogger; - } - - /** - * Get log aggregation service - */ - getLogAggregation(): LogAggregationService { - return this.logAggregation; - } - - /** - * Get tracing service - */ - getTracing(): DistributedTracingService { - return this.tracing; - } - - /** - * Get metrics service - */ - getMetrics(): MetricsAnalysisService { - return this.metrics; - } - - /** - * Get anomaly detection service - */ - getAnomalyDetection(): AnomalyDetectionService { - return this.anomalyDetection; - } - - /** - * Initialize observability for a request - */ - initializeRequestObservability(correlationId: string, userId?: string) { - this.structuredLogger.setCorrelationId(correlationId); - if (userId) { - this.structuredLogger.setUserId(userId); + private readonly logger = new Logger(ObservabilityService.name); + constructor(private readonly structuredLogger: StructuredLoggerService, private readonly logAggregation: LogAggregationService, private readonly tracing: DistributedTracingService, private readonly metrics: MetricsAnalysisService, private readonly anomalyDetection: AnomalyDetectionService) { } + /** + * Get comprehensive observability dashboard + */ + async getObservabilityDashboard() { + const [logStats, traceStats, metricsStats, anomalyStats, systemHealth] = await Promise.all([ + this.logAggregation.getLogStatistics(), + this.tracing.getTraceStatistics(), + this.metrics.getAllMetricsStatistics(), + this.anomalyDetection.getAnomalyStatistics(), + this.anomalyDetection.getSystemHealth(), + ]); + return { + logs: logStats, + traces: traceStats, + metrics: metricsStats, + anomalies: anomalyStats, + health: systemHealth, + timestamp: new Date(), + }; } - } - - /** - * Track request with full observability - */ - async trackRequest( - method: string, - url: string, - fn: () => Promise, - correlationId: string, - ): Promise { - const startTime = new Date(); - - // Initialize logging context - this.initializeRequestObservability(correlationId); - - // Start trace span - return this.tracing.executeInSpan( - `${method} ${url}`, - async (_span) => { - try { - // Execute request - const result = await fn(); - - // Calculate duration - const duration = Date.now() - startTime.getTime(); - - // Log request - this.structuredLogger.logRequest(method, url, 200, duration); - - // Track metrics - this.metrics.trackApiResponseTime(url, duration, 200); - - return result; - } catch (error) { - // Calculate duration - const duration = Date.now() - startTime.getTime(); - - // Log error - this.structuredLogger.error(`Request failed: ${method} ${url}`, error as Error); - - // Track metrics - this.metrics.trackApiResponseTime(url, duration, 500); - - throw error; + /** + * Get logger instance + */ + getLogger(): StructuredLoggerService { + return this.structuredLogger; + } + /** + * Get log aggregation service + */ + getLogAggregation(): LogAggregationService { + return this.logAggregation; + } + /** + * Get tracing service + */ + getTracing(): DistributedTracingService { + return this.tracing; + } + /** + * Get metrics service + */ + getMetrics(): MetricsAnalysisService { + return this.metrics; + } + /** + * Get anomaly detection service + */ + getAnomalyDetection(): AnomalyDetectionService { + return this.anomalyDetection; + } + /** + * Initialize observability for a request + */ + initializeRequestObservability(correlationId: string, userId?: string) { + this.structuredLogger.setCorrelationId(correlationId); + if (userId) { + this.structuredLogger.setUserId(userId); } - }, - { - 'http.method': method, - 'http.url': url, - 'correlation.id': correlationId, - }, - ); - } - - /** - * Get system overview - */ - async getSystemOverview() { - const dashboard = await this.getObservabilityDashboard(); - const recentErrors = await this.logAggregation.getErrorLogs(10); - const recentAnomalies = this.anomalyDetection.getRecentAnomalies(60); - - return { - status: dashboard.health.status, - issues: dashboard.health.issues, - recentErrors: recentErrors.length, - recentAnomalies: recentAnomalies.length, - metrics: { - totalLogs: dashboard.logs.total, - totalTraces: dashboard.traces.total, - errorRate: dashboard.logs.errorRate, - avgResponseTime: dashboard.traces.avgDuration, - }, - timestamp: new Date(), - }; - } + } + /** + * Track request with full observability + */ + async trackRequest(method: string, url: string, fn: () => Promise, correlationId: string): Promise { + const startTime = new Date(); + // Initialize logging context + this.initializeRequestObservability(correlationId); + // Start trace span + return this.tracing.executeInSpan(`${method} ${url}`, async (_span) => { + try { + // Execute request + const result = await fn(); + // Calculate duration + const duration = Date.now() - startTime.getTime(); + // Log request + this.structuredLogger.logRequest(method, url, 200, duration); + // Track metrics + this.metrics.trackApiResponseTime(url, duration, 200); + return result; + } + catch (error) { + // Calculate duration + const duration = Date.now() - startTime.getTime(); + // Log error + this.structuredLogger.error(`Request failed: ${method} ${url}`, error as Error); + // Track metrics + this.metrics.trackApiResponseTime(url, duration, 500); + throw error; + } + }, { + 'http.method': method, + 'http.url': url, + 'correlation.id': correlationId, + }); + } + /** + * Get system overview + */ + async getSystemOverview() { + const dashboard = await this.getObservabilityDashboard(); + const recentErrors = await this.logAggregation.getErrorLogs(10); + const recentAnomalies = this.anomalyDetection.getRecentAnomalies(60); + return { + status: dashboard.health.status, + issues: dashboard.health.issues, + recentErrors: recentErrors.length, + recentAnomalies: recentAnomalies.length, + metrics: { + totalLogs: dashboard.logs.total, + totalTraces: dashboard.traces.total, + errorRate: dashboard.logs.errorRate, + avgResponseTime: dashboard.traces.avgDuration, + }, + timestamp: new Date(), + }; + } } diff --git a/src/observability/tracing/distributed-tracing.service.ts b/src/observability/tracing/distributed-tracing.service.ts index 14aafbf2..303308a5 100644 --- a/src/observability/tracing/distributed-tracing.service.ts +++ b/src/observability/tracing/distributed-tracing.service.ts @@ -1,298 +1,250 @@ import { Injectable, Logger } from '@nestjs/common'; import { trace, Span, SpanStatusCode, context } from '@opentelemetry/api'; import { TraceSpan, SpanStatus, SpanEvent } from '../interfaces/observability.interfaces'; - /** * Distributed Tracing Service * Provides distributed tracing across all services using OpenTelemetry */ @Injectable() export class DistributedTracingService { - private readonly logger = new Logger(DistributedTracingService.name); - private readonly tracer = trace.getTracer('teachlink', '1.0.0'); - private spans: Map = new Map(); - - /** - * Start a new trace span - */ - startSpan(name: string, attributes?: Record): Span { - const span = this.tracer.startSpan(name); - - if (attributes) { - Object.entries(attributes).forEach(([key, value]) => { - span.setAttribute(key, value); - }); + private readonly logger = new Logger(DistributedTracingService.name); + private readonly tracer = trace.getTracer('teachlink', '1.0.0'); + private spans: Map = new Map(); + /** + * Start a new trace span + */ + startSpan(name: string, attributes?: Record): Span { + const span = this.tracer.startSpan(name); + if (attributes) { + Object.entries(attributes).forEach(([key, value]) => { + span.setAttribute(key, value); + }); + } + // Store span info + const spanContext = span.spanContext(); + const traceSpan: TraceSpan = { + traceId: spanContext.traceId, + spanId: spanContext.spanId, + name, + startTime: new Date(), + status: SpanStatus.UNSET, + attributes: attributes || {}, + events: [], + }; + this.spans.set(spanContext.spanId, traceSpan); + this.logger.debug(`Started span: ${name} (${spanContext.spanId})`); + return span; } - - // Store span info - const spanContext = span.spanContext(); - const traceSpan: TraceSpan = { - traceId: spanContext.traceId, - spanId: spanContext.spanId, - name, - startTime: new Date(), - status: SpanStatus.UNSET, - attributes: attributes || {}, - events: [], - }; - - this.spans.set(spanContext.spanId, traceSpan); - - this.logger.debug(`Started span: ${name} (${spanContext.spanId})`); - return span; - } - - /** - * End a span - */ - endSpan(span: Span, status?: SpanStatus): void { - const spanContext = span.spanContext(); - const traceSpan = this.spans.get(spanContext.spanId); - - if (traceSpan) { - traceSpan.endTime = new Date(); - traceSpan.duration = traceSpan.endTime.getTime() - traceSpan.startTime.getTime(); - traceSpan.status = status || SpanStatus.OK; + /** + * End a span + */ + endSpan(span: Span, status?: SpanStatus): void { + const spanContext = span.spanContext(); + const traceSpan = this.spans.get(spanContext.spanId); + if (traceSpan) { + traceSpan.endTime = new Date(); + traceSpan.duration = traceSpan.endTime.getTime() - traceSpan.startTime.getTime(); + traceSpan.status = status || SpanStatus.OK; + } + if (status === SpanStatus.ERROR) { + span.setStatus({ code: SpanStatusCode.ERROR }); + } + else { + span.setStatus({ code: SpanStatusCode.OK }); + } + span.end(); + this.logger.debug(`Ended span: ${traceSpan?.name} (${spanContext.spanId})`); } - - if (status === SpanStatus.ERROR) { - span.setStatus({ code: SpanStatusCode.ERROR }); - } else { - span.setStatus({ code: SpanStatusCode.OK }); + /** + * Add attributes to a span + */ + setSpanAttributes(span: Span, attributes: Record): void { + Object.entries(attributes).forEach(([key, value]) => { + span.setAttribute(key, value); + }); + const spanContext = span.spanContext(); + const traceSpan = this.spans.get(spanContext.spanId); + if (traceSpan) { + traceSpan.attributes = { ...traceSpan.attributes, ...attributes }; + } } - - span.end(); - this.logger.debug(`Ended span: ${traceSpan?.name} (${spanContext.spanId})`); - } - - /** - * Add attributes to a span - */ - setSpanAttributes(span: Span, attributes: Record): void { - Object.entries(attributes).forEach(([key, value]) => { - span.setAttribute(key, value); - }); - - const spanContext = span.spanContext(); - const traceSpan = this.spans.get(spanContext.spanId); - if (traceSpan) { - traceSpan.attributes = { ...traceSpan.attributes, ...attributes }; + /** + * Record an exception in a span + */ + recordException(span: Span, error: Error): void { + span.recordException(error); + span.setStatus({ + code: SpanStatusCode.ERROR, + message: error.message, + }); + const spanContext = span.spanContext(); + const traceSpan = this.spans.get(spanContext.spanId); + if (traceSpan) { + traceSpan.status = SpanStatus.ERROR; + traceSpan.attributes.error = { + name: error.name, + message: error.message, + stack: error.stack, + }; + } + this.logger.error(`Exception in span: ${error.message}`, error.stack); } - } - - /** - * Record an exception in a span - */ - recordException(span: Span, error: Error): void { - span.recordException(error); - span.setStatus({ - code: SpanStatusCode.ERROR, - message: error.message, - }); - - const spanContext = span.spanContext(); - const traceSpan = this.spans.get(spanContext.spanId); - if (traceSpan) { - traceSpan.status = SpanStatus.ERROR; - traceSpan.attributes.error = { - name: error.name, - message: error.message, - stack: error.stack, - }; + /** + * Add an event to a span + */ + addSpanEvent(span: Span, name: string, attributes?: Record): void { + span.addEvent(name, attributes); + const spanContext = span.spanContext(); + const traceSpan = this.spans.get(spanContext.spanId); + if (traceSpan) { + const event: SpanEvent = { + name, + timestamp: new Date(), + attributes, + }; + traceSpan.events.push(event); + } } - - this.logger.error(`Exception in span: ${error.message}`, error.stack); - } - - /** - * Add an event to a span - */ - addSpanEvent(span: Span, name: string, attributes?: Record): void { - span.addEvent(name, attributes); - - const spanContext = span.spanContext(); - const traceSpan = this.spans.get(spanContext.spanId); - if (traceSpan) { - const event: SpanEvent = { - name, - timestamp: new Date(), - attributes, - }; - traceSpan.events.push(event); + /** + * Execute function within a span + */ + async executeInSpan(name: string, fn: (span: Span) => Promise, attributes?: Record): Promise { + const span = this.startSpan(name, attributes); + try { + const result = await fn(span); + this.endSpan(span, SpanStatus.OK); + return result; + } + catch (error) { + this.recordException(span, error as Error); + this.endSpan(span, SpanStatus.ERROR); + throw error; + } } - } - - /** - * Execute function within a span - */ - async executeInSpan( - name: string, - fn: (span: Span) => Promise, - attributes?: Record, - ): Promise { - const span = this.startSpan(name, attributes); - - try { - const result = await fn(span); - this.endSpan(span, SpanStatus.OK); - return result; - } catch (error) { - this.recordException(span, error as Error); - this.endSpan(span, SpanStatus.ERROR); - throw error; + /** + * Get current active span + */ + getCurrentSpan(): Span | undefined { + return trace.getActiveSpan(); + } + /** + * Create a child span from parent + */ + createChildSpan(parentSpan: Span, name: string, attributes?: Record): Span { + const ctx = trace.setSpan(context.active(), parentSpan); + return context.with(ctx, () => { + const childSpan = this.startSpan(name, attributes); + const parentContext = parentSpan.spanContext(); + const childContext = childSpan.spanContext(); + const childTraceSpan = this.spans.get(childContext.spanId); + if (childTraceSpan) { + childTraceSpan.parentSpanId = parentContext.spanId; + } + return childSpan; + }); + } + /** + * Inject trace context into headers for distributed calls + */ + injectTraceContext(span: Span): Record { + const spanContext = span.spanContext(); + return { + 'x-trace-id': spanContext.traceId, + 'x-span-id': spanContext.spanId, + 'x-trace-flags': spanContext.traceFlags.toString(), + }; + } + /** + * Extract trace context from headers + */ + extractTraceContext(headers: Record): { + traceId?: string; + spanId?: string; + traceFlags?: number; + } { + return { + traceId: headers['x-trace-id'], + spanId: headers['x-span-id'], + traceFlags: headers['x-trace-flags'] ? parseInt(headers['x-trace-flags']) : undefined, + }; + } + /** + * Get trace by ID + */ + getTraceById(traceId: string): TraceSpan[] { + return Array.from(this.spans.values()).filter((span) => span.traceId === traceId); + } + /** + * Get span by ID + */ + getSpanById(spanId: string): TraceSpan | undefined { + return this.spans.get(spanId); + } + /** + * Get all spans + */ + getAllSpans(): TraceSpan[] { + return Array.from(this.spans.values()); + } + /** + * Clear old spans + */ + clearOldSpans(olderThan: Date): number { + let cleared = 0; + this.spans.forEach((span, spanId) => { + if (span.startTime < olderThan) { + this.spans.delete(spanId); + cleared++; + } + }); + this.logger.log(`Cleared ${cleared} old spans`); + return cleared; + } + /** + * Get trace statistics + */ + getTraceStatistics() { + const spans = Array.from(this.spans.values()); + const completedSpans = spans.filter((s) => s.endTime); + const durations = completedSpans.filter((s) => s.duration).map((s) => s.duration ?? 0); + return { + total: spans.length, + completed: completedSpans.length, + active: spans.length - completedSpans.length, + avgDuration: durations.length > 0 ? durations.reduce((a, b) => a + b, 0) / durations.length : 0, + minDuration: durations.length > 0 ? Math.min(...durations) : 0, + maxDuration: durations.length > 0 ? Math.max(...durations) : 0, + errorCount: spans.filter((s) => s.status === SpanStatus.ERROR).length, + }; + } + /** + * Trace HTTP request + */ + async traceHttpRequest(method: string, url: string, fn: (span: Span) => Promise): Promise { + return this.executeInSpan(`HTTP ${method} ${url}`, fn, { + 'http.method': method, + 'http.url': url, + 'span.kind': 'client', + }); + } + /** + * Trace database query + */ + async traceDatabaseQuery(query: string, fn: (span: Span) => Promise): Promise { + return this.executeInSpan('Database Query', fn, { + 'db.statement': query, + 'db.system': 'postgresql', + 'span.kind': 'client', + }); + } + /** + * Trace external service call + */ + async traceExternalCall(serviceName: string, operation: string, fn: (span: Span) => Promise): Promise { + return this.executeInSpan(`${serviceName}.${operation}`, fn, { + 'service.name': serviceName, + operation, + 'span.kind': 'client', + }); } - } - - /** - * Get current active span - */ - getCurrentSpan(): Span | undefined { - return trace.getActiveSpan(); - } - - /** - * Create a child span from parent - */ - createChildSpan(parentSpan: Span, name: string, attributes?: Record): Span { - const ctx = trace.setSpan(context.active(), parentSpan); - - return context.with(ctx, () => { - const childSpan = this.startSpan(name, attributes); - - const parentContext = parentSpan.spanContext(); - const childContext = childSpan.spanContext(); - const childTraceSpan = this.spans.get(childContext.spanId); - - if (childTraceSpan) { - childTraceSpan.parentSpanId = parentContext.spanId; - } - - return childSpan; - }); - } - - /** - * Inject trace context into headers for distributed calls - */ - injectTraceContext(span: Span): Record { - const spanContext = span.spanContext(); - return { - 'x-trace-id': spanContext.traceId, - 'x-span-id': spanContext.spanId, - 'x-trace-flags': spanContext.traceFlags.toString(), - }; - } - - /** - * Extract trace context from headers - */ - extractTraceContext(headers: Record): { - traceId?: string; - spanId?: string; - traceFlags?: number; - } { - return { - traceId: headers['x-trace-id'], - spanId: headers['x-span-id'], - traceFlags: headers['x-trace-flags'] ? parseInt(headers['x-trace-flags']) : undefined, - }; - } - - /** - * Get trace by ID - */ - getTraceById(traceId: string): TraceSpan[] { - return Array.from(this.spans.values()).filter((span) => span.traceId === traceId); - } - - /** - * Get span by ID - */ - getSpanById(spanId: string): TraceSpan | undefined { - return this.spans.get(spanId); - } - - /** - * Get all spans - */ - getAllSpans(): TraceSpan[] { - return Array.from(this.spans.values()); - } - - /** - * Clear old spans - */ - clearOldSpans(olderThan: Date): number { - let cleared = 0; - this.spans.forEach((span, spanId) => { - if (span.startTime < olderThan) { - this.spans.delete(spanId); - cleared++; - } - }); - this.logger.log(`Cleared ${cleared} old spans`); - return cleared; - } - - /** - * Get trace statistics - */ - getTraceStatistics() { - const spans = Array.from(this.spans.values()); - const completedSpans = spans.filter((s) => s.endTime); - - const durations = completedSpans.filter((s) => s.duration).map((s) => s.duration ?? 0); - - return { - total: spans.length, - completed: completedSpans.length, - active: spans.length - completedSpans.length, - avgDuration: - durations.length > 0 ? durations.reduce((a, b) => a + b, 0) / durations.length : 0, - minDuration: durations.length > 0 ? Math.min(...durations) : 0, - maxDuration: durations.length > 0 ? Math.max(...durations) : 0, - errorCount: spans.filter((s) => s.status === SpanStatus.ERROR).length, - }; - } - - /** - * Trace HTTP request - */ - async traceHttpRequest( - method: string, - url: string, - fn: (span: Span) => Promise, - ): Promise { - return this.executeInSpan(`HTTP ${method} ${url}`, fn, { - 'http.method': method, - 'http.url': url, - 'span.kind': 'client', - }); - } - - /** - * Trace database query - */ - async traceDatabaseQuery(query: string, fn: (span: Span) => Promise): Promise { - return this.executeInSpan('Database Query', fn, { - 'db.statement': query, - 'db.system': 'postgresql', - 'span.kind': 'client', - }); - } - - /** - * Trace external service call - */ - async traceExternalCall( - serviceName: string, - operation: string, - fn: (span: Span) => Promise, - ): Promise { - return this.executeInSpan(`${serviceName}.${operation}`, fn, { - 'service.name': serviceName, - operation, - 'span.kind': 'client', - }); - } } diff --git a/src/orchestration/discovery/service-discovery.service.ts b/src/orchestration/discovery/service-discovery.service.ts index e0a1ed51..a3ae1a67 100644 --- a/src/orchestration/discovery/service-discovery.service.ts +++ b/src/orchestration/discovery/service-discovery.service.ts @@ -1,31 +1,26 @@ import { Injectable } from '@nestjs/common'; - interface RegisteredService { - name: string; - baseUrl: string; - healthy: boolean; + name: string; + baseUrl: string; + healthy: boolean; } - @Injectable() export class ServiceDiscoveryService { - private services = new Map(); - - register(service: RegisteredService) { - this.services.set(service.name, service); - } - - async getService(name: string): Promise { - const service = this.services.get(name); - if (!service || !service.healthy) { - throw new Error(`Service ${name} unavailable`); + private services = new Map(); + register(service: RegisteredService) { + this.services.set(service.name, service); } - return service; - } - - async markUnhealthy(name: string) { - const service = this.services.get(name); - if (service) { - service.healthy = false; + async getService(name: string): Promise { + const service = this.services.get(name); + if (!service || !service.healthy) { + throw new Error(`Service ${name} unavailable`); + } + return service; + } + async markUnhealthy(name: string) { + const service = this.services.get(name); + if (service) { + service.healthy = false; + } } - } } diff --git a/src/orchestration/health/health-checker.service.ts b/src/orchestration/health/health-checker.service.ts index 78894eea..315032f9 100644 --- a/src/orchestration/health/health-checker.service.ts +++ b/src/orchestration/health/health-checker.service.ts @@ -1,20 +1,18 @@ import { Injectable } from '@nestjs/common'; import { ServiceDiscoveryService } from '../discovery/service-discovery.service'; import axios from 'axios'; - @Injectable() export class HealthCheckerService { - constructor(private readonly discovery: ServiceDiscoveryService) {} - - async check(serviceName: string) { - const service = await this.discovery.getService(serviceName); - - try { - await axios.get(`${service.baseUrl}/health`); - return true; - } catch { - await this.discovery.markUnhealthy(serviceName); - return false; + constructor(private readonly discovery: ServiceDiscoveryService) { } + async check(serviceName: string) { + const service = await this.discovery.getService(serviceName); + try { + await axios.get(`${service.baseUrl}/health`); + return true; + } + catch { + await this.discovery.markUnhealthy(serviceName); + return false; + } } - } } diff --git a/src/orchestration/locks/distributed-lock.service.ts b/src/orchestration/locks/distributed-lock.service.ts index 219c9dff..fafae8d7 100644 --- a/src/orchestration/locks/distributed-lock.service.ts +++ b/src/orchestration/locks/distributed-lock.service.ts @@ -1,22 +1,18 @@ import { Injectable } from '@nestjs/common'; import Redis from 'ioredis'; - @Injectable() export class DistributedLockService { - private redis = new Redis(process.env.REDIS_URL); - - constructor() { - this.redis.on('error', () => { - // Prevent unhandled error events during Redis outages. - }); - } - - async acquireLock(key: string, ttl = 5000): Promise { - const result = await this.redis.set(key, 'locked', 'PX', ttl, 'NX'); - return result === 'OK'; - } - - async releaseLock(key: string): Promise { - await this.redis.del(key); - } + private redis = new Redis(process.env.REDIS_URL); + constructor() { + this.redis.on('error', () => { + // Prevent unhandled error events during Redis outages. + }); + } + async acquireLock(key: string, ttl = 5000): Promise { + const result = await this.redis.set(key, 'locked', 'PX', ttl, 'NX'); + return result === 'OK'; + } + async releaseLock(key: string): Promise { + await this.redis.del(key); + } } diff --git a/src/orchestration/orchestration.module.ts b/src/orchestration/orchestration.module.ts index bf610c06..7860d589 100644 --- a/src/orchestration/orchestration.module.ts +++ b/src/orchestration/orchestration.module.ts @@ -5,23 +5,23 @@ import { WorkflowEngineService } from './workflow/workflow-engine.service'; import { DistributedLockService } from './locks/distributed-lock.service'; import { ServiceDiscoveryService } from './discovery/service-discovery.service'; import { HealthCheckerService } from './health/health-checker.service'; - @Global() @Module({ - imports: [HttpModule], - providers: [ - ServiceMeshService, - WorkflowEngineService, - DistributedLockService, - ServiceDiscoveryService, - HealthCheckerService, - ], - exports: [ - ServiceMeshService, - WorkflowEngineService, - DistributedLockService, - ServiceDiscoveryService, - HealthCheckerService, - ], + imports: [HttpModule], + providers: [ + ServiceMeshService, + WorkflowEngineService, + DistributedLockService, + ServiceDiscoveryService, + HealthCheckerService, + ], + exports: [ + ServiceMeshService, + WorkflowEngineService, + DistributedLockService, + ServiceDiscoveryService, + HealthCheckerService, + ], }) -export class OrchestrationModule {} +export class OrchestrationModule { +} diff --git a/src/orchestration/service-mesh/service-mesh.service.spec.ts b/src/orchestration/service-mesh/service-mesh.service.spec.ts index 40d5f993..d3e5bc4b 100644 --- a/src/orchestration/service-mesh/service-mesh.service.spec.ts +++ b/src/orchestration/service-mesh/service-mesh.service.spec.ts @@ -1,24 +1,18 @@ import { ServiceMeshService } from './service-mesh.service'; import { of } from 'rxjs'; - describe('ServiceMeshService', () => { - it('propagates correlation ID to external API call headers', async () => { - const serviceDiscovery: any = { - getService: jest.fn().mockResolvedValue({ baseUrl: 'http://localhost' }), - markUnhealthy: jest.fn(), - }; - const httpService: any = { - request: jest.fn().mockReturnValue(of({ data: { ok: true } })), - }; - - const service = new ServiceMeshService(serviceDiscovery, httpService); - - await expect(service.request('dummy', '/ping', 'GET')).resolves.toEqual({ ok: true }); - - expect(httpService.request).toHaveBeenCalledWith( - expect.objectContaining({ - headers: expect.objectContaining({ 'x-request-id': expect.any(String) }), - }), - ); - }); + it('propagates correlation ID to external API call headers', async () => { + const serviceDiscovery: unknown = { + getService: jest.fn().mockResolvedValue({ baseUrl: 'http://localhost' }), + markUnhealthy: jest.fn(), + }; + const httpService: unknown = { + request: jest.fn().mockReturnValue(of({ data: { ok: true } })), + }; + const service = new ServiceMeshService(serviceDiscovery, httpService); + await expect(service.request('dummy', '/ping', 'GET')).resolves.toEqual({ ok: true }); + expect(httpService.request).toHaveBeenCalledWith(expect.objectContaining({ + headers: expect.objectContaining({ 'x-request-id': expect.any(String) }), + })); + }); }); diff --git a/src/orchestration/service-mesh/service-mesh.service.ts b/src/orchestration/service-mesh/service-mesh.service.ts index f439a2ba..92336af6 100644 --- a/src/orchestration/service-mesh/service-mesh.service.ts +++ b/src/orchestration/service-mesh/service-mesh.service.ts @@ -3,44 +3,27 @@ import { HttpService } from '@nestjs/axios'; import { firstValueFrom } from 'rxjs'; import { AxiosResponse } from 'axios'; import { ServiceDiscoveryService } from '../discovery/service-discovery.service'; -import { - injectCorrelationIdToHeaders, - getCorrelationId, -} from '../../common/utils/correlation.utils'; - +import { injectCorrelationIdToHeaders, getCorrelationId, } from '../../common/utils/correlation.utils'; @Injectable() export class ServiceMeshService { - constructor( - private readonly discovery: ServiceDiscoveryService, - private readonly httpService: HttpService, - ) {} - - async request( - serviceName: string, - path: string, - method: 'GET' | 'POST' | 'PUT' | 'DELETE', - data?: any, - ): Promise { - const service = await this.discovery.getService(serviceName); - const url = `${service.baseUrl}${path}`; - - const correlationId = getCorrelationId(); - - try { - const response: AxiosResponse = await firstValueFrom( - this.httpService.request({ - url, - method, - data, - timeout: 5000, - headers: injectCorrelationIdToHeaders(undefined, correlationId), - }), - ); - - return response.data; - } catch (error) { - await this.discovery.markUnhealthy(serviceName); - throw error; + constructor(private readonly discovery: ServiceDiscoveryService, private readonly httpService: HttpService) { } + async request(serviceName: string, path: string, method: 'GET' | 'POST' | 'PUT' | 'DELETE', data?: unknown): Promise { + const service = await this.discovery.getService(serviceName); + const url = `${service.baseUrl}${path}`; + const correlationId = getCorrelationId(); + try { + const response: AxiosResponse = await firstValueFrom(this.httpService.request({ + url, + method, + data, + timeout: 5000, + headers: injectCorrelationIdToHeaders(undefined, correlationId), + })); + return response.data; + } + catch (error) { + await this.discovery.markUnhealthy(serviceName); + throw error; + } } - } } diff --git a/src/orchestration/workflow/workflow-engine.service.ts b/src/orchestration/workflow/workflow-engine.service.ts index 2cc39c2b..7beef31f 100644 --- a/src/orchestration/workflow/workflow-engine.service.ts +++ b/src/orchestration/workflow/workflow-engine.service.ts @@ -1,33 +1,28 @@ import { Injectable, Logger } from '@nestjs/common'; - interface WorkflowStep { - name: string; - execute: () => Promise; - compensate?: () => Promise; + name: string; + execute: () => Promise; + compensate?: () => Promise; } - @Injectable() export class WorkflowEngineService { - private readonly logger = new Logger(WorkflowEngineService.name); - - async executeWorkflow(steps: WorkflowStep[]) { - const completedSteps: WorkflowStep[] = []; - - try { - for (const step of steps) { - await step.execute(); - completedSteps.push(step); - } - } catch (error) { - this.logger.error('Workflow failed. Triggering compensation.'); - - for (const step of completedSteps.reverse()) { - if (step.compensate) { - await step.compensate(); + private readonly logger = new Logger(WorkflowEngineService.name); + async executeWorkflow(steps: WorkflowStep[]) { + const completedSteps: WorkflowStep[] = []; + try { + for (const step of steps) { + await step.execute(); + completedSteps.push(step); + } + } + catch (error) { + this.logger.error('Workflow failed. Triggering compensation.'); + for (const step of completedSteps.reverse()) { + if (step.compensate) { + await step.compensate(); + } + } + throw error; } - } - - throw error; } - } } diff --git a/src/payments/dto/create-payment.dto.ts b/src/payments/dto/create-payment.dto.ts index 66c1e1e2..6775ea9f 100644 --- a/src/payments/dto/create-payment.dto.ts +++ b/src/payments/dto/create-payment.dto.ts @@ -1,37 +1,22 @@ -import { - IsString, - IsNumber, - IsOptional, - IsEnum, - IsPositive, - IsUUID, - Min, - Max, -} from 'class-validator'; +import { IsString, IsNumber, IsOptional, IsEnum, IsPositive, IsUUID, Min, Max, } from 'class-validator'; import { PaymentMethod } from '../entities/payment.entity'; - export class CreatePaymentDto { - @IsUUID('4', { message: 'courseId must be a valid UUID v4' }) - courseId: string; - - @IsNumber({}, { message: 'Amount must be a numeric value' }) - @IsPositive({ message: 'Amount must be strictly positive' }) - @Min(0.5, { message: 'Minimum checkout amount is 0.5' }) - @Max(1000000, { message: 'Amount exceeds maximum limit' }) - amount: number; - - @IsString({ message: 'Currency must be a string code' }) - @IsOptional() - currency?: string = 'USD'; - - @IsEnum(PaymentMethod, { message: 'Invalid payment method selected' }) - @IsOptional() - method?: PaymentMethod; - - @IsString() - @IsOptional() - provider?: string = 'stripe'; - - @IsOptional() - metadata?: Record; + @IsUUID('4', { message: 'courseId must be a valid UUID v4' }) + courseId: string; + @IsNumber({}, { message: 'Amount must be a numeric value' }) + @IsPositive({ message: 'Amount must be strictly positive' }) + @Min(0.5, { message: 'Minimum checkout amount is 0.5' }) + @Max(1000000, { message: 'Amount exceeds maximum limit' }) + amount: number; + @IsString({ message: 'Currency must be a string code' }) + @IsOptional() + currency?: string = 'USD'; + @IsEnum(PaymentMethod, { message: 'Invalid payment method selected' }) + @IsOptional() + method?: PaymentMethod; + @IsString() + @IsOptional() + provider?: string = 'stripe'; + @IsOptional() + metadata?: Record; } diff --git a/src/payments/dto/create-subscription.dto.ts b/src/payments/dto/create-subscription.dto.ts index 51528575..a7e0ab46 100644 --- a/src/payments/dto/create-subscription.dto.ts +++ b/src/payments/dto/create-subscription.dto.ts @@ -1,27 +1,22 @@ import { IsString, IsEnum, IsOptional, IsNotEmpty, IsUUID } from 'class-validator'; import { PaymentMethod } from '../entities/payment.entity'; import { SubscriptionInterval } from '../entities/subscription.entity'; - export class CreateSubscriptionDto { - @IsString() - @IsNotEmpty() - @IsUUID() - courseId: string; - - @IsEnum(SubscriptionInterval) - @IsNotEmpty() - interval: SubscriptionInterval; - - @IsEnum(PaymentMethod) - @IsNotEmpty() - provider: PaymentMethod; - - @IsString() - @IsNotEmpty() - @IsUUID() - priceId: string; - - @IsOptional() - @IsString() - metadata?: Record; + @IsString() + @IsNotEmpty() + @IsUUID() + courseId: string; + @IsEnum(SubscriptionInterval) + @IsNotEmpty() + interval: SubscriptionInterval; + @IsEnum(PaymentMethod) + @IsNotEmpty() + provider: PaymentMethod; + @IsString() + @IsNotEmpty() + @IsUUID() + priceId: string; + @IsOptional() + @IsString() + metadata?: Record; } diff --git a/src/payments/dto/refund.dto.ts b/src/payments/dto/refund.dto.ts index dd9edb06..94ef09c8 100644 --- a/src/payments/dto/refund.dto.ts +++ b/src/payments/dto/refund.dto.ts @@ -1,39 +1,24 @@ -import { - IsString, - IsNumber, - IsOptional, - IsDateString, - IsEnum, - IsNotEmpty, - IsUUID, - IsDate, -} from 'class-validator'; +import { IsString, IsNumber, IsOptional, IsDateString, IsEnum, IsNotEmpty, IsUUID, IsDate, } from 'class-validator'; import { RefundStatus } from '../entities/refund.entity'; - export class RefundDto { - @IsString() - @IsNotEmpty() - @IsUUID() - paymentId: string; - - @IsString() - @IsNotEmpty() - reason: string; - - @IsNumber() - @IsOptional() - amount?: number; - - @IsString() - @IsOptional() - refundMethod?: string; - - @IsOptional() - @IsDateString() - @IsDate() - refundDate?: Date; - - @IsEnum(RefundStatus) - @IsOptional() - status?: RefundStatus; + @IsString() + @IsNotEmpty() + @IsUUID() + paymentId: string; + @IsString() + @IsNotEmpty() + reason: string; + @IsNumber() + @IsOptional() + amount?: number; + @IsString() + @IsOptional() + refundMethod?: string; + @IsOptional() + @IsDateString() + @IsDate() + refundDate?: Date; + @IsEnum(RefundStatus) + @IsOptional() + status?: RefundStatus; } diff --git a/src/payments/dto/update-payment.dto.ts b/src/payments/dto/update-payment.dto.ts index 0293332c..a214c4e7 100644 --- a/src/payments/dto/update-payment.dto.ts +++ b/src/payments/dto/update-payment.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/mapped-types'; import { CreatePaymentDto } from './create-payment.dto'; - -export class UpdatePaymentDto extends PartialType(CreatePaymentDto) {} +export class UpdatePaymentDto extends PartialType(CreatePaymentDto) { +} diff --git a/src/payments/entities/invoice.entity.ts b/src/payments/entities/invoice.entity.ts index ca1c671c..fc4f704a 100644 --- a/src/payments/entities/invoice.entity.ts +++ b/src/payments/entities/invoice.entity.ts @@ -1,98 +1,69 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { Payment } from './payment.entity'; import { User } from '../../users/entities/user.entity'; - export enum InvoiceStatus { - DRAFT = 'draft', - SENT = 'sent', - PAID = 'paid', - OVERDUE = 'overdue', - CANCELLED = 'cancelled', + DRAFT = 'draft', + SENT = 'sent', + PAID = 'paid', + OVERDUE = 'overdue', + CANCELLED = 'cancelled' } - interface InvoiceItem { - description: string; - amount: number; - quantity: number; - taxRate?: number; + description: string; + amount: number; + quantity: number; + taxRate?: number; } - @Entity('invoices') export class Invoice { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'varchar', unique: true }) - @Index() - invoiceNumber: string; - - @Column({ type: 'decimal', precision: 10, scale: 2 }) - @Index() - amount: number; - - @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) - taxAmount: number; - - @Column({ type: 'decimal', precision: 10, scale: 2 }) - @Index() - totalAmount: number; - - @Column({ type: 'varchar', length: 3, default: 'USD' }) - currency: string; - - @Column({ type: 'jsonb' }) - items: InvoiceItem[]; - - @Column({ - type: 'enum', - enum: InvoiceStatus, - default: InvoiceStatus.DRAFT, - }) - @Index() - status: InvoiceStatus; - - @Column({ type: 'date', nullable: true }) - @Index() - issuedDate: Date; - - @Column({ type: 'date', nullable: true }) - @Index() - dueDate: Date; - - @Column({ type: 'text', nullable: true }) - notes: string; - - @Column({ type: 'text', nullable: true }) - terms: string; - - @ManyToOne(() => Payment, (payment) => payment.id) - @JoinColumn({ name: 'payment_id' }) - payment: Payment; - - @Column({ name: 'payment_id' }) - @Index() - paymentId: string; - - @ManyToOne(() => User, (user) => user.id) - @JoinColumn({ name: 'user_id' }) - user: User; - - @Column({ name: 'user_id' }) - @Index() - userId: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'varchar', unique: true }) + @Index() + invoiceNumber: string; + @Column({ type: 'decimal', precision: 10, scale: 2 }) + @Index() + amount: number; + @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) + taxAmount: number; + @Column({ type: 'decimal', precision: 10, scale: 2 }) + @Index() + totalAmount: number; + @Column({ type: 'varchar', length: 3, default: 'USD' }) + currency: string; + @Column({ type: 'jsonb' }) + items: InvoiceItem[]; + @Column({ + type: 'enum', + enum: InvoiceStatus, + default: InvoiceStatus.DRAFT, + }) + @Index() + status: InvoiceStatus; + @Column({ type: 'date', nullable: true }) + @Index() + issuedDate: Date; + @Column({ type: 'date', nullable: true }) + @Index() + dueDate: Date; + @Column({ type: 'text', nullable: true }) + notes: string; + @Column({ type: 'text', nullable: true }) + terms: string; + @ManyToOne(() => Payment, (payment) => payment.id) + @JoinColumn({ name: 'payment_id' }) + payment: Payment; + @Column({ name: 'payment_id' }) + @Index() + paymentId: string; + @ManyToOne(() => User, (user) => user.id) + @JoinColumn({ name: 'user_id' }) + user: User; + @Column({ name: 'user_id' }) + @Index() + userId: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/payments/entities/payment.entity.ts b/src/payments/entities/payment.entity.ts index d4165fe7..5595760b 100644 --- a/src/payments/entities/payment.entity.ts +++ b/src/payments/entities/payment.entity.ts @@ -1,86 +1,59 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; import { Course } from '../../courses/entities/course.entity'; - export enum PaymentStatus { - PENDING = 'pending', - PROCESSING = 'processing', - COMPLETED = 'completed', - FAILED = 'failed', - REFUNDED = 'refunded', - CANCELLED = 'cancelled', + PENDING = 'pending', + PROCESSING = 'processing', + COMPLETED = 'completed', + FAILED = 'failed', + REFUNDED = 'refunded', + CANCELLED = 'cancelled' } - export enum PaymentMethod { - CREDIT_CARD = 'credit_card', - BANK_TRANSFER = 'bank_transfer', - PAYPAL = 'paypal', - CRYPTO = 'crypto', - WALLET = 'wallet', + CREDIT_CARD = 'credit_card', + BANK_TRANSFER = 'bank_transfer', + PAYPAL = 'paypal', + CRYPTO = 'crypto', + WALLET = 'wallet' } - @Entity('payments') @Index(['userId', 'status']) export class Payment { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'decimal', precision: 10, scale: 2 }) - amount: number; - - @Column({ type: 'varchar', length: 3, default: 'USD' }) - currency: string; - - @Column({ type: 'enum', enum: PaymentStatus, default: PaymentStatus.PENDING }) - @Index() - status: PaymentStatus; - - @Column({ type: 'enum', enum: PaymentMethod }) - method: PaymentMethod; - - @Column({ type: 'varchar', nullable: true }) - provider: string; // 'stripe', 'paypal', etc. - - @Column({ type: 'varchar', nullable: true }) - providerPaymentId: string; // External payment ID from provider - - @Column({ type: 'jsonb', nullable: true }) - metadata: Record; - - @ManyToOne(() => User, (user) => user.courses) - @JoinColumn({ name: 'user_id' }) - user: User; - - @Column({ name: 'user_id' }) - @Index() - userId: string; - - @ManyToOne(() => Course, (course) => course.id) - @JoinColumn({ name: 'course_id' }) - course: Course; - - @Column({ name: 'course_id', nullable: true }) - @Index() - courseId: string; - - @Column({ type: 'boolean', default: false }) - isSubscription: boolean; - - @Column({ type: 'varchar', nullable: true }) - subscriptionId: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'decimal', precision: 10, scale: 2 }) + amount: number; + @Column({ type: 'varchar', length: 3, default: 'USD' }) + currency: string; + @Column({ type: 'enum', enum: PaymentStatus, default: PaymentStatus.PENDING }) + @Index() + status: PaymentStatus; + @Column({ type: 'enum', enum: PaymentMethod }) + method: PaymentMethod; + @Column({ type: 'varchar', nullable: true }) + provider: string; // 'stripe', 'paypal', etc. + @Column({ type: 'varchar', nullable: true }) + providerPaymentId: string; // External payment ID from provider + @Column({ type: 'jsonb', nullable: true }) + metadata: Record; + @ManyToOne(() => User, (user) => user.courses) + @JoinColumn({ name: 'user_id' }) + user: User; + @Column({ name: 'user_id' }) + @Index() + userId: string; + @ManyToOne(() => Course, (course) => course.id) + @JoinColumn({ name: 'course_id' }) + course: Course; + @Column({ name: 'course_id', nullable: true }) + @Index() + courseId: string; + @Column({ type: 'boolean', default: false }) + isSubscription: boolean; + @Column({ type: 'varchar', nullable: true }) + subscriptionId: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/payments/entities/refund.entity.ts b/src/payments/entities/refund.entity.ts index 1053a01b..516e6d77 100644 --- a/src/payments/entities/refund.entity.ts +++ b/src/payments/entities/refund.entity.ts @@ -1,55 +1,35 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, - JoinColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, ManyToOne, JoinColumn, } from 'typeorm'; import { Payment } from './payment.entity'; - export enum RefundStatus { - PENDING = 'pending', - APPROVED = 'approved', - REJECTED = 'rejected', - PROCESSED = 'processed', - FAILED = 'failed', + PENDING = 'pending', + APPROVED = 'approved', + REJECTED = 'rejected', + PROCESSED = 'processed', + FAILED = 'failed' } - @Entity('refunds') export class Refund { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'decimal', precision: 10, scale: 2, nullable: true }) - amount: number; - - @Column({ type: 'text', nullable: true }) - reason: string; - - @Column({ type: 'varchar', nullable: true }) - refundMethod: string; // 'original_method', 'store_credit', etc. - - @Column({ type: 'enum', enum: RefundStatus, default: RefundStatus.PENDING }) - status: RefundStatus; - - @Column({ type: 'varchar', nullable: true }) - providerRefundId: string; // External refund ID from payment provider - - @Column({ type: 'jsonb', nullable: true }) - metadata: Record; - - @ManyToOne(() => Payment, (payment) => payment.id) - @JoinColumn({ name: 'payment_id' }) - payment: Payment; - - @Column({ name: 'payment_id' }) - paymentId: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'decimal', precision: 10, scale: 2, nullable: true }) + amount: number; + @Column({ type: 'text', nullable: true }) + reason: string; + @Column({ type: 'varchar', nullable: true }) + refundMethod: string; // 'original_method', 'store_credit', etc. + @Column({ type: 'enum', enum: RefundStatus, default: RefundStatus.PENDING }) + status: RefundStatus; + @Column({ type: 'varchar', nullable: true }) + providerRefundId: string; // External refund ID from payment provider + @Column({ type: 'jsonb', nullable: true }) + metadata: Record; + @ManyToOne(() => Payment, (payment) => payment.id) + @JoinColumn({ name: 'payment_id' }) + payment: Payment; + @Column({ name: 'payment_id' }) + paymentId: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/payments/entities/subscription.entity.ts b/src/payments/entities/subscription.entity.ts index e7b41585..04811a0c 100644 --- a/src/payments/entities/subscription.entity.ts +++ b/src/payments/entities/subscription.entity.ts @@ -1,82 +1,55 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { User } from '../../users/entities/user.entity'; - export enum SubscriptionStatus { - ACTIVE = 'active', - CANCELLED = 'cancelled', - PAST_DUE = 'past_due', - UNPAID = 'unpaid', - TRIALING = 'trialing', - INCOMPLETE = 'incomplete', + ACTIVE = 'active', + CANCELLED = 'cancelled', + PAST_DUE = 'past_due', + UNPAID = 'unpaid', + TRIALING = 'trialing', + INCOMPLETE = 'incomplete' } - export enum SubscriptionInterval { - MONTHLY = 'monthly', - YEARLY = 'yearly', - QUARTERLY = 'quarterly', - WEEKLY = 'weekly', + MONTHLY = 'monthly', + YEARLY = 'yearly', + QUARTERLY = 'quarterly', + WEEKLY = 'weekly' } - @Entity('subscriptions') @Index(['userId', 'status']) export class Subscription { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'varchar', unique: true, nullable: true }) - providerSubscriptionId: string; - - @Column({ type: 'enum', enum: SubscriptionStatus, default: SubscriptionStatus.ACTIVE }) - @Index() - status: SubscriptionStatus; - - @Column({ type: 'enum', enum: SubscriptionInterval }) - interval: SubscriptionInterval; - - @Column({ type: 'decimal', precision: 10, scale: 2 }) - amount: number; - - @Column({ type: 'varchar', length: 3, default: 'USD' }) - currency: string; - - @Column({ type: 'timestamp', nullable: true }) - currentPeriodStart: Date; - - @Column({ type: 'timestamp', nullable: true }) - currentPeriodEnd: Date; - - @Column({ type: 'boolean', default: false }) - cancelAtPeriodEnd: boolean; - - @Column({ type: 'timestamp', nullable: true }) - cancelledAt: Date; - - @Column({ type: 'timestamp', nullable: true }) - trialStart: Date; - - @Column({ type: 'timestamp', nullable: true }) - trialEnd: Date; - - @ManyToOne(() => User, (user) => user.courses) - @JoinColumn({ name: 'user_id' }) - user: User; - - @Column({ name: 'user_id' }) - @Index() - userId: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'varchar', unique: true, nullable: true }) + providerSubscriptionId: string; + @Column({ type: 'enum', enum: SubscriptionStatus, default: SubscriptionStatus.ACTIVE }) + @Index() + status: SubscriptionStatus; + @Column({ type: 'enum', enum: SubscriptionInterval }) + interval: SubscriptionInterval; + @Column({ type: 'decimal', precision: 10, scale: 2 }) + amount: number; + @Column({ type: 'varchar', length: 3, default: 'USD' }) + currency: string; + @Column({ type: 'timestamp', nullable: true }) + currentPeriodStart: Date; + @Column({ type: 'timestamp', nullable: true }) + currentPeriodEnd: Date; + @Column({ type: 'boolean', default: false }) + cancelAtPeriodEnd: boolean; + @Column({ type: 'timestamp', nullable: true }) + cancelledAt: Date; + @Column({ type: 'timestamp', nullable: true }) + trialStart: Date; + @Column({ type: 'timestamp', nullable: true }) + trialEnd: Date; + @ManyToOne(() => User, (user) => user.courses) + @JoinColumn({ name: 'user_id' }) + user: User; + @Column({ name: 'user_id' }) + @Index() + userId: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; } diff --git a/src/payments/interfaces/payment-provider.interface.ts b/src/payments/interfaces/payment-provider.interface.ts index 47719ff7..636b2034 100644 --- a/src/payments/interfaces/payment-provider.interface.ts +++ b/src/payments/interfaces/payment-provider.interface.ts @@ -1,61 +1,46 @@ export interface PaymentIntentResult { - paymentIntentId: string; - clientSecret: string; - requiresAction: boolean; + paymentIntentId: string; + clientSecret: string; + requiresAction: boolean; } - export interface RefundResult { - refundId: string; - status: string; + refundId: string; + status: string; } - export interface PaymentMetadata { - userId: string; - courseId: string; - [key: string]: string | number | boolean; + userId: string; + courseId: string; + [key: string]: string | number | boolean; } - export interface PaymentProvider { - createPaymentIntent( - amount: number, - currency: string, - metadata: PaymentMetadata, - ): Promise; - refundPayment(paymentId: string, amount?: number): Promise; - handleWebhook( - payload: Record, - signature: string, - ): Promise>; -} - + createPaymentIntent(amount: number, currency: string, metadata: PaymentMetadata): Promise; + refundPayment(paymentId: string, amount?: number): Promise; + handleWebhook(payload: Record, signature: string): Promise>; +} export interface SubscriptionWebhookEvent { - data: { - object: { - id: string; - status: string; + data: { + object: { + id: string; + status: string; + }; }; - }; } - export interface RefundWebhookData { - id: string; - amount: number; + id: string; + amount: number; } - export interface CreatePaymentIntentResult { - paymentId: string; - clientSecret: string; - requiresAction: boolean; + paymentId: string; + clientSecret: string; + requiresAction: boolean; } - export interface CreateSubscriptionResult { - subscriptionId: string; - status: string; - currentPeriodEnd: Date; + subscriptionId: string; + status: string; + currentPeriodEnd: Date; } - export interface ProcessRefundResult { - refundId: string; - status: string; - amount: number; + refundId: string; + status: string; + amount: number; } diff --git a/src/payments/payments.controller.spec.ts b/src/payments/payments.controller.spec.ts index d559c3f5..b71b4d1f 100644 --- a/src/payments/payments.controller.spec.ts +++ b/src/payments/payments.controller.spec.ts @@ -5,74 +5,54 @@ import { PaymentsController } from './payments.controller'; import { PaymentsService } from './payments.service'; import { expectNotFound, expectUnauthorized, expectValidationFailure } from '../../test/utils'; import { UserRole } from '../users/entities/user.entity'; - describe('PaymentsController', () => { - let controller: PaymentsController; - let paymentsService: { - createPaymentIntent: jest.Mock; - processRefund: jest.Mock; - getInvoice: jest.Mock; - }; - - const request = { user: { id: 'user-1', email: 'test@example.com', role: UserRole.STUDENT } }; - const createPaymentDto: CreatePaymentDto = { - courseId: 'course-1', - amount: 120, - provider: 'stripe', - }; - - beforeEach(async () => { - paymentsService = { - createPaymentIntent: jest.fn(), - processRefund: jest.fn(), - getInvoice: jest.fn(), + let controller: PaymentsController; + let paymentsService: { + createPaymentIntent: jest.Mock; + processRefund: jest.Mock; + getInvoice: jest.Mock; }; - - const module: TestingModule = await Test.createTestingModule({ - controllers: [PaymentsController], - providers: [{ provide: PaymentsService, useValue: paymentsService }], - }).compile(); - - controller = module.get(PaymentsController); - }); - - it('returns payment intent for valid request', async () => { - paymentsService.createPaymentIntent.mockResolvedValue({ - paymentId: 'payment-1', - clientSecret: 'cs_123', - requiresAction: false, + const request = { user: { id: 'user-1', email: 'test@example.com', role: UserRole.STUDENT } }; + const createPaymentDto: CreatePaymentDto = { + courseId: 'course-1', + amount: 120, + provider: 'stripe', + }; + beforeEach(async () => { + paymentsService = { + createPaymentIntent: jest.fn(), + processRefund: jest.fn(), + getInvoice: jest.fn(), + }; + const module: TestingModule = await Test.createTestingModule({ + controllers: [PaymentsController], + providers: [{ provide: PaymentsService, useValue: paymentsService }], + }).compile(); + controller = module.get(PaymentsController); + }); + it('returns payment intent for valid request', async () => { + paymentsService.createPaymentIntent.mockResolvedValue({ + paymentId: 'payment-1', + clientSecret: 'cs_123', + requiresAction: false, + }); + await expect(controller.createPaymentIntent(request, createPaymentDto)).resolves.toMatchObject({ + paymentId: 'payment-1', + clientSecret: 'cs_123', + requiresAction: false, + }); + expect(paymentsService.createPaymentIntent).toHaveBeenCalledWith('user-1', createPaymentDto); + }); + it('returns validation failure for invalid refund request', async () => { + paymentsService.processRefund.mockRejectedValue(new BadRequestException('Invalid refund amount')); + await expectValidationFailure(() => controller.processRefund({ paymentId: 'payment-1', amount: -1, reason: 'duplicate' })); + }); + it('returns not found when invoice is missing', async () => { + paymentsService.getInvoice.mockRejectedValue(new NotFoundException('Payment not found')); + await expectNotFound(() => controller.getInvoice('missing', request)); }); - - await expect(controller.createPaymentIntent(request, createPaymentDto)).resolves.toMatchObject({ - paymentId: 'payment-1', - clientSecret: 'cs_123', - requiresAction: false, + it('returns unauthorized when access token is invalid', async () => { + paymentsService.createPaymentIntent.mockRejectedValue(new UnauthorizedException('Invalid token')); + await expectUnauthorized(() => controller.createPaymentIntent(request, createPaymentDto)); }); - - expect(paymentsService.createPaymentIntent).toHaveBeenCalledWith('user-1', createPaymentDto); - }); - - it('returns validation failure for invalid refund request', async () => { - paymentsService.processRefund.mockRejectedValue( - new BadRequestException('Invalid refund amount'), - ); - - await expectValidationFailure(() => - controller.processRefund({ paymentId: 'payment-1', amount: -1, reason: 'duplicate' }), - ); - }); - - it('returns not found when invoice is missing', async () => { - paymentsService.getInvoice.mockRejectedValue(new NotFoundException('Payment not found')); - - await expectNotFound(() => controller.getInvoice('missing', request)); - }); - - it('returns unauthorized when access token is invalid', async () => { - paymentsService.createPaymentIntent.mockRejectedValue( - new UnauthorizedException('Invalid token'), - ); - - await expectUnauthorized(() => controller.createPaymentIntent(request, createPaymentDto)); - }); }); diff --git a/src/payments/payments.controller.ts b/src/payments/payments.controller.ts index 544301fe..924bde71 100644 --- a/src/payments/payments.controller.ts +++ b/src/payments/payments.controller.ts @@ -12,86 +12,83 @@ import { UserRole } from '../users/entities/user.entity'; import { Payment } from './entities/payment.entity'; import { Subscription } from './entities/subscription.entity'; import { Invoice } from './entities/invoice.entity'; -import { - CreatePaymentIntentResult, - CreateSubscriptionResult, - ProcessRefundResult, -} from './interfaces/payment-provider.interface'; - +import { CreatePaymentIntentResult, CreateSubscriptionResult, ProcessRefundResult, } from './interfaces/payment-provider.interface'; interface AuthenticatedRequest { - user: { - id: string; - email: string; - role: UserRole; - }; + user: { + id: string; + email: string; + role: UserRole; + }; } - @ApiTags('payments') @Controller('payments') @UseGuards(JwtAuthGuard, RolesGuard) export class PaymentsController { - constructor(private readonly paymentsService: PaymentsService) {} - - @Post('create-intent') - @Throttle({ default: { limit: 10, ttl: 3600000 } }) // 10 requests per hour - @Roles(UserRole.STUDENT, UserRole.TEACHER) - @ApiOperation({ summary: 'Create a payment intent for course purchase' }) - @ApiResponse({ status: 201, description: 'Payment intent created' }) - async createPaymentIntent( - @Request() req: AuthenticatedRequest, - @Body() createPaymentDto: CreatePaymentDto, - ): Promise { - return this.paymentsService.createPaymentIntent(req.user.id, createPaymentDto); - } - - @Post('subscriptions') - @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour - @Roles(UserRole.STUDENT, UserRole.TEACHER) - @ApiOperation({ summary: 'Create a subscription for premium course' }) - @ApiResponse({ status: 201, description: 'Subscription created' }) - async createSubscription( - @Request() req: AuthenticatedRequest, - @Body() createSubscriptionDto: CreateSubscriptionDto, - ): Promise { - return this.paymentsService.createSubscription(req.user.id, createSubscriptionDto); - } - - @Post('refund') - @Roles(UserRole.ADMIN, UserRole.TEACHER) - @ApiOperation({ summary: 'Process a refund' }) - @ApiResponse({ status: 200, description: 'Refund processed' }) - async processRefund(@Body() refundDto: RefundDto): Promise { - return this.paymentsService.processRefund(refundDto); - } - - @Get('invoices/:paymentId') - @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) - @ApiOperation({ summary: 'Get invoice for a payment' }) - @ApiResponse({ status: 200, description: 'Invoice retrieved' }) - async getInvoice( - @Param('paymentId') paymentId: string, - @Request() req: AuthenticatedRequest, - ): Promise { - return this.paymentsService.getInvoice(paymentId, req.user.id); - } - - @Get('user/payments') - @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) - @ApiOperation({ summary: 'Get user payment history' }) - @ApiResponse({ status: 200, description: 'Payment history retrieved' }) - async getUserPayments( - @Request() req: AuthenticatedRequest, - @Query('limit') limit: number = 10, - @Query('page') page: number = 1, - ): Promise { - return this.paymentsService.getUserPayments(req.user.id, limit, page); - } - - @Get('user/subscriptions') - @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) - @ApiOperation({ summary: 'Get user subscriptions' }) - @ApiResponse({ status: 200, description: 'Subscriptions retrieved' }) - async getUserSubscriptions(@Request() req: AuthenticatedRequest): Promise { - return this.paymentsService.getUserSubscriptions(req.user.id); - } + constructor(private readonly paymentsService: PaymentsService) { } + @Post('create-intent') + @Throttle({ default: { limit: 10, ttl: 3600000 } }) // 10 requests per hour + @Roles(UserRole.STUDENT, UserRole.TEACHER) + @ApiOperation({ summary: 'Create a payment intent for course purchase' }) + @ApiResponse({ status: 201, description: 'Payment intent created' }) + async createPaymentIntent( + @Request() + req: AuthenticatedRequest, + @Body() + createPaymentDto: CreatePaymentDto): Promise { + return this.paymentsService.createPaymentIntent(req.user.id, createPaymentDto); + } + @Post('subscriptions') + @Throttle({ default: { limit: 5, ttl: 3600000 } }) // 5 requests per hour + @Roles(UserRole.STUDENT, UserRole.TEACHER) + @ApiOperation({ summary: 'Create a subscription for premium course' }) + @ApiResponse({ status: 201, description: 'Subscription created' }) + async createSubscription( + @Request() + req: AuthenticatedRequest, + @Body() + createSubscriptionDto: CreateSubscriptionDto): Promise { + return this.paymentsService.createSubscription(req.user.id, createSubscriptionDto); + } + @Post('refund') + @Roles(UserRole.ADMIN, UserRole.TEACHER) + @ApiOperation({ summary: 'Process a refund' }) + @ApiResponse({ status: 200, description: 'Refund processed' }) + async processRefund( + @Body() + refundDto: RefundDto): Promise { + return this.paymentsService.processRefund(refundDto); + } + @Get('invoices/:paymentId') + @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) + @ApiOperation({ summary: 'Get invoice for a payment' }) + @ApiResponse({ status: 200, description: 'Invoice retrieved' }) + async getInvoice( + @Param('paymentId') + paymentId: string, + @Request() + req: AuthenticatedRequest): Promise { + return this.paymentsService.getInvoice(paymentId, req.user.id); + } + @Get('user/payments') + @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) + @ApiOperation({ summary: 'Get user payment history' }) + @ApiResponse({ status: 200, description: 'Payment history retrieved' }) + async getUserPayments( + @Request() + req: AuthenticatedRequest, + @Query('limit') + limit: number = 10, + @Query('page') + page: number = 1): Promise { + return this.paymentsService.getUserPayments(req.user.id, limit, page); + } + @Get('user/subscriptions') + @Roles(UserRole.STUDENT, UserRole.TEACHER, UserRole.ADMIN) + @ApiOperation({ summary: 'Get user subscriptions' }) + @ApiResponse({ status: 200, description: 'Subscriptions retrieved' }) + async getUserSubscriptions( + @Request() + req: AuthenticatedRequest): Promise { + return this.paymentsService.getUserSubscriptions(req.user.id); + } } diff --git a/src/payments/payments.module.ts b/src/payments/payments.module.ts index d1da1112..2095f758 100644 --- a/src/payments/payments.module.ts +++ b/src/payments/payments.module.ts @@ -22,33 +22,30 @@ import { UsersModule } from '../users/users.module'; import { User } from '../users/entities/user.entity'; import { TransactionService } from '../common/database/transaction.service'; import { TransactionHelperService } from '../common/database/transaction-helper.service'; - @Module({ - imports: [ - TypeOrmModule.forFeature([Payment, Subscription, Invoice, Refund, User, WebhookRetry]), - BullModule.registerQueue( - { - name: QUEUE_NAMES.SUBSCRIPTIONS, - }, - { - name: QUEUE_NAMES.WEBHOOKS, - }, - ), - UsersModule, - ], - controllers: [PaymentsController, WebhookController, WebhookManagementController], - providers: [ - PaymentsService, - WebhookService, - WebhookQueueService, - WebhookRetryProcessor, - SubscriptionsService, - SubscriptionJobProcessor, - StripeService, - ProviderFactoryService, - TransactionService, - TransactionHelperService, - ], - exports: [PaymentsService, ProviderFactoryService, WebhookQueueService], + imports: [ + TypeOrmModule.forFeature([Payment, Subscription, Invoice, Refund, User, WebhookRetry]), + BullModule.registerQueue({ + name: QUEUE_NAMES.SUBSCRIPTIONS, + }, { + name: QUEUE_NAMES.WEBHOOKS, + }), + UsersModule, + ], + controllers: [PaymentsController, WebhookController, WebhookManagementController], + providers: [ + PaymentsService, + WebhookService, + WebhookQueueService, + WebhookRetryProcessor, + SubscriptionsService, + SubscriptionJobProcessor, + StripeService, + ProviderFactoryService, + TransactionService, + TransactionHelperService, + ], + exports: [PaymentsService, ProviderFactoryService, WebhookQueueService], }) -export class PaymentsModule {} +export class PaymentsModule { +} diff --git a/src/payments/payments.service.spec.ts b/src/payments/payments.service.spec.ts index cf278d0d..d6c19ff1 100644 --- a/src/payments/payments.service.spec.ts +++ b/src/payments/payments.service.spec.ts @@ -11,205 +11,166 @@ import { User } from '../users/entities/user.entity'; import { TransactionService } from '../common/database/transaction.service'; import { ProviderFactoryService } from './providers/provider-factory.service'; import { expectNotFound, expectUnauthorized, expectValidationFailure } from '../../test/utils'; - type RepoMock = { - create: jest.Mock; - save: jest.Mock; - findOne: jest.Mock; - find: jest.Mock; - update: jest.Mock; + create: jest.Mock; + save: jest.Mock; + findOne: jest.Mock; + find: jest.Mock; + update: jest.Mock; }; - function createRepositoryMock(): RepoMock { - return { - create: jest.fn(), - save: jest.fn(), - findOne: jest.fn(), - find: jest.fn(), - update: jest.fn(), - }; + return { + create: jest.fn(), + save: jest.fn(), + findOne: jest.fn(), + find: jest.fn(), + update: jest.fn(), + }; } - describe('PaymentsService', () => { - let service: PaymentsService; - let paymentRepository: RepoMock; - let userRepository: RepoMock; - let refundRepository: RepoMock; - let invoiceRepository: RepoMock; - let providerFactoryMock: { getProvider: jest.Mock }; - - const baseCreatePaymentDto: CreatePaymentDto = { - courseId: 'course-1', - amount: 100, - currency: 'USD', - provider: 'stripe', - metadata: { source: 'test' }, - }; - - beforeEach(async () => { - const paymentRepoMock = createRepositoryMock(); - const subscriptionRepoMock = createRepositoryMock(); - const userRepoMock = createRepositoryMock(); - const refundRepoMock = createRepositoryMock(); - const invoiceRepoMock = createRepositoryMock(); - - const mockTransactionService = { - runWithRetry: jest.fn( - (operation: (manager: { create: jest.Mock; save: jest.Mock }) => Promise) => - operation({ - create: jest.fn((_Entity: unknown, data: Record) => ({ - id: 'payment-1', - ...data, - })), - save: jest.fn().mockResolvedValue(undefined), - }), - ), + let service: PaymentsService; + let paymentRepository: RepoMock; + let userRepository: RepoMock; + let refundRepository: RepoMock; + let invoiceRepository: RepoMock; + let providerFactoryMock: { + getProvider: jest.Mock; }; - - providerFactoryMock = { getProvider: jest.fn() }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - PaymentsService, - { - provide: getRepositoryToken(Payment), - useValue: paymentRepoMock, - }, - { - provide: getRepositoryToken(Subscription), - useValue: subscriptionRepoMock, - }, - { - provide: getRepositoryToken(User), - useValue: userRepoMock, - }, - { - provide: getRepositoryToken(Refund), - useValue: refundRepoMock, - }, - { - provide: getRepositoryToken(Invoice), - useValue: invoiceRepoMock, - }, - { - provide: TransactionService, - useValue: mockTransactionService, - }, - { - provide: ProviderFactoryService, - useValue: providerFactoryMock, - }, - ], - }).compile(); - - service = module.get(PaymentsService); - paymentRepository = module.get(getRepositoryToken(Payment)); - userRepository = module.get(getRepositoryToken(User)); - refundRepository = module.get(getRepositoryToken(Refund)); - invoiceRepository = module.get(getRepositoryToken(Invoice)); - }); - - it('creates payment intent for valid user', async () => { - userRepository.findOne.mockResolvedValue({ id: 'user-1' }); - paymentRepository.create.mockReturnValue({ - id: 'payment-1', - ...baseCreatePaymentDto, - status: PaymentStatus.PENDING, - }); - paymentRepository.save.mockResolvedValue(undefined); - - const provider = { - createPaymentIntent: jest.fn().mockResolvedValue({ - paymentIntentId: 'pi_123', - clientSecret: 'cs_123', - requiresAction: false, - }), + const baseCreatePaymentDto: CreatePaymentDto = { + courseId: 'course-1', + amount: 100, + currency: 'USD', + provider: 'stripe', + metadata: { source: 'test' }, }; - providerFactoryMock.getProvider.mockReturnValue(provider); - - await expect( - service.createPaymentIntent('user-1', baseCreatePaymentDto), - ).resolves.toMatchObject({ - paymentId: 'payment-1', - clientSecret: 'cs_123', - requiresAction: false, + beforeEach(async () => { + const paymentRepoMock = createRepositoryMock(); + const subscriptionRepoMock = createRepositoryMock(); + const userRepoMock = createRepositoryMock(); + const refundRepoMock = createRepositoryMock(); + const invoiceRepoMock = createRepositoryMock(); + const mockTransactionService = { + runWithRetry: jest.fn((operation: (manager: { + create: jest.Mock; + save: jest.Mock; + }) => Promise) => operation({ + create: jest.fn((_Entity: unknown, data: Record) => ({ + id: 'payment-1', + ...data, + })), + save: jest.fn().mockResolvedValue(undefined), + })), + }; + providerFactoryMock = { getProvider: jest.fn() }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + PaymentsService, + { + provide: getRepositoryToken(Payment), + useValue: paymentRepoMock, + }, + { + provide: getRepositoryToken(Subscription), + useValue: subscriptionRepoMock, + }, + { + provide: getRepositoryToken(User), + useValue: userRepoMock, + }, + { + provide: getRepositoryToken(Refund), + useValue: refundRepoMock, + }, + { + provide: getRepositoryToken(Invoice), + useValue: invoiceRepoMock, + }, + { + provide: TransactionService, + useValue: mockTransactionService, + }, + { + provide: ProviderFactoryService, + useValue: providerFactoryMock, + }, + ], + }).compile(); + service = module.get(PaymentsService); + paymentRepository = module.get(getRepositoryToken(Payment)); + userRepository = module.get(getRepositoryToken(User)); + refundRepository = module.get(getRepositoryToken(Refund)); + invoiceRepository = module.get(getRepositoryToken(Invoice)); + }); + it('creates payment intent for valid user', async () => { + userRepository.findOne.mockResolvedValue({ id: 'user-1' }); + paymentRepository.create.mockReturnValue({ + id: 'payment-1', + ...baseCreatePaymentDto, + status: PaymentStatus.PENDING, + }); + paymentRepository.save.mockResolvedValue(undefined); + const provider = { + createPaymentIntent: jest.fn().mockResolvedValue({ + paymentIntentId: 'pi_123', + clientSecret: 'cs_123', + requiresAction: false, + }), + }; + providerFactoryMock.getProvider.mockReturnValue(provider); + await expect(service.createPaymentIntent('user-1', baseCreatePaymentDto)).resolves.toMatchObject({ + paymentId: 'payment-1', + clientSecret: 'cs_123', + requiresAction: false, + }); + }); + it('returns not found when user does not exist', async () => { + userRepository.findOne.mockResolvedValue(null); + await expectNotFound(() => service.createPaymentIntent('missing-user', baseCreatePaymentDto)); + }); + it('returns not found when refund payment does not exist', async () => { + paymentRepository.findOne.mockResolvedValue(null); + await expectNotFound(() => service.processRefund({ paymentId: 'missing', reason: 'duplicate' })); + }); + it('returns validation failure when refunding non-completed payment', async () => { + paymentRepository.findOne.mockResolvedValue({ + id: 'payment-1', + provider: 'stripe', + status: PaymentStatus.PENDING, + }); + await expectValidationFailure(() => service.processRefund({ paymentId: 'payment-1', reason: 'duplicate' })); + }); + it('returns not found when invoice payment is missing', async () => { + paymentRepository.findOne.mockResolvedValue(null); + await expectNotFound(() => service.getInvoice('payment-1', 'user-1')); + }); + it('supports unauthorized flow when provider rejects a request', async () => { + userRepository.findOne.mockResolvedValue({ id: 'user-1' }); + providerFactoryMock.getProvider.mockReturnValue({ + createPaymentIntent: jest + .fn() + .mockRejectedValue(new UnauthorizedException('Invalid provider token')), + }); + await expectUnauthorized(() => service.createPaymentIntent('user-1', baseCreatePaymentDto)); }); - }); - - it('returns not found when user does not exist', async () => { - userRepository.findOne.mockResolvedValue(null); - - await expectNotFound(() => service.createPaymentIntent('missing-user', baseCreatePaymentDto)); - }); - - it('returns not found when refund payment does not exist', async () => { - paymentRepository.findOne.mockResolvedValue(null); - - await expectNotFound(() => - service.processRefund({ paymentId: 'missing', reason: 'duplicate' }), - ); - }); - - it('returns validation failure when refunding non-completed payment', async () => { - paymentRepository.findOne.mockResolvedValue({ - id: 'payment-1', - provider: 'stripe', - status: PaymentStatus.PENDING, + it('uses pagination offset for user payment history', async () => { + paymentRepository.find.mockResolvedValue([]); + await service.getUserPayments('user-1', 20, 3); + expect(paymentRepository.find).toHaveBeenCalledWith(expect.objectContaining({ + where: { userId: 'user-1' }, + skip: 40, + take: 20, + })); }); - - await expectValidationFailure(() => - service.processRefund({ paymentId: 'payment-1', reason: 'duplicate' }), - ); - }); - - it('returns not found when invoice payment is missing', async () => { - paymentRepository.findOne.mockResolvedValue(null); - - await expectNotFound(() => service.getInvoice('payment-1', 'user-1')); - }); - - it('supports unauthorized flow when provider rejects a request', async () => { - userRepository.findOne.mockResolvedValue({ id: 'user-1' }); - providerFactoryMock.getProvider.mockReturnValue({ - createPaymentIntent: jest - .fn() - .mockRejectedValue(new UnauthorizedException('Invalid provider token')), + it('throws business validation error type for non-completed refund', async () => { + paymentRepository.findOne.mockResolvedValue({ + id: 'payment-2', + provider: 'stripe', + status: PaymentStatus.PENDING, + }); + await expect(service.processRefund({ paymentId: 'payment-2', reason: 'duplicate' })).rejects.toBeInstanceOf(BadRequestException); }); - - await expectUnauthorized(() => service.createPaymentIntent('user-1', baseCreatePaymentDto)); - }); - - it('uses pagination offset for user payment history', async () => { - paymentRepository.find.mockResolvedValue([]); - - await service.getUserPayments('user-1', 20, 3); - - expect(paymentRepository.find).toHaveBeenCalledWith( - expect.objectContaining({ - where: { userId: 'user-1' }, - skip: 40, - take: 20, - }), - ); - }); - - it('throws business validation error type for non-completed refund', async () => { - paymentRepository.findOne.mockResolvedValue({ - id: 'payment-2', - provider: 'stripe', - status: PaymentStatus.PENDING, + it('throws not found type when user is missing', async () => { + userRepository.findOne.mockResolvedValue(null); + await expect(service.createPaymentIntent('missing-user', baseCreatePaymentDto)).rejects.toBeInstanceOf(NotFoundException); }); - - await expect( - service.processRefund({ paymentId: 'payment-2', reason: 'duplicate' }), - ).rejects.toBeInstanceOf(BadRequestException); - }); - - it('throws not found type when user is missing', async () => { - userRepository.findOne.mockResolvedValue(null); - - await expect( - service.createPaymentIntent('missing-user', baseCreatePaymentDto), - ).rejects.toBeInstanceOf(NotFoundException); - }); }); diff --git a/src/payments/payments.service.ts b/src/payments/payments.service.ts index 4665518b..fb0e3ad5 100644 --- a/src/payments/payments.service.ts +++ b/src/payments/payments.service.ts @@ -13,289 +13,219 @@ import { TransactionService } from '../common/database/transaction.service'; import { Transactional } from '../common/database/transactional.decorator'; import { ensureUserExists } from '../common/utils/user.utils'; import { ProviderFactoryService } from './providers/provider-factory.service'; -import { - CreatePaymentIntentResult, - CreateSubscriptionResult, - ProcessRefundResult, - SubscriptionWebhookEvent, - RefundWebhookData, -} from './interfaces/payment-provider.interface'; - +import { CreatePaymentIntentResult, CreateSubscriptionResult, ProcessRefundResult, SubscriptionWebhookEvent, RefundWebhookData, } from './interfaces/payment-provider.interface'; @Injectable() export class PaymentsService { - constructor( + constructor( @InjectRepository(Payment) - private readonly paymentRepository: Repository, + private readonly paymentRepository: Repository, @InjectRepository(Subscription) - private readonly subscriptionRepository: Repository, + private readonly subscriptionRepository: Repository, @InjectRepository(User) - private readonly userRepository: Repository, + private readonly userRepository: Repository, @InjectRepository(Refund) - private readonly refundRepository: Repository, + private readonly refundRepository: Repository, @InjectRepository(Invoice) - private readonly invoiceRepository: Repository, - private readonly transactionService: TransactionService, - private readonly providerFactory: ProviderFactoryService, - ) {} - - /** - * Create a payment intent with a provider and record the pending payment. - * Refactored to use @Transactional decorator for consistent transaction management. - */ - @Transactional() - async createPaymentIntent( - userId: string, - createPaymentDto: CreatePaymentDto, - ): Promise { - const { courseId, amount, currency, provider, metadata } = createPaymentDto; - - // Verify user exists - const userOrNull = await this.userRepository.findOne({ - where: { id: userId }, - }); - const user = ensureUserExists(userOrNull); - - // Get payment provider - const paymentProvider = this.providerFactory.getProvider(provider ?? 'stripe'); - - // Create payment intent - const paymentIntent = await paymentProvider.createPaymentIntent(amount, currency ?? 'USD', { - ...metadata, - userId, - courseId, - }); - - // Create payment record - const payment = this.paymentRepository.create({ - amount, - currency, - method: PaymentMethod.CREDIT_CARD, - provider, - providerPaymentId: paymentIntent.paymentIntentId, - status: PaymentStatus.PENDING, - metadata, - user, - userId, - courseId, - }); - - await this.paymentRepository.save(payment); - - return { - paymentId: payment.id, - clientSecret: paymentIntent.clientSecret, - requiresAction: paymentIntent.requiresAction, - }; - } - - async createSubscription( - userId: string, - createSubscriptionDto: CreateSubscriptionDto, - ): Promise { - const { interval } = createSubscriptionDto; - - // Verify user exists - const userOrNull = await this.userRepository.findOne({ - where: { id: userId }, - }); - - ensureUserExists(userOrNull); - - // Get payment provider - // const paymentProvider = this.providerFactory.getProvider(provider); - - // Create subscription record - const subscription = this.subscriptionRepository.create({ - providerSubscriptionId: `sub_${Math.random().toString(36).substr(2, 9)}`, - status: SubscriptionStatus.ACTIVE, - interval, - amount: 0, // Would come from priceId - currency: 'USD', - currentPeriodStart: new Date(), - currentPeriodEnd: new Date(Date.now() + 30 * 24 * 60 * 60 * 1000), // 30 days from now - user: { id: userId } as User, - userId, - }); - - await this.subscriptionRepository.save(subscription); - - return { - subscriptionId: subscription.id, - status: subscription.status, - currentPeriodEnd: subscription.currentPeriodEnd, - }; - } - - /** - * Process a refund for a payment - * Uses @Transactional to ensure both payment status update and refund record creation - * succeed or fail together, preventing orphaned refund records or inconsistent payment states. - */ - @Transactional() - async processRefund(refundDto: RefundDto): Promise { - const { paymentId, amount, reason } = refundDto; - - // Find payment - const payment = await this.paymentRepository.findOne({ - where: { id: paymentId }, - }); - - if (!payment) { - throw new NotFoundException('Payment not found'); + private readonly invoiceRepository: Repository, private readonly transactionService: TransactionService, private readonly providerFactory: ProviderFactoryService) { } + /** + * Create a payment intent with a provider and record the pending payment. + * Refactored to use @Transactional decorator for consistent transaction management. + */ + @Transactional() + async createPaymentIntent(userId: string, createPaymentDto: CreatePaymentDto): Promise { + const { courseId, amount, currency, provider, metadata } = createPaymentDto; + // Verify user exists + const userOrNull = await this.userRepository.findOne({ + where: { id: userId }, + }); + const user = ensureUserExists(userOrNull); + // Get payment provider + const paymentProvider = this.providerFactory.getProvider(provider ?? 'stripe'); + // Create payment intent + const paymentIntent = await paymentProvider.createPaymentIntent(amount, currency ?? 'USD', { + ...metadata, + userId, + courseId, + }); + // Create payment record + const payment = this.paymentRepository.create({ + amount, + currency, + method: PaymentMethod.CREDIT_CARD, + provider, + providerPaymentId: paymentIntent.paymentIntentId, + status: PaymentStatus.PENDING, + metadata, + user, + userId, + courseId, + }); + await this.paymentRepository.save(payment); + return { + paymentId: payment.id, + clientSecret: paymentIntent.clientSecret, + requiresAction: paymentIntent.requiresAction, + }; } - - if (payment.status !== PaymentStatus.COMPLETED) { - throw new BadRequestException('Only completed payments can be refunded'); + async createSubscription(userId: string, createSubscriptionDto: CreateSubscriptionDto): Promise { + const { interval } = createSubscriptionDto; + // Verify user exists + const userOrNull = await this.userRepository.findOne({ + where: { id: userId }, + }); + ensureUserExists(userOrNull); + // Get payment provider + // const paymentProvider = this.providerFactory.getProvider(provider); + // Create subscription record + const subscription = this.subscriptionRepository.create({ + providerSubscriptionId: `sub_${Math.random().toString(36).substr(2, 9)}`, + status: SubscriptionStatus.ACTIVE, + interval, + amount: 0, // Would come from priceId + currency: 'USD', + currentPeriodStart: new Date(), + currentPeriodEnd: new Date(Date.now() + 30 * 24 * 60 * 60 * 1000), // 30 days from now + user: { id: userId } as User, + userId, + }); + await this.subscriptionRepository.save(subscription); + return { + subscriptionId: subscription.id, + status: subscription.status, + currentPeriodEnd: subscription.currentPeriodEnd, + }; } - - // Get provider - const paymentProvider = this.providerFactory.getProvider(payment.provider); - - // Process refund with provider - const refundResult = await paymentProvider.refundPayment(payment.providerPaymentId, amount); - - // Update payment status - payment.status = PaymentStatus.REFUNDED; - await this.paymentRepository.save(payment); - - // Create refund record - const refund = this.refundRepository.create({ - paymentId: payment.id, - amount: amount || payment.amount, - reason, - refundMethod: 'original_method', - providerRefundId: refundResult.refundId, - status: RefundStatus.PROCESSED, - }); - - await this.refundRepository.save(refund); - - return { - refundId: refund.id, - status: refund.status, - amount: refund.amount, - }; - } - - async getUserPayments(userId: string, limit: number, page: number): Promise { - const skip = (page - 1) * limit; - - return await this.paymentRepository.find({ - where: { userId }, - order: { createdAt: 'DESC' }, - skip, - take: limit, - }); - } - - async getUserSubscriptions(userId: string): Promise { - return await this.subscriptionRepository.find({ - where: { userId }, - order: { createdAt: 'DESC' }, - }); - } - - /** - * Get or generate an invoice for a payment. - * Uses @Transactional to prevent race conditions during invoice generation. - */ - @Transactional() - async getInvoice(paymentId: string, userId: string): Promise { - // Find payment - const payment = await this.paymentRepository.findOne({ - where: { id: paymentId, userId }, - }); - - if (!payment) { - throw new NotFoundException('Payment not found'); + /** + * Process a refund for a payment + * Uses @Transactional to ensure both payment status update and refund record creation + * succeed or fail together, preventing orphaned refund records or inconsistent payment states. + */ + @Transactional() + async processRefund(refundDto: RefundDto): Promise { + const { paymentId, amount, reason } = refundDto; + // Find payment + const payment = await this.paymentRepository.findOne({ + where: { id: paymentId }, + }); + if (!payment) { + throw new NotFoundException('Payment not found'); + } + if (payment.status !== PaymentStatus.COMPLETED) { + throw new BadRequestException('Only completed payments can be refunded'); + } + // Get provider + const paymentProvider = this.providerFactory.getProvider(payment.provider); + // Process refund with provider + const refundResult = await paymentProvider.refundPayment(payment.providerPaymentId, amount); + // Update payment status + payment.status = PaymentStatus.REFUNDED; + await this.paymentRepository.save(payment); + // Create refund record + const refund = this.refundRepository.create({ + paymentId: payment.id, + amount: amount || payment.amount, + reason, + refundMethod: 'original_method', + providerRefundId: refundResult.refundId, + status: RefundStatus.PROCESSED, + }); + await this.refundRepository.save(refund); + return { + refundId: refund.id, + status: refund.status, + amount: refund.amount, + }; } - - // Check if invoice already exists - let invoice = await this.invoiceRepository.findOne({ - where: { paymentId: payment.id }, - }); - - if (!invoice) { - // Generate new invoice - invoice = this.invoiceRepository.create({ - paymentId: payment.id, - userId: payment.userId, - amount: payment.amount, - currency: payment.currency, - invoiceNumber: `INV-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`, - items: [ - { - description: 'Payment for course', - amount: payment.amount, - quantity: 1, - }, - ], - taxAmount: 0, - totalAmount: payment.amount, - status: InvoiceStatus.PAID, - }); - - await this.invoiceRepository.save(invoice); + async getUserPayments(userId: string, limit: number, page: number): Promise { + const skip = (page - 1) * limit; + return await this.paymentRepository.find({ + where: { userId }, + order: { createdAt: 'DESC' }, + skip, + take: limit, + }); } - - return invoice; - } - - async updatePaymentStatus( - paymentId: string, - status: PaymentStatus, - metadata?: Record, - ): Promise { - await this.paymentRepository.update( - { providerPaymentId: paymentId }, - { status, ...(metadata ? { metadata: metadata as Record } : {}) }, - ); - } - - async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { - // Handle subscription events from webhook - const subscriptionId = event.data.object.id; - const status = event.data.object.status; - - // Update subscription in database - await this.subscriptionRepository.update( - { providerSubscriptionId: subscriptionId }, - { status: status as SubscriptionStatus }, - ); - } - - /** - * Process refund triggered by a webhook - * Uses @Transactional to ensure atomicity between refund record creation and payment status update. - */ - @Transactional() - async processRefundFromWebhook( - paymentIntentId: string, - refundData: RefundWebhookData, - ): Promise { - // Find payment by provider ID - const payment = await this.paymentRepository.findOne({ - where: { providerPaymentId: paymentIntentId }, - }); - - if (!payment) { - throw new NotFoundException('Payment not found'); + async getUserSubscriptions(userId: string): Promise { + return await this.subscriptionRepository.find({ + where: { userId }, + order: { createdAt: 'DESC' }, + }); + } + /** + * Get or generate an invoice for a payment. + * Uses @Transactional to prevent race conditions during invoice generation. + */ + @Transactional() + async getInvoice(paymentId: string, userId: string): Promise { + // Find payment + const payment = await this.paymentRepository.findOne({ + where: { id: paymentId, userId }, + }); + if (!payment) { + throw new NotFoundException('Payment not found'); + } + // Check if invoice already exists + let invoice = await this.invoiceRepository.findOne({ + where: { paymentId: payment.id }, + }); + if (!invoice) { + // Generate new invoice + invoice = this.invoiceRepository.create({ + paymentId: payment.id, + userId: payment.userId, + amount: payment.amount, + currency: payment.currency, + invoiceNumber: `INV-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`, + items: [ + { + description: 'Payment for course', + amount: payment.amount, + quantity: 1, + }, + ], + taxAmount: 0, + totalAmount: payment.amount, + status: InvoiceStatus.PAID, + }); + await this.invoiceRepository.save(invoice); + } + return invoice; + } + async updatePaymentStatus(paymentId: string, status: PaymentStatus, metadata?: Record): Promise { + await this.paymentRepository.update({ providerPaymentId: paymentId }, { status, ...(metadata ? { metadata: metadata as Record } : {}) }); + } + async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { + // Handle subscription events from webhook + const subscriptionId = event.data.object.id; + const status = event.data.object.status; + // Update subscription in database + await this.subscriptionRepository.update({ providerSubscriptionId: subscriptionId }, { status: status as SubscriptionStatus }); + } + /** + * Process refund triggered by a webhook + * Uses @Transactional to ensure atomicity between refund record creation and payment status update. + */ + @Transactional() + async processRefundFromWebhook(paymentIntentId: string, refundData: RefundWebhookData): Promise { + // Find payment by provider ID + const payment = await this.paymentRepository.findOne({ + where: { providerPaymentId: paymentIntentId }, + }); + if (!payment) { + throw new NotFoundException('Payment not found'); + } + // Create refund record + const refund = this.refundRepository.create({ + paymentId: payment.id, + amount: refundData.amount, + reason: 'webhook_refund', + refundMethod: 'original_method', + providerRefundId: refundData.id, + status: RefundStatus.PROCESSED, + }); + await this.refundRepository.save(refund); + // Update payment status + payment.status = PaymentStatus.REFUNDED; + await this.paymentRepository.save(payment); } - - // Create refund record - const refund = this.refundRepository.create({ - paymentId: payment.id, - amount: refundData.amount, - reason: 'webhook_refund', - refundMethod: 'original_method', - providerRefundId: refundData.id, - status: RefundStatus.PROCESSED, - }); - - await this.refundRepository.save(refund); - - // Update payment status - payment.status = PaymentStatus.REFUNDED; - await this.paymentRepository.save(payment); - } } diff --git a/src/payments/providers/payment-provider.interface.ts b/src/payments/providers/payment-provider.interface.ts index 797dca5f..d58a27f5 100644 --- a/src/payments/providers/payment-provider.interface.ts +++ b/src/payments/providers/payment-provider.interface.ts @@ -1,43 +1,23 @@ export interface PaymentProvider { - name: string; - - createPaymentIntent( - amount: number, - currency: string, - metadata?: Record, - ): Promise<{ - clientSecret: string; - paymentIntentId: string; - requiresAction?: boolean; - }>; - - createSubscription( - customerId: string, - priceId: string, - metadata?: Record, - ): Promise<{ - subscriptionId: string; - status: string; - currentPeriodEnd: Date; - }>; - - cancelSubscription(subscriptionId: string): Promise; - - refundPayment( - paymentId: string, - amount?: number, - ): Promise<{ - refundId: string; - status: string; - }>; - - handleWebhook( - payload: any, - signature: string, - ): Promise<{ - type: string; - data: any; - }>; - - verifyWebhookSignature(payload: any, signature: string): Promise; + name: string; + createPaymentIntent(amount: number, currency: string, metadata?: Record): Promise<{ + clientSecret: string; + paymentIntentId: string; + requiresAction?: boolean; + }>; + createSubscription(customerId: string, priceId: string, metadata?: Record): Promise<{ + subscriptionId: string; + status: string; + currentPeriodEnd: Date; + }>; + cancelSubscription(subscriptionId: string): Promise; + refundPayment(paymentId: string, amount?: number): Promise<{ + refundId: string; + status: string; + }>; + handleWebhook(payload: unknown, signature: string): Promise<{ + type: string; + data: unknown; + }>; + verifyWebhookSignature(payload: unknown, signature: string): Promise; } diff --git a/src/payments/providers/provider-factory.service.ts b/src/payments/providers/provider-factory.service.ts index 8df38888..9acb54c1 100644 --- a/src/payments/providers/provider-factory.service.ts +++ b/src/payments/providers/provider-factory.service.ts @@ -1,16 +1,14 @@ import { Injectable } from '@nestjs/common'; import { StripeService } from './stripe.service'; - @Injectable() export class ProviderFactoryService { - constructor(private readonly stripeService: StripeService) {} - - getProvider(provider: string): StripeService { - switch (provider.toLowerCase()) { - case 'stripe': - return this.stripeService; - default: - throw new Error(`Unsupported payment provider: ${provider}`); + constructor(private readonly stripeService: StripeService) { } + getProvider(provider: string): StripeService { + switch (provider.toLowerCase()) { + case 'stripe': + return this.stripeService; + default: + throw new Error(`Unsupported payment provider: ${provider}`); + } } - } } diff --git a/src/payments/providers/stripe.service.ts b/src/payments/providers/stripe.service.ts index a47c8c8e..a69da824 100644 --- a/src/payments/providers/stripe.service.ts +++ b/src/payments/providers/stripe.service.ts @@ -1,65 +1,54 @@ import { Injectable, InternalServerErrorException, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import Stripe from 'stripe'; - @Injectable() export class StripeService { - private readonly stripe: Stripe; - private readonly stripeWebhookSecret: string; - private readonly logger = new Logger(StripeService.name); - - constructor(private readonly configService: ConfigService) { - const secretKey = this.configService.get('STRIPE_SECRET_KEY'); - this.stripeWebhookSecret = this.configService.get('STRIPE_WEBHOOK_SECRET'); - - if (!secretKey) { - this.logger.error('STRIPE_SECRET_KEY is not defined in environment variables'); + private readonly stripe: Stripe; + private readonly stripeWebhookSecret: string; + private readonly logger = new Logger(StripeService.name); + constructor(private readonly configService: ConfigService) { + const secretKey = this.configService.get('STRIPE_SECRET_KEY'); + this.stripeWebhookSecret = this.configService.get('STRIPE_WEBHOOK_SECRET'); + if (!secretKey) { + this.logger.error('STRIPE_SECRET_KEY is not defined in environment variables'); + } + this.stripe = new Stripe(secretKey || 'sk_test_placeholder'); } - - this.stripe = new Stripe(secretKey || 'sk_test_placeholder'); - } - - async createPaymentIntent(amount: number, currency: string, metadata: any): Promise { - const paymentIntent = await this.stripe.paymentIntents.create({ - amount: Math.round(amount * 100), // Stripe expects amount in cents - currency: currency.toLowerCase(), - metadata, - }); - - return { - paymentIntentId: paymentIntent.id, - clientSecret: paymentIntent.client_secret, - requiresAction: paymentIntent.status === 'requires_action', - }; - } - - async refundPayment(paymentId: string, amount?: number): Promise { - const refundData: Stripe.RefundCreateParams = { - payment_intent: paymentId, - }; - - if (amount) { - refundData.amount = Math.round(amount * 100); + async createPaymentIntent(amount: number, currency: string, metadata: unknown): Promise { + const paymentIntent = await this.stripe.paymentIntents.create({ + amount: Math.round(amount * 100), // Stripe expects amount in cents + currency: currency.toLowerCase(), + metadata, + }); + return { + paymentIntentId: paymentIntent.id, + clientSecret: paymentIntent.client_secret, + requiresAction: paymentIntent.status === 'requires_action', + }; } - - const refund = await this.stripe.refunds.create(refundData); - - return { - refundId: refund.id, - status: refund.status, - }; - } - - async handleWebhook(payload: string | Buffer, signature: string): Promise { - if (!this.stripeWebhookSecret) { - throw new InternalServerErrorException('Stripe webhook secret is missing'); + async refundPayment(paymentId: string, amount?: number): Promise { + const refundData: Stripe.RefundCreateParams = { + payment_intent: paymentId, + }; + if (amount) { + refundData.amount = Math.round(amount * 100); + } + const refund = await this.stripe.refunds.create(refundData); + return { + refundId: refund.id, + status: refund.status, + }; } - - try { - return this.stripe.webhooks.constructEvent(payload, signature, this.stripeWebhookSecret); - } catch (err: any) { - this.logger.error(`Webhook signature verification failed: ${err.message}`); - throw new Error(`Webhook Error: ${err.message}`); + async handleWebhook(payload: string | Buffer, signature: string): Promise { + if (!this.stripeWebhookSecret) { + throw new InternalServerErrorException('Stripe webhook secret is missing'); + } + try { + return this.stripe.webhooks.constructEvent(payload, signature, this.stripeWebhookSecret); + } + catch (err: unknown) { + this.logger.error(`Webhook signature verification failed: ${err.message}`); + throw new Error(`Webhook Error: ${err.message}`); + } } - } } diff --git a/src/payments/subscriptions/subscription-job.processor.ts b/src/payments/subscriptions/subscription-job.processor.ts index b4192d9b..067a0d34 100644 --- a/src/payments/subscriptions/subscription-job.processor.ts +++ b/src/payments/subscriptions/subscription-job.processor.ts @@ -2,14 +2,13 @@ import { Processor, Process } from '@nestjs/bull'; import { Job } from 'bull'; import { Logger } from '@nestjs/common'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; - @Processor(QUEUE_NAMES.SUBSCRIPTIONS) export class SubscriptionJobProcessor { - private readonly logger = new Logger(SubscriptionJobProcessor.name); - @Process(JOB_NAMES.PROCESS_SUBSCRIPTION) - async handleSubscription(job: Job): Promise { - // Process subscription job - this.logger.log('Processing subscription job:', job.data); - return { success: true }; - } + private readonly logger = new Logger(SubscriptionJobProcessor.name); + @Process(JOB_NAMES.PROCESS_SUBSCRIPTION) + async handleSubscription(job: Job): Promise { + // Process subscription job + this.logger.log('Processing subscription job:', job.data); + return { success: true }; + } } diff --git a/src/payments/subscriptions/subscriptions.service.ts b/src/payments/subscriptions/subscriptions.service.ts index dd177832..80220ea5 100644 --- a/src/payments/subscriptions/subscriptions.service.ts +++ b/src/payments/subscriptions/subscriptions.service.ts @@ -1,10 +1,9 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class SubscriptionsService { - // Placeholder implementation - async processSubscription(): Promise { - // Logic to process subscription payments - return { success: true }; - } + // Placeholder implementation + async processSubscription(): Promise { + // Logic to process subscription payments + return { success: true }; + } } diff --git a/src/payments/webhooks/dto/webhook-retry.dto.ts b/src/payments/webhooks/dto/webhook-retry.dto.ts index d60ba58b..78c7d0bd 100644 --- a/src/payments/webhooks/dto/webhook-retry.dto.ts +++ b/src/payments/webhooks/dto/webhook-retry.dto.ts @@ -1,71 +1,55 @@ import { IsUUID, IsEnum, IsInt, IsDateString, IsOptional, IsObject } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { WebhookStatus, WebhookProvider } from '../entities/webhook-retry.entity'; - export class WebhookRetryDto { - @ApiProperty({ description: 'Webhook retry ID' }) - @IsUUID() - id: string; - - @ApiProperty({ enum: WebhookProvider, description: 'Payment provider' }) - @IsEnum(WebhookProvider) - provider: WebhookProvider; - - @ApiProperty({ description: 'External event ID from provider' }) - externalEventId: string; - - @ApiProperty({ enum: WebhookStatus, description: 'Current webhook status' }) - @IsEnum(WebhookStatus) - status: WebhookStatus; - - @ApiProperty({ description: 'Number of retry attempts' }) - @IsInt() - retryCount: number; - - @ApiProperty({ description: 'Maximum number of retries' }) - @IsInt() - maxRetries: number; - - @ApiPropertyOptional({ description: 'Timestamp of next retry' }) - @IsOptional() - @IsDateString() - nextRetryTime?: Date; - - @ApiPropertyOptional({ description: 'Last error message' }) - @IsOptional() - lastError?: string; - - @ApiPropertyOptional({ description: 'Error details (stack trace, etc.)' }) - @IsOptional() - @IsObject() - errorDetails?: Record; - - @ApiProperty({ description: 'Webhook creation timestamp' }) - @IsDateString() - createdAt: Date; - - @ApiProperty({ description: 'Webhook last update timestamp' }) - @IsDateString() - updatedAt: Date; - - @ApiPropertyOptional({ description: 'Webhook processing completion timestamp' }) - @IsOptional() - @IsDateString() - processedAt?: Date; + @ApiProperty({ description: 'Webhook retry ID' }) + @IsUUID() + id: string; + @ApiProperty({ enum: WebhookProvider, description: 'Payment provider' }) + @IsEnum(WebhookProvider) + provider: WebhookProvider; + @ApiProperty({ description: 'External event ID from provider' }) + externalEventId: string; + @ApiProperty({ enum: WebhookStatus, description: 'Current webhook status' }) + @IsEnum(WebhookStatus) + status: WebhookStatus; + @ApiProperty({ description: 'Number of retry attempts' }) + @IsInt() + retryCount: number; + @ApiProperty({ description: 'Maximum number of retries' }) + @IsInt() + maxRetries: number; + @ApiPropertyOptional({ description: 'Timestamp of next retry' }) + @IsOptional() + @IsDateString() + nextRetryTime?: Date; + @ApiPropertyOptional({ description: 'Last error message' }) + @IsOptional() + lastError?: string; + @ApiPropertyOptional({ description: 'Error details (stack trace, etc.)' }) + @IsOptional() + @IsObject() + errorDetails?: Record; + @ApiProperty({ description: 'Webhook creation timestamp' }) + @IsDateString() + createdAt: Date; + @ApiProperty({ description: 'Webhook last update timestamp' }) + @IsDateString() + updatedAt: Date; + @ApiPropertyOptional({ description: 'Webhook processing completion timestamp' }) + @IsOptional() + @IsDateString() + processedAt?: Date; } - export class WebhookRetryResponseDto { - @ApiProperty({ description: 'Success status' }) - success: boolean; - - @ApiPropertyOptional({ description: 'Webhook retry ID' }) - webhookRetryId?: string; - - @ApiPropertyOptional({ description: 'Error message if failed' }) - error?: string; + @ApiProperty({ description: 'Success status' }) + success: boolean; + @ApiPropertyOptional({ description: 'Webhook retry ID' }) + webhookRetryId?: string; + @ApiPropertyOptional({ description: 'Error message if failed' }) + error?: string; } - export class DeadLetterWebhookDto extends WebhookRetryDto { - @ApiProperty({ description: 'Indicates this is a dead letter webhook' }) - isDead: boolean = true; + @ApiProperty({ description: 'Indicates this is a dead letter webhook' }) + isDead: boolean = true; } diff --git a/src/payments/webhooks/entities/webhook-retry.entity.ts b/src/payments/webhooks/entities/webhook-retry.entity.ts index 1c62d0f4..df73c506 100644 --- a/src/payments/webhooks/entities/webhook-retry.entity.ts +++ b/src/payments/webhooks/entities/webhook-retry.entity.ts @@ -1,72 +1,48 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - Index, -} from 'typeorm'; - +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, Index, } from 'typeorm'; export enum WebhookStatus { - PENDING = 'pending', - PROCESSING = 'processing', - SUCCEEDED = 'succeeded', - FAILED = 'failed', - DEAD_LETTER = 'dead_letter', + PENDING = 'pending', + PROCESSING = 'processing', + SUCCEEDED = 'succeeded', + FAILED = 'failed', + DEAD_LETTER = 'dead_letter' } - export enum WebhookProvider { - STRIPE = 'stripe', - PAYPAL = 'paypal', + STRIPE = 'stripe', + PAYPAL = 'paypal' } - @Entity('webhook_retries') @Index(['provider', 'externalEventId'], { unique: true }) @Index(['status', 'nextRetryTime']) @Index(['createdAt']) export class WebhookRetry { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ type: 'enum', enum: WebhookProvider }) - provider: WebhookProvider; - - @Column() - externalEventId: string; - - @Column({ type: 'enum', enum: WebhookStatus, default: WebhookStatus.PENDING }) - status: WebhookStatus; - - @Column({ type: 'jsonb', nullable: true }) - payload: Record; - - @Column({ type: 'text', nullable: true }) - signature: string; - - @Column({ type: 'int', default: 0 }) - retryCount: number; - - @Column({ type: 'int', default: 3 }) - maxRetries: number; - - @Column({ type: 'timestamp', nullable: true }) - nextRetryTime: Date; - - @Column({ type: 'text', nullable: true }) - lastError: string; - - @Column({ type: 'jsonb', nullable: true }) - errorDetails: Record; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @Column({ type: 'timestamp', nullable: true }) - processedAt: Date; - - @Column({ type: 'jsonb', nullable: true }) - headers: Record; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ type: 'enum', enum: WebhookProvider }) + provider: WebhookProvider; + @Column() + externalEventId: string; + @Column({ type: 'enum', enum: WebhookStatus, default: WebhookStatus.PENDING }) + status: WebhookStatus; + @Column({ type: 'jsonb', nullable: true }) + payload: Record; + @Column({ type: 'text', nullable: true }) + signature: string; + @Column({ type: 'int', default: 0 }) + retryCount: number; + @Column({ type: 'int', default: 3 }) + maxRetries: number; + @Column({ type: 'timestamp', nullable: true }) + nextRetryTime: Date; + @Column({ type: 'text', nullable: true }) + lastError: string; + @Column({ type: 'jsonb', nullable: true }) + errorDetails: Record; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @Column({ type: 'timestamp', nullable: true }) + processedAt: Date; + @Column({ type: 'jsonb', nullable: true }) + headers: Record; } diff --git a/src/payments/webhooks/migration-helper.ts b/src/payments/webhooks/migration-helper.ts index 2fcc84a9..909006cd 100644 --- a/src/payments/webhooks/migration-helper.ts +++ b/src/payments/webhooks/migration-helper.ts @@ -10,7 +10,6 @@ * - TypeORM migration runner * - Your custom migration system */ - // SQL Migration for PostgreSQL export const createWebhookRetriesTableSQL = ` -- Create ENUM types if they don't exist @@ -60,7 +59,6 @@ CREATE INDEX IF NOT EXISTS idx_webhook_dead_letter ON webhook_retries("createdAt") WHERE status = 'dead_letter'; `; - // TypeORM Migration Template export const typeOrmMigrationTemplate = ` import { MigrationInterface, QueryRunner } from 'typeorm'; @@ -132,8 +130,8 @@ export class CreateWebhookRetriesTable1681234567890 implements MigrationInterfac `; // Database cleanup and maintenance queries export const maintenanceQueries = { - // Get dead letter queue stats - getDeadLetterStats: ` + // Get dead letter queue stats + getDeadLetterStats: ` SELECT COUNT(*) as total_dead_letters, COUNT(DISTINCT provider) as providers, @@ -142,9 +140,8 @@ export const maintenanceQueries = { FROM webhook_retries WHERE status = 'dead_letter'; `, - - // Get retry statistics - getRetryStats: ` + // Get retry statistics + getRetryStats: ` SELECT status, COUNT(*) as count, @@ -154,9 +151,8 @@ export const maintenanceQueries = { FROM webhook_retries GROUP BY status; `, - - // Get webhooks pending processing - getPending: ` + // Get webhooks pending processing + getPending: ` SELECT id, provider, "externalEventId", "nextRetryTime", "retryCount" FROM webhook_retries WHERE status = 'pending' @@ -164,16 +160,14 @@ export const maintenanceQueries = { ORDER BY "nextRetryTime" ASC LIMIT 100; `, - - // Archive old succeeded webhooks (cleanup) - archiveSuccessful: ` + // Archive old succeeded webhooks (cleanup) + archiveSuccessful: ` DELETE FROM webhook_retries WHERE status = 'succeeded' AND "processedAt" < NOW() - INTERVAL '30 days'; `, - - // Archive old dead letter webhooks (after review period) - archiveDeadLetter: ` + // Archive old dead letter webhooks (after review period) + archiveDeadLetter: ` DELETE FROM webhook_retries WHERE status = 'dead_letter' AND "createdAt" < NOW() - INTERVAL '90 days'; diff --git a/src/payments/webhooks/stripe-webhook.guard.ts b/src/payments/webhooks/stripe-webhook.guard.ts index e5ac3b5f..65b3e3a5 100644 --- a/src/payments/webhooks/stripe-webhook.guard.ts +++ b/src/payments/webhooks/stripe-webhook.guard.ts @@ -1,23 +1,18 @@ import { Injectable, CanActivate, ExecutionContext } from '@nestjs/common'; import { Request } from 'express'; - @Injectable() export class StripeWebhookGuard implements CanActivate { - canActivate(context: ExecutionContext): boolean { - const request = context.switchToHttp().getRequest(); - - // Verify the webhook signature here - // This is a simplified implementation - in production, you'd verify the signature - const signature = request.headers['stripe-signature']; - - if (!signature) { - return false; + canActivate(context: ExecutionContext): boolean { + const request = context.switchToHttp().getRequest(); + // Verify the webhook signature here + // This is a simplified implementation - in production, you'd verify the signature + const signature = request.headers['stripe-signature']; + if (!signature) { + return false; + } + // In a real implementation, you would verify the signature using Stripe's library + // const stripe = new Stripe(process.env.STRIPE_SECRET_KEY); + // stripe.webhooks.constructEvent(request.body, signature, process.env.STRIPE_WEBHOOK_SECRET); + return true; } - - // In a real implementation, you would verify the signature using Stripe's library - // const stripe = new Stripe(process.env.STRIPE_SECRET_KEY); - // stripe.webhooks.constructEvent(request.body, signature, process.env.STRIPE_WEBHOOK_SECRET); - - return true; - } } diff --git a/src/payments/webhooks/webhook-management.controller.ts b/src/payments/webhooks/webhook-management.controller.ts index e155b89e..d75c30d0 100644 --- a/src/payments/webhooks/webhook-management.controller.ts +++ b/src/payments/webhooks/webhook-management.controller.ts @@ -2,48 +2,52 @@ import { Controller, Get, HttpCode, HttpStatus, Param, Post, Query } from '@nest import { ApiTags, ApiOperation, ApiResponse, ApiQuery } from '@nestjs/swagger'; import { WebhookQueueService } from './webhook-queue.service'; import { WebhookRetry } from './entities/webhook-retry.entity'; - @ApiTags('webhooks') @Controller('webhooks') export class WebhookManagementController { - constructor(private readonly webhookQueueService: WebhookQueueService) {} - - @Get('status/:id') - @ApiOperation({ summary: 'Get webhook retry status' }) - @ApiResponse({ status: 200, description: 'Webhook status retrieved' }) - async getWebhookStatus(@Param('id') id: string): Promise { - return this.webhookQueueService.getWebhookStatus(id); - } - - @Get('dead-letter') - @ApiOperation({ summary: 'Get dead letter webhooks' }) - @ApiResponse({ status: 200, description: 'Dead letter webhooks retrieved' }) - @ApiQuery({ name: 'limit', required: false, type: Number, example: 100 }) - async getDeadLetterWebhooks(@Query('limit') limit?: number): Promise { - return this.webhookQueueService.getDeadLetterWebhooks(limit || 100); - } - - @Get('pending') - @ApiOperation({ summary: 'Get pending webhooks' }) - @ApiResponse({ status: 200, description: 'Pending webhooks retrieved' }) - @ApiQuery({ name: 'limit', required: false, type: Number, example: 100 }) - async getPendingWebhooks(@Query('limit') limit?: number): Promise { - return this.webhookQueueService.getPendingWebhooks(limit || 100); - } - - @Get('processing') - @ApiOperation({ summary: 'Get processing webhooks' }) - @ApiResponse({ status: 200, description: 'Processing webhooks retrieved' }) - async getProcessingWebhooks(): Promise { - return this.webhookQueueService.getProcessingWebhooks(); - } - - @Post('requeue/:id') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Requeue a dead letter webhook' }) - @ApiResponse({ status: 200, description: 'Webhook requeued' }) - async requeueDeadLetterWebhook(@Param('id') id: string): Promise<{ success: boolean }> { - await this.webhookQueueService.requeueDeadLetterWebhook(id); - return { success: true }; - } + constructor(private readonly webhookQueueService: WebhookQueueService) { } + @Get('status/:id') + @ApiOperation({ summary: 'Get webhook retry status' }) + @ApiResponse({ status: 200, description: 'Webhook status retrieved' }) + async getWebhookStatus( + @Param('id') + id: string): Promise { + return this.webhookQueueService.getWebhookStatus(id); + } + @Get('dead-letter') + @ApiOperation({ summary: 'Get dead letter webhooks' }) + @ApiResponse({ status: 200, description: 'Dead letter webhooks retrieved' }) + @ApiQuery({ name: 'limit', required: false, type: Number, example: 100 }) + async getDeadLetterWebhooks( + @Query('limit') + limit?: number): Promise { + return this.webhookQueueService.getDeadLetterWebhooks(limit || 100); + } + @Get('pending') + @ApiOperation({ summary: 'Get pending webhooks' }) + @ApiResponse({ status: 200, description: 'Pending webhooks retrieved' }) + @ApiQuery({ name: 'limit', required: false, type: Number, example: 100 }) + async getPendingWebhooks( + @Query('limit') + limit?: number): Promise { + return this.webhookQueueService.getPendingWebhooks(limit || 100); + } + @Get('processing') + @ApiOperation({ summary: 'Get processing webhooks' }) + @ApiResponse({ status: 200, description: 'Processing webhooks retrieved' }) + async getProcessingWebhooks(): Promise { + return this.webhookQueueService.getProcessingWebhooks(); + } + @Post('requeue/:id') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Requeue a dead letter webhook' }) + @ApiResponse({ status: 200, description: 'Webhook requeued' }) + async requeueDeadLetterWebhook( + @Param('id') + id: string): Promise<{ + success: boolean; + }> { + await this.webhookQueueService.requeueDeadLetterWebhook(id); + return { success: true }; + } } diff --git a/src/payments/webhooks/webhook-queue.service.spec.ts b/src/payments/webhooks/webhook-queue.service.spec.ts index 45e07650..01dca9d1 100644 --- a/src/payments/webhooks/webhook-queue.service.spec.ts +++ b/src/payments/webhooks/webhook-queue.service.spec.ts @@ -5,186 +5,151 @@ import { Queue } from 'bull'; import { getQueueToken } from '@nestjs/bull'; import { WebhookQueueService } from './webhook-queue.service'; import { WebhookRetry, WebhookStatus, WebhookProvider } from './entities/webhook-retry.entity'; - describe('WebhookQueueService', () => { - let service: WebhookQueueService; - let repository: Repository; - let queue: Queue; - - const mockRepository = { - findOne: jest.fn(), - create: jest.fn(), - save: jest.fn(), - find: jest.fn(), - }; - - const mockQueue = { - add: jest.fn(), - }; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [ - WebhookQueueService, - { - provide: getRepositoryToken(WebhookRetry), - useValue: mockRepository, - }, - { - provide: getQueueToken('webhooks'), - useValue: mockQueue, - }, - ], - }).compile(); - - service = module.get(WebhookQueueService); - repository = module.get>(getRepositoryToken(WebhookRetry)); - queue = module.get(getQueueToken('webhooks')); - }); - - afterEach(() => { - jest.resetAllMocks(); - }); - - describe('queueWebhook', () => { - it('should create and queue a new webhook', async () => { - const payload = Buffer.from('test payload'); - const webhookPayload = { - webhookRetryId: '', - provider: WebhookProvider.STRIPE, - payload, - signature: 'test-signature', - externalEventId: 'evt_123', - }; - - const webhookRetry = { - id: '550e8400-e29b-41d4-a716-446655440000', - provider: WebhookProvider.STRIPE, - externalEventId: 'evt_123', - status: WebhookStatus.PENDING, - payload: webhookPayload.payload as unknown as Record, - signature: 'test-signature', - retryCount: 0, - }; - - mockRepository.findOne.mockResolvedValue(null); - mockRepository.create.mockReturnValue(webhookRetry); - mockRepository.save.mockResolvedValue(webhookRetry); - mockQueue.add.mockResolvedValue({ id: 1 }); - - const result = await service.queueWebhook(webhookPayload); - - expect(result).toBe(webhookRetry.id); - expect(mockRepository.findOne).toHaveBeenCalledWith({ - where: { - externalEventId: webhookPayload.externalEventId, - provider: webhookPayload.provider, - }, - }); - expect(mockQueue.add).toHaveBeenCalled(); + let service: WebhookQueueService; + let repository: Repository; + let queue: Queue; + const mockRepository = { + findOne: jest.fn(), + create: jest.fn(), + save: jest.fn(), + find: jest.fn(), + }; + const mockQueue = { + add: jest.fn(), + }; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + providers: [ + WebhookQueueService, + { + provide: getRepositoryToken(WebhookRetry), + useValue: mockRepository, + }, + { + provide: getQueueToken('webhooks'), + useValue: mockQueue, + }, + ], + }).compile(); + service = module.get(WebhookQueueService); + repository = module.get>(getRepositoryToken(WebhookRetry)); + queue = module.get(getQueueToken('webhooks')); }); - - it('should update existing webhook and requeue', async () => { - const payload = Buffer.from('test payload'); - const webhookPayload = { - webhookRetryId: '', - provider: WebhookProvider.STRIPE, - payload, - signature: 'test-signature', - externalEventId: 'evt_123', - }; - - const existingWebhook = { - id: '550e8400-e29b-41d4-a716-446655440000', - provider: WebhookProvider.STRIPE, - externalEventId: 'evt_123', - status: WebhookStatus.FAILED, - retryCount: 2, - }; - - mockRepository.findOne.mockResolvedValue(existingWebhook); - mockRepository.save.mockResolvedValue({ ...existingWebhook, status: WebhookStatus.PENDING }); - mockQueue.add.mockResolvedValue({ id: 2 }); - - const result = await service.queueWebhook(webhookPayload); - - expect(result).toBe(existingWebhook.id); - expect(mockRepository.save).toHaveBeenCalled(); + afterEach(() => { + jest.resetAllMocks(); }); - }); - - describe('getDeadLetterWebhooks', () => { - it('should fetch dead letter webhooks', async () => { - const deadLetterWebhooks = [ - { - id: '550e8400-e29b-41d4-a716-446655440000', - status: WebhookStatus.DEAD_LETTER, - retryCount: 3, - }, - ]; - - mockRepository.find.mockResolvedValue(deadLetterWebhooks); - - const result = await service.getDeadLetterWebhooks(100); - - expect(result).toEqual(deadLetterWebhooks); - expect(mockRepository.find).toHaveBeenCalledWith({ - where: { status: WebhookStatus.DEAD_LETTER }, - order: { createdAt: 'DESC' }, - take: 100, - }); + describe('queueWebhook', () => { + it('should create and queue a new webhook', async () => { + const payload = Buffer.from('test payload'); + const webhookPayload = { + webhookRetryId: '', + provider: WebhookProvider.STRIPE, + payload, + signature: 'test-signature', + externalEventId: 'evt_123', + }; + const webhookRetry = { + id: '550e8400-e29b-41d4-a716-446655440000', + provider: WebhookProvider.STRIPE, + externalEventId: 'evt_123', + status: WebhookStatus.PENDING, + payload: webhookPayload.payload as unknown as Record, + signature: 'test-signature', + retryCount: 0, + }; + mockRepository.findOne.mockResolvedValue(null); + mockRepository.create.mockReturnValue(webhookRetry); + mockRepository.save.mockResolvedValue(webhookRetry); + mockQueue.add.mockResolvedValue({ id: 1 }); + const result = await service.queueWebhook(webhookPayload); + expect(result).toBe(webhookRetry.id); + expect(mockRepository.findOne).toHaveBeenCalledWith({ + where: { + externalEventId: webhookPayload.externalEventId, + provider: webhookPayload.provider, + }, + }); + expect(mockQueue.add).toHaveBeenCalled(); + }); + it('should update existing webhook and requeue', async () => { + const payload = Buffer.from('test payload'); + const webhookPayload = { + webhookRetryId: '', + provider: WebhookProvider.STRIPE, + payload, + signature: 'test-signature', + externalEventId: 'evt_123', + }; + const existingWebhook = { + id: '550e8400-e29b-41d4-a716-446655440000', + provider: WebhookProvider.STRIPE, + externalEventId: 'evt_123', + status: WebhookStatus.FAILED, + retryCount: 2, + }; + mockRepository.findOne.mockResolvedValue(existingWebhook); + mockRepository.save.mockResolvedValue({ ...existingWebhook, status: WebhookStatus.PENDING }); + mockQueue.add.mockResolvedValue({ id: 2 }); + const result = await service.queueWebhook(webhookPayload); + expect(result).toBe(existingWebhook.id); + expect(mockRepository.save).toHaveBeenCalled(); + }); }); - }); - - describe('requeueDeadLetterWebhook', () => { - it('should requeue a dead letter webhook', async () => { - const webhookId = '550e8400-e29b-41d4-a716-446655440000'; - const deadLetterWebhook = { - id: webhookId, - status: WebhookStatus.DEAD_LETTER, - provider: WebhookProvider.STRIPE, - externalEventId: 'evt_123', - payload: { test: 'data' }, - retryCount: 3, - }; - - mockRepository.findOne - .mockResolvedValueOnce(deadLetterWebhook) - .mockResolvedValueOnce(deadLetterWebhook); - - mockRepository.save.mockResolvedValue({ - ...deadLetterWebhook, - status: WebhookStatus.PENDING, - retryCount: 0, - }); - - mockQueue.add.mockResolvedValue({ id: 3 }); - - await service.requeueDeadLetterWebhook(webhookId); - - expect(mockRepository.save).toHaveBeenCalled(); - expect(mockQueue.add).toHaveBeenCalled(); + describe('getDeadLetterWebhooks', () => { + it('should fetch dead letter webhooks', async () => { + const deadLetterWebhooks = [ + { + id: '550e8400-e29b-41d4-a716-446655440000', + status: WebhookStatus.DEAD_LETTER, + retryCount: 3, + }, + ]; + mockRepository.find.mockResolvedValue(deadLetterWebhooks); + const result = await service.getDeadLetterWebhooks(100); + expect(result).toEqual(deadLetterWebhooks); + expect(mockRepository.find).toHaveBeenCalledWith({ + where: { status: WebhookStatus.DEAD_LETTER }, + order: { createdAt: 'DESC' }, + take: 100, + }); + }); }); - - it('should throw error if webhook not found', async () => { - mockRepository.findOne.mockResolvedValue(null); - - await expect(service.requeueDeadLetterWebhook('non-existent')).rejects.toThrow( - 'Webhook retry not found', - ); + describe('requeueDeadLetterWebhook', () => { + it('should requeue a dead letter webhook', async () => { + const webhookId = '550e8400-e29b-41d4-a716-446655440000'; + const deadLetterWebhook = { + id: webhookId, + status: WebhookStatus.DEAD_LETTER, + provider: WebhookProvider.STRIPE, + externalEventId: 'evt_123', + payload: { test: 'data' }, + retryCount: 3, + }; + mockRepository.findOne + .mockResolvedValueOnce(deadLetterWebhook) + .mockResolvedValueOnce(deadLetterWebhook); + mockRepository.save.mockResolvedValue({ + ...deadLetterWebhook, + status: WebhookStatus.PENDING, + retryCount: 0, + }); + mockQueue.add.mockResolvedValue({ id: 3 }); + await service.requeueDeadLetterWebhook(webhookId); + expect(mockRepository.save).toHaveBeenCalled(); + expect(mockQueue.add).toHaveBeenCalled(); + }); + it('should throw error if webhook not found', async () => { + mockRepository.findOne.mockResolvedValue(null); + await expect(service.requeueDeadLetterWebhook('non-existent')).rejects.toThrow('Webhook retry not found'); + }); + it('should throw error if webhook is not in dead letter status', async () => { + const webhook = { + id: '550e8400-e29b-41d4-a716-446655440000', + status: WebhookStatus.PENDING, + }; + mockRepository.findOne.mockResolvedValue(webhook); + await expect(service.requeueDeadLetterWebhook(webhook.id)).rejects.toThrow('Webhook is not in dead letter status'); + }); }); - - it('should throw error if webhook is not in dead letter status', async () => { - const webhook = { - id: '550e8400-e29b-41d4-a716-446655440000', - status: WebhookStatus.PENDING, - }; - - mockRepository.findOne.mockResolvedValue(webhook); - - await expect(service.requeueDeadLetterWebhook(webhook.id)).rejects.toThrow( - 'Webhook is not in dead letter status', - ); - }); - }); }); diff --git a/src/payments/webhooks/webhook-queue.service.ts b/src/payments/webhooks/webhook-queue.service.ts index 2f4fc953..3ba81d0a 100644 --- a/src/payments/webhooks/webhook-queue.service.ts +++ b/src/payments/webhooks/webhook-queue.service.ts @@ -5,179 +5,157 @@ import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; import { Repository } from 'typeorm'; import { InjectRepository } from '@nestjs/typeorm'; import { WebhookRetry, WebhookStatus, WebhookProvider } from './entities/webhook-retry.entity'; - export interface WebhookQueuePayload { - webhookRetryId: string; - provider: WebhookProvider; - payload: Buffer | Record; - signature?: string; - externalEventId: string; - headers?: Record; + webhookRetryId: string; + provider: WebhookProvider; + payload: Buffer | Record; + signature?: string; + externalEventId: string; + headers?: Record; } - @Injectable() export class WebhookQueueService { - private readonly logger = new Logger(WebhookQueueService.name); - - constructor( + private readonly logger = new Logger(WebhookQueueService.name); + constructor( @InjectQueue(QUEUE_NAMES.WEBHOOKS) - private readonly webhookQueue: Queue, + private readonly webhookQueue: Queue, @InjectRepository(WebhookRetry) - private readonly webhookRetryRepository: Repository, - ) {} - - /** - * Queue a webhook for processing with retry logic - */ - async queueWebhook(payload: WebhookQueuePayload): Promise { - try { - // Check if webhook already exists (idempotency) - const existingWebhook = await this.webhookRetryRepository.findOne({ - where: { - externalEventId: payload.externalEventId, - provider: payload.provider, - }, - }); - - let webhookRetry: WebhookRetry; - - if (existingWebhook) { - // Update existing webhook retry - existingWebhook.status = WebhookStatus.PENDING; - existingWebhook.payload = payload.payload as Record; - existingWebhook.signature = payload.signature; - existingWebhook.retryCount = 0; - existingWebhook.headers = payload.headers; - webhookRetry = await this.webhookRetryRepository.save(existingWebhook); - this.logger.log(`Updated webhook retry: ${webhookRetry.id}`); - } else { - // Create new webhook retry record - webhookRetry = this.webhookRetryRepository.create({ - provider: payload.provider, - externalEventId: payload.externalEventId, - payload: payload.payload as Record, - signature: payload.signature, - status: WebhookStatus.PENDING, - retryCount: 0, - headers: payload.headers, + private readonly webhookRetryRepository: Repository) { } + /** + * Queue a webhook for processing with retry logic + */ + async queueWebhook(payload: WebhookQueuePayload): Promise { + try { + // Check if webhook already exists (idempotency) + const existingWebhook = await this.webhookRetryRepository.findOne({ + where: { + externalEventId: payload.externalEventId, + provider: payload.provider, + }, + }); + let webhookRetry: WebhookRetry; + if (existingWebhook) { + // Update existing webhook retry + existingWebhook.status = WebhookStatus.PENDING; + existingWebhook.payload = payload.payload as Record; + existingWebhook.signature = payload.signature; + existingWebhook.retryCount = 0; + existingWebhook.headers = payload.headers; + webhookRetry = await this.webhookRetryRepository.save(existingWebhook); + this.logger.log(`Updated webhook retry: ${webhookRetry.id}`); + } + else { + // Create new webhook retry record + webhookRetry = this.webhookRetryRepository.create({ + provider: payload.provider, + externalEventId: payload.externalEventId, + payload: payload.payload as Record, + signature: payload.signature, + status: WebhookStatus.PENDING, + retryCount: 0, + headers: payload.headers, + }); + webhookRetry = await this.webhookRetryRepository.save(webhookRetry); + this.logger.log(`Created new webhook retry: ${webhookRetry.id}`); + } + // Queue the job for processing + const job = await this.webhookQueue.add(JOB_NAMES.PROCESS_WEBHOOK, payload, { + attempts: 1, // Let our processor handle retries + backoff: { + type: 'exponential', + delay: 1000, + }, + removeOnComplete: false, + removeOnFail: false, + }); + this.logger.log(`Webhook queued for processing. Retry ID: ${webhookRetry.id}, Job ID: ${job.id}`); + return webhookRetry.id; + } + catch (error: unknown) { + this.logger.error(`Failed to queue webhook: ${error.message}`); + throw error; + } + } + /** + * Requeue a failed webhook from the dead letter queue + */ + async requeueDeadLetterWebhook(webhookRetryId: string): Promise { + try { + const webhookRetry = await this.webhookRetryRepository.findOne({ + where: { id: webhookRetryId }, + }); + if (!webhookRetry) { + throw new Error(`Webhook retry not found: ${webhookRetryId}`); + } + if (webhookRetry.status !== WebhookStatus.DEAD_LETTER) { + throw new Error(`Webhook is not in dead letter status. Current status: ${webhookRetry.status}`); + } + // Reset retry count and status + webhookRetry.status = WebhookStatus.PENDING; + webhookRetry.retryCount = 0; + webhookRetry.lastError = null; + webhookRetry.errorDetails = null; + webhookRetry.nextRetryTime = null; + await this.webhookRetryRepository.save(webhookRetry); + // Re-queue the job + const payload: WebhookQueuePayload = { + webhookRetryId: webhookRetry.id, + provider: webhookRetry.provider, + payload: webhookRetry.payload, + signature: webhookRetry.signature, + externalEventId: webhookRetry.externalEventId, + headers: webhookRetry.headers, + }; + await this.webhookQueue.add('process-webhook', payload, { + attempts: 1, + backoff: { + type: 'exponential', + delay: 1000, + }, + removeOnComplete: false, + removeOnFail: false, + }); + this.logger.log(`Requeued webhook from dead letter: ${webhookRetryId}`); + } + catch (error: unknown) { + this.logger.error(`Failed to requeue webhook: ${error.message}`); + throw error; + } + } + /** + * Get webhook retry status + */ + async getWebhookStatus(webhookRetryId: string): Promise { + return this.webhookRetryRepository.findOne({ + where: { id: webhookRetryId }, + }); + } + /** + * Get all dead letter webhooks + */ + async getDeadLetterWebhooks(limit: number = 100): Promise { + return this.webhookRetryRepository.find({ + where: { status: WebhookStatus.DEAD_LETTER }, + order: { createdAt: 'DESC' }, + take: limit, }); - webhookRetry = await this.webhookRetryRepository.save(webhookRetry); - this.logger.log(`Created new webhook retry: ${webhookRetry.id}`); - } - - // Queue the job for processing - const job = await this.webhookQueue.add(JOB_NAMES.PROCESS_WEBHOOK, payload, { - attempts: 1, // Let our processor handle retries - backoff: { - type: 'exponential', - delay: 1000, - }, - removeOnComplete: false, - removeOnFail: false, - }); - - this.logger.log( - `Webhook queued for processing. Retry ID: ${webhookRetry.id}, Job ID: ${job.id}`, - ); - - return webhookRetry.id; - } catch (error: any) { - this.logger.error(`Failed to queue webhook: ${error.message}`); - throw error; } - } - - /** - * Requeue a failed webhook from the dead letter queue - */ - async requeueDeadLetterWebhook(webhookRetryId: string): Promise { - try { - const webhookRetry = await this.webhookRetryRepository.findOne({ - where: { id: webhookRetryId }, - }); - - if (!webhookRetry) { - throw new Error(`Webhook retry not found: ${webhookRetryId}`); - } - - if (webhookRetry.status !== WebhookStatus.DEAD_LETTER) { - throw new Error( - `Webhook is not in dead letter status. Current status: ${webhookRetry.status}`, - ); - } - - // Reset retry count and status - webhookRetry.status = WebhookStatus.PENDING; - webhookRetry.retryCount = 0; - webhookRetry.lastError = null; - webhookRetry.errorDetails = null; - webhookRetry.nextRetryTime = null; - await this.webhookRetryRepository.save(webhookRetry); - - // Re-queue the job - const payload: WebhookQueuePayload = { - webhookRetryId: webhookRetry.id, - provider: webhookRetry.provider, - payload: webhookRetry.payload, - signature: webhookRetry.signature, - externalEventId: webhookRetry.externalEventId, - headers: webhookRetry.headers, - }; - - await this.webhookQueue.add('process-webhook', payload, { - attempts: 1, - backoff: { - type: 'exponential', - delay: 1000, - }, - removeOnComplete: false, - removeOnFail: false, - }); - - this.logger.log(`Requeued webhook from dead letter: ${webhookRetryId}`); - } catch (error: any) { - this.logger.error(`Failed to requeue webhook: ${error.message}`); - throw error; + /** + * Get pending webhooks + */ + async getPendingWebhooks(limit: number = 100): Promise { + return this.webhookRetryRepository.find({ + where: { status: WebhookStatus.PENDING }, + order: { createdAt: 'ASC' }, + take: limit, + }); + } + /** + * Get processing webhooks + */ + async getProcessingWebhooks(): Promise { + return this.webhookRetryRepository.find({ + where: { status: WebhookStatus.PROCESSING }, + }); } - } - - /** - * Get webhook retry status - */ - async getWebhookStatus(webhookRetryId: string): Promise { - return this.webhookRetryRepository.findOne({ - where: { id: webhookRetryId }, - }); - } - - /** - * Get all dead letter webhooks - */ - async getDeadLetterWebhooks(limit: number = 100): Promise { - return this.webhookRetryRepository.find({ - where: { status: WebhookStatus.DEAD_LETTER }, - order: { createdAt: 'DESC' }, - take: limit, - }); - } - - /** - * Get pending webhooks - */ - async getPendingWebhooks(limit: number = 100): Promise { - return this.webhookRetryRepository.find({ - where: { status: WebhookStatus.PENDING }, - order: { createdAt: 'ASC' }, - take: limit, - }); - } - - /** - * Get processing webhooks - */ - async getProcessingWebhooks(): Promise { - return this.webhookRetryRepository.find({ - where: { status: WebhookStatus.PROCESSING }, - }); - } } diff --git a/src/payments/webhooks/webhook-retry.e2e-spec.ts b/src/payments/webhooks/webhook-retry.e2e-spec.ts index a2627797..14afd83d 100644 --- a/src/payments/webhooks/webhook-retry.e2e-spec.ts +++ b/src/payments/webhooks/webhook-retry.e2e-spec.ts @@ -4,12 +4,7 @@ import request from 'supertest'; import { TypeOrmModule } from '@nestjs/typeorm'; import { BullModule } from '@nestjs/bull'; import { PaymentsModule } from '../payments.module'; -import { - WebhookRetry, - WebhookStatus, - WebhookProvider, -} from '../webhooks/entities/webhook-retry.entity'; - +import { WebhookRetry, WebhookStatus, WebhookProvider, } from '../webhooks/entities/webhook-retry.entity'; /** * Integration Tests for Webhook Retry System * @@ -18,229 +13,194 @@ import { * * npm test -- webhook-retry.e2e-spec */ - describe('Webhook Retry System (e2e)', () => { - let app: INestApplication; - let webhookRepository; - - beforeAll(async () => { - const moduleFixture: TestingModule = await Test.createTestingModule({ - imports: [ - TypeOrmModule.forRoot({ - type: 'postgres', - host: process.env.DATABASE_HOST || 'localhost', - port: parseInt(process.env.DATABASE_PORT || '5432'), - username: process.env.DATABASE_USER || 'postgres', - password: process.env.DATABASE_PASSWORD || 'postgres', - database: process.env.DATABASE_NAME || 'teachlink_test', - entities: [WebhookRetry], - synchronize: true, - dropSchema: true, - }), - BullModule.forRoot({ - redis: { - host: process.env.REDIS_HOST || 'localhost', - port: parseInt(process.env.REDIS_PORT || '6379'), - }, - }), - PaymentsModule, - ], - }).compile(); - - app = moduleFixture.createNestApplication(); - await app.init(); - - webhookRepository = moduleFixture.get('WebhookRetryRepository'); - }); - - afterAll(async () => { - await app.close(); - }); - - describe('POST /webhooks/stripe', () => { - it('should queue a Stripe webhook and return retry ID', async () => { - const payload = JSON.stringify({ - type: 'payment_intent.succeeded', - id: 'evt_1234567890', - data: { - object: { - id: 'pi_123456', - metadata: {}, - }, - }, - }); - - const response = await request(app.getHttpServer()) - .post('/webhooks/stripe') - .set('stripe-signature', 'test-signature') - .send(payload) - .expect(200); - - expect(response.body.received).toBe(true); - expect(response.body.webhookRetryId).toBeDefined(); - - // Verify webhook was created in database - const webhook = await webhookRepository.findOne({ - where: { id: response.body.webhookRetryId }, - }); - - expect(webhook).toBeDefined(); - expect(webhook.provider).toBe(WebhookProvider.STRIPE); - expect(webhook.status).toBe(WebhookStatus.PENDING); - expect(webhook.retryCount).toBe(0); + let app: INestApplication; + let webhookRepository; + beforeAll(async () => { + const moduleFixture: TestingModule = await Test.createTestingModule({ + imports: [ + TypeOrmModule.forRoot({ + type: 'postgres', + host: process.env.DATABASE_HOST || 'localhost', + port: parseInt(process.env.DATABASE_PORT || '5432'), + username: process.env.DATABASE_USER || 'postgres', + password: process.env.DATABASE_PASSWORD || 'postgres', + database: process.env.DATABASE_NAME || 'teachlink_test', + entities: [WebhookRetry], + synchronize: true, + dropSchema: true, + }), + BullModule.forRoot({ + redis: { + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379'), + }, + }), + PaymentsModule, + ], + }).compile(); + app = moduleFixture.createNestApplication(); + await app.init(); + webhookRepository = moduleFixture.get('WebhookRetryRepository'); }); - - it('should reject webhook with invalid signature', async () => { - const payload = JSON.stringify({ - type: 'payment_intent.succeeded', - id: 'evt_9876543210', - data: { - object: { - id: 'pi_789012', - metadata: {}, - }, - }, - }); - - await request(app.getHttpServer()) - .post('/webhooks/stripe') - .set('stripe-signature', 'invalid-signature') - .send(payload) - .expect(400); + afterAll(async () => { + await app.close(); }); - }); - - describe('POST /webhooks/paypal', () => { - it('should queue a PayPal webhook and return retry ID', async () => { - const payload = { - event_type: 'PAYMENT.SALE.COMPLETED', - resource: { - id: 'sale_123456', - parent_payment: 'payment_123456', - amount: 100.0, - }, - }; - - const response = await request(app.getHttpServer()) - .post('/webhooks/paypal') - .set('paypal-transmission-id', 'trans_123') - .set('paypal-transmission-time', new Date().toISOString()) - .set('paypal-transmission-sig', 'sig_123') - .set('paypal-cert-url', 'https://example.com/cert.pem') - .set('paypal-auth-algo', 'SHA256withRSA') - .send(payload) - .expect(200); - - expect(response.body.received).toBe(true); - expect(response.body.webhookRetryId).toBeDefined(); + describe('POST /webhooks/stripe', () => { + it('should queue a Stripe webhook and return retry ID', async () => { + const payload = JSON.stringify({ + type: 'payment_intent.succeeded', + id: 'evt_1234567890', + data: { + object: { + id: 'pi_123456', + metadata: {}, + }, + }, + }); + const response = await request(app.getHttpServer()) + .post('/webhooks/stripe') + .set('stripe-signature', 'test-signature') + .send(payload) + .expect(200); + expect(response.body.received).toBe(true); + expect(response.body.webhookRetryId).toBeDefined(); + // Verify webhook was created in database + const webhook = await webhookRepository.findOne({ + where: { id: response.body.webhookRetryId }, + }); + expect(webhook).toBeDefined(); + expect(webhook.provider).toBe(WebhookProvider.STRIPE); + expect(webhook.status).toBe(WebhookStatus.PENDING); + expect(webhook.retryCount).toBe(0); + }); + it('should reject webhook with invalid signature', async () => { + const payload = JSON.stringify({ + type: 'payment_intent.succeeded', + id: 'evt_9876543210', + data: { + object: { + id: 'pi_789012', + metadata: {}, + }, + }, + }); + await request(app.getHttpServer()) + .post('/webhooks/stripe') + .set('stripe-signature', 'invalid-signature') + .send(payload) + .expect(400); + }); }); - }); - - describe('GET /webhooks/status/:id', () => { - it('should return webhook status', async () => { - const webhook = new WebhookRetry(); - webhook.provider = WebhookProvider.STRIPE; - webhook.externalEventId = 'evt_status_test'; - webhook.status = WebhookStatus.SUCCEEDED; - webhook.retryCount = 0; - - const saved = await webhookRepository.save(webhook); - - const response = await request(app.getHttpServer()) - .get(`/webhooks/status/${saved.id}`) - .expect(200); - - expect(response.body.id).toBe(saved.id); - expect(response.body.status).toBe(WebhookStatus.SUCCEEDED); + describe('POST /webhooks/paypal', () => { + it('should queue a PayPal webhook and return retry ID', async () => { + const payload = { + event_type: 'PAYMENT.SALE.COMPLETED', + resource: { + id: 'sale_123456', + parent_payment: 'payment_123456', + amount: 100.0, + }, + }; + const response = await request(app.getHttpServer()) + .post('/webhooks/paypal') + .set('paypal-transmission-id', 'trans_123') + .set('paypal-transmission-time', new Date().toISOString()) + .set('paypal-transmission-sig', 'sig_123') + .set('paypal-cert-url', 'https://example.com/cert.pem') + .set('paypal-auth-algo', 'SHA256withRSA') + .send(payload) + .expect(200); + expect(response.body.received).toBe(true); + expect(response.body.webhookRetryId).toBeDefined(); + }); }); - }); - - describe('GET /webhooks/dead-letter', () => { - it('should return dead letter webhooks', async () => { - // Create a dead letter webhook - const webhook = new WebhookRetry(); - webhook.provider = WebhookProvider.STRIPE; - webhook.externalEventId = 'evt_dead_letter_test'; - webhook.status = WebhookStatus.DEAD_LETTER; - webhook.retryCount = 3; - webhook.lastError = 'Network timeout'; - - await webhookRepository.save(webhook); - - const response = await request(app.getHttpServer()).get('/webhooks/dead-letter').expect(200); - - expect(Array.isArray(response.body)).toBe(true); - expect(response.body.length).toBeGreaterThan(0); - expect(response.body[0].status).toBe(WebhookStatus.DEAD_LETTER); + describe('GET /webhooks/status/:id', () => { + it('should return webhook status', async () => { + const webhook = new WebhookRetry(); + webhook.provider = WebhookProvider.STRIPE; + webhook.externalEventId = 'evt_status_test'; + webhook.status = WebhookStatus.SUCCEEDED; + webhook.retryCount = 0; + const saved = await webhookRepository.save(webhook); + const response = await request(app.getHttpServer()) + .get(`/webhooks/status/${saved.id}`) + .expect(200); + expect(response.body.id).toBe(saved.id); + expect(response.body.status).toBe(WebhookStatus.SUCCEEDED); + }); }); - }); - - describe('POST /webhooks/requeue/:id', () => { - it('should requeue a dead letter webhook', async () => { - // Create a dead letter webhook - const webhook = new WebhookRetry(); - webhook.provider = WebhookProvider.STRIPE; - webhook.externalEventId = 'evt_requeue_test'; - webhook.status = WebhookStatus.DEAD_LETTER; - webhook.retryCount = 3; - webhook.payload = {}; - - const saved = await webhookRepository.save(webhook); - - const response = await request(app.getHttpServer()) - .post(`/webhooks/requeue/${saved.id}`) - .expect(200); - - expect(response.body.success).toBe(true); - - // Verify webhook status was updated - const updated = await webhookRepository.findOne({ - where: { id: saved.id }, - }); - - expect(updated.status).toBe(WebhookStatus.PENDING); - expect(updated.retryCount).toBe(0); + describe('GET /webhooks/dead-letter', () => { + it('should return dead letter webhooks', async () => { + // Create a dead letter webhook + const webhook = new WebhookRetry(); + webhook.provider = WebhookProvider.STRIPE; + webhook.externalEventId = 'evt_dead_letter_test'; + webhook.status = WebhookStatus.DEAD_LETTER; + webhook.retryCount = 3; + webhook.lastError = 'Network timeout'; + await webhookRepository.save(webhook); + const response = await request(app.getHttpServer()).get('/webhooks/dead-letter').expect(200); + expect(Array.isArray(response.body)).toBe(true); + expect(response.body.length).toBeGreaterThan(0); + expect(response.body[0].status).toBe(WebhookStatus.DEAD_LETTER); + }); }); - }); - - describe('Webhook Idempotency', () => { - it('should not create duplicate webhooks for the same event', async () => { - const payload = JSON.stringify({ - type: 'payment_intent.succeeded', - id: 'evt_idempotency_test', - data: { - object: { - id: 'pi_idempotency', - metadata: {}, - }, - }, - }); - - // Send the same webhook twice - const response1 = await request(app.getHttpServer()) - .post('/webhooks/stripe') - .set('stripe-signature', 'test-signature') - .send(payload) - .expect(200); - - const response2 = await request(app.getHttpServer()) - .post('/webhooks/stripe') - .set('stripe-signature', 'test-signature') - .send(payload) - .expect(200); - - // Both should return the same webhook ID (update existing) - expect(response1.body.webhookRetryId).toBe(response2.body.webhookRetryId); - - // Verify only one webhook was created - const count = await webhookRepository.count({ - where: { - provider: WebhookProvider.STRIPE, - externalEventId: 'evt_idempotency_test', - }, - }); - - expect(count).toBe(1); + describe('POST /webhooks/requeue/:id', () => { + it('should requeue a dead letter webhook', async () => { + // Create a dead letter webhook + const webhook = new WebhookRetry(); + webhook.provider = WebhookProvider.STRIPE; + webhook.externalEventId = 'evt_requeue_test'; + webhook.status = WebhookStatus.DEAD_LETTER; + webhook.retryCount = 3; + webhook.payload = {}; + const saved = await webhookRepository.save(webhook); + const response = await request(app.getHttpServer()) + .post(`/webhooks/requeue/${saved.id}`) + .expect(200); + expect(response.body.success).toBe(true); + // Verify webhook status was updated + const updated = await webhookRepository.findOne({ + where: { id: saved.id }, + }); + expect(updated.status).toBe(WebhookStatus.PENDING); + expect(updated.retryCount).toBe(0); + }); + }); + describe('Webhook Idempotency', () => { + it('should not create duplicate webhooks for the same event', async () => { + const payload = JSON.stringify({ + type: 'payment_intent.succeeded', + id: 'evt_idempotency_test', + data: { + object: { + id: 'pi_idempotency', + metadata: {}, + }, + }, + }); + // Send the same webhook twice + const response1 = await request(app.getHttpServer()) + .post('/webhooks/stripe') + .set('stripe-signature', 'test-signature') + .send(payload) + .expect(200); + const response2 = await request(app.getHttpServer()) + .post('/webhooks/stripe') + .set('stripe-signature', 'test-signature') + .send(payload) + .expect(200); + // Both should return the same webhook ID (update existing) + expect(response1.body.webhookRetryId).toBe(response2.body.webhookRetryId); + // Verify only one webhook was created + const count = await webhookRepository.count({ + where: { + provider: WebhookProvider.STRIPE, + externalEventId: 'evt_idempotency_test', + }, + }); + expect(count).toBe(1); + }); }); - }); }); diff --git a/src/payments/webhooks/webhook-retry.processor.spec.ts b/src/payments/webhooks/webhook-retry.processor.spec.ts index 3df63446..63a8eaff 100644 --- a/src/payments/webhooks/webhook-retry.processor.spec.ts +++ b/src/payments/webhooks/webhook-retry.processor.spec.ts @@ -5,175 +5,151 @@ import { WebhookRetryProcessor } from './webhook-retry.processor'; import { WebhookRetry, WebhookStatus, WebhookProvider } from './entities/webhook-retry.entity'; import { ProviderFactoryService } from '../providers/provider-factory.service'; import { PaymentsService } from '../payments.service'; - describe('WebhookRetryProcessor', () => { - let processor: WebhookRetryProcessor; - let repository: Repository; - let providerFactory: ProviderFactoryService; - let paymentsService: PaymentsService; - - const mockRepository = { - findOne: jest.fn(), - save: jest.fn(), - }; - - const mockProviderFactory = { - getProvider: jest.fn(), - }; - - const mockPaymentsService = { - updatePaymentStatus: jest.fn(), - processRefundFromWebhook: jest.fn(), - handleSubscriptionEvent: jest.fn(), - }; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [ - WebhookRetryProcessor, - { - provide: getRepositoryToken(WebhookRetry), - useValue: mockRepository, - }, - { - provide: ProviderFactoryService, - useValue: mockProviderFactory, - }, - { - provide: PaymentsService, - useValue: mockPaymentsService, - }, - ], - }).compile(); - - processor = module.get(WebhookRetryProcessor); - repository = module.get>(getRepositoryToken(WebhookRetry)); - providerFactory = module.get(ProviderFactoryService); - paymentsService = module.get(PaymentsService); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - describe('processWebhook', () => { - it('should process webhook successfully', async () => { - const webhookRetryId = '550e8400-e29b-41d4-a716-446655440000'; - const webhookRetry = { - id: webhookRetryId, - status: WebhookStatus.PENDING, - provider: WebhookProvider.STRIPE, - payload: {}, - signature: 'test-signature', - retryCount: 0, - maxRetries: 3, - }; - - const mockStripeProvider = { - handleWebhook: jest.fn().mockResolvedValue({ - type: 'payment_intent.succeeded', - id: 'evt_123', - data: { - object: { - id: 'pi_123', - metadata: {}, - }, - }, - }), - }; - - mockRepository.findOne - .mockResolvedValueOnce(webhookRetry) - .mockResolvedValueOnce(webhookRetry); - mockProviderFactory.getProvider.mockReturnValue(mockStripeProvider); - mockPaymentsService.updatePaymentStatus.mockResolvedValue(undefined); - mockRepository.save.mockResolvedValue({ - ...webhookRetry, - status: WebhookStatus.SUCCEEDED, - processedAt: new Date(), - }); - - const job = { - data: { - webhookRetryId, - provider: WebhookProvider.STRIPE, - payload: Buffer.from('test'), - signature: 'test-signature', - externalEventId: 'evt_123', - }, - } as any; - - await processor.processWebhook(job); - - expect(mockRepository.findOne).toHaveBeenCalled(); - expect(mockRepository.save).toHaveBeenCalled(); + let processor: WebhookRetryProcessor; + let repository: Repository; + let providerFactory: ProviderFactoryService; + let paymentsService: PaymentsService; + const mockRepository = { + findOne: jest.fn(), + save: jest.fn(), + }; + const mockProviderFactory = { + getProvider: jest.fn(), + }; + const mockPaymentsService = { + updatePaymentStatus: jest.fn(), + processRefundFromWebhook: jest.fn(), + handleSubscriptionEvent: jest.fn(), + }; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + providers: [ + WebhookRetryProcessor, + { + provide: getRepositoryToken(WebhookRetry), + useValue: mockRepository, + }, + { + provide: ProviderFactoryService, + useValue: mockProviderFactory, + }, + { + provide: PaymentsService, + useValue: mockPaymentsService, + }, + ], + }).compile(); + processor = module.get(WebhookRetryProcessor); + repository = module.get>(getRepositoryToken(WebhookRetry)); + providerFactory = module.get(ProviderFactoryService); + paymentsService = module.get(PaymentsService); }); - - it('should handle webhook processing error with retry', async () => { - const webhookRetryId = '550e8400-e29b-41d4-a716-446655440000'; - const webhookRetry = { - id: webhookRetryId, - status: WebhookStatus.PROCESSING, - provider: WebhookProvider.STRIPE, - retryCount: 0, - maxRetries: 3, - lastError: null, - errorDetails: null, - nextRetryTime: null, - }; - - const mockStripeProvider = { - handleWebhook: jest.fn().mockRejectedValue(new Error('Signature verification failed')), - }; - - mockRepository.findOne.mockResolvedValueOnce(webhookRetry); - mockProviderFactory.getProvider.mockReturnValue(mockStripeProvider); - mockRepository.findOne.mockResolvedValueOnce(webhookRetry); - mockRepository.save.mockResolvedValue({ - ...webhookRetry, - retryCount: 1, - lastError: 'Signature verification failed', - status: WebhookStatus.PENDING, - }); - - const job = { - data: { - webhookRetryId, - provider: WebhookProvider.STRIPE, - payload: Buffer.from('test'), - signature: 'test-signature', - externalEventId: 'evt_123', - }, - retry: jest.fn().mockResolvedValue(undefined), - } as any; - - await processor.processWebhook(job); - - expect(mockRepository.save).toHaveBeenCalled(); - expect(job.retry).toHaveBeenCalled(); + afterEach(() => { + jest.clearAllMocks(); }); - }); - - describe('exponential backoff calculation', () => { - it('should calculate correct exponential backoff', async () => { - const delays: number[] = []; - - // Access the private method through reflection for testing - const calculateNextRetryTime = (processor as any).calculateNextRetryTime.bind(processor); - - for (let i = 0; i < 5; i++) { - const delay = calculateNextRetryTime(i); - delays.push(delay); - } - - // Verify exponential growth (with jitter tolerance) - expect(delays[0]).toBeLessThan(delays[1]); - expect(delays[1]).toBeLessThan(delays[2]); - expect(delays[2]).toBeLessThan(delays[3]); - expect(delays[3]).toBeLessThan(delays[4]); - - // Max delay should be capped at 1 hour - expect(delays[4]).toBeLessThanOrEqual(3600000); + describe('processWebhook', () => { + it('should process webhook successfully', async () => { + const webhookRetryId = '550e8400-e29b-41d4-a716-446655440000'; + const webhookRetry = { + id: webhookRetryId, + status: WebhookStatus.PENDING, + provider: WebhookProvider.STRIPE, + payload: {}, + signature: 'test-signature', + retryCount: 0, + maxRetries: 3, + }; + const mockStripeProvider = { + handleWebhook: jest.fn().mockResolvedValue({ + type: 'payment_intent.succeeded', + id: 'evt_123', + data: { + object: { + id: 'pi_123', + metadata: {}, + }, + }, + }), + }; + mockRepository.findOne + .mockResolvedValueOnce(webhookRetry) + .mockResolvedValueOnce(webhookRetry); + mockProviderFactory.getProvider.mockReturnValue(mockStripeProvider); + mockPaymentsService.updatePaymentStatus.mockResolvedValue(undefined); + mockRepository.save.mockResolvedValue({ + ...webhookRetry, + status: WebhookStatus.SUCCEEDED, + processedAt: new Date(), + }); + const job = { + data: { + webhookRetryId, + provider: WebhookProvider.STRIPE, + payload: Buffer.from('test'), + signature: 'test-signature', + externalEventId: 'evt_123', + }, + } as unknown; + await processor.processWebhook(job); + expect(mockRepository.findOne).toHaveBeenCalled(); + expect(mockRepository.save).toHaveBeenCalled(); + }); + it('should handle webhook processing error with retry', async () => { + const webhookRetryId = '550e8400-e29b-41d4-a716-446655440000'; + const webhookRetry = { + id: webhookRetryId, + status: WebhookStatus.PROCESSING, + provider: WebhookProvider.STRIPE, + retryCount: 0, + maxRetries: 3, + lastError: null, + errorDetails: null, + nextRetryTime: null, + }; + const mockStripeProvider = { + handleWebhook: jest.fn().mockRejectedValue(new Error('Signature verification failed')), + }; + mockRepository.findOne.mockResolvedValueOnce(webhookRetry); + mockProviderFactory.getProvider.mockReturnValue(mockStripeProvider); + mockRepository.findOne.mockResolvedValueOnce(webhookRetry); + mockRepository.save.mockResolvedValue({ + ...webhookRetry, + retryCount: 1, + lastError: 'Signature verification failed', + status: WebhookStatus.PENDING, + }); + const job = { + data: { + webhookRetryId, + provider: WebhookProvider.STRIPE, + payload: Buffer.from('test'), + signature: 'test-signature', + externalEventId: 'evt_123', + }, + retry: jest.fn().mockResolvedValue(undefined), + } as unknown; + await processor.processWebhook(job); + expect(mockRepository.save).toHaveBeenCalled(); + expect(job.retry).toHaveBeenCalled(); + }); + }); + describe('exponential backoff calculation', () => { + it('should calculate correct exponential backoff', async () => { + const delays: number[] = []; + // Access the private method through reflection for testing + const calculateNextRetryTime = (processor as unknown).calculateNextRetryTime.bind(processor); + for (let i = 0; i < 5; i++) { + const delay = calculateNextRetryTime(i); + delays.push(delay); + } + // Verify exponential growth (with jitter tolerance) + expect(delays[0]).toBeLessThan(delays[1]); + expect(delays[1]).toBeLessThan(delays[2]); + expect(delays[2]).toBeLessThan(delays[3]); + expect(delays[3]).toBeLessThan(delays[4]); + // Max delay should be capped at 1 hour + expect(delays[4]).toBeLessThanOrEqual(3600000); + }); }); - }); }); diff --git a/src/payments/webhooks/webhook-retry.processor.ts b/src/payments/webhooks/webhook-retry.processor.ts index 89c8d0fe..a30102ab 100644 --- a/src/payments/webhooks/webhook-retry.processor.ts +++ b/src/payments/webhooks/webhook-retry.processor.ts @@ -7,259 +7,190 @@ import { Repository } from 'typeorm'; import { WebhookRetry, WebhookStatus, WebhookProvider } from './entities/webhook-retry.entity'; import { ProviderFactoryService } from '../providers/provider-factory.service'; import { PaymentsService } from '../payments.service'; -import { - SubscriptionWebhookEvent, - RefundWebhookData, -} from '../interfaces/payment-provider.interface'; +import { SubscriptionWebhookEvent, RefundWebhookData, } from '../interfaces/payment-provider.interface'; import { PaymentStatus } from '../entities/payment.entity'; - interface StripePaymentIntent { - id: string; - metadata: Record; + id: string; + metadata: Record; } - interface StripeCharge { - payment_intent: string; - refunds: { - data: RefundWebhookData[]; - }; + payment_intent: string; + refunds: { + data: RefundWebhookData[]; + }; } - interface PayPalResource { - id: string; - parent_payment: string; - amount: number; + id: string; + parent_payment: string; + amount: number; } - interface PayPalWebhookPayload { - event_type: string; - resource: PayPalResource; + event_type: string; + resource: PayPalResource; } - interface WebhookJobData { - webhookRetryId: string; - provider: WebhookProvider; - payload: Buffer | Record; - signature?: string; - externalEventId: string; - headers?: Record; + webhookRetryId: string; + provider: WebhookProvider; + payload: Buffer | Record; + signature?: string; + externalEventId: string; + headers?: Record; } - @Injectable() @Processor(QUEUE_NAMES.WEBHOOKS) export class WebhookRetryProcessor { - private readonly logger = new Logger(WebhookRetryProcessor.name); - - // Exponential backoff configuration - private readonly initialDelayMs = 1000; // 1 second - private readonly maxDelayMs = 3600000; // 1 hour - private readonly backoffMultiplier = 2; - - constructor( + private readonly logger = new Logger(WebhookRetryProcessor.name); + // Exponential backoff configuration + private readonly initialDelayMs = 1000; // 1 second + private readonly maxDelayMs = 3600000; // 1 hour + private readonly backoffMultiplier = 2; + constructor( @InjectRepository(WebhookRetry) - private readonly webhookRetryRepository: Repository, - private readonly providerFactory: ProviderFactoryService, - private readonly paymentsService: PaymentsService, - ) {} - - @Process(JOB_NAMES.PROCESS_WEBHOOK) - async processWebhook(job: Job) { - const { - webhookRetryId, - provider, - payload, - signature, - externalEventId: _externalEventId, - headers, - } = job.data; - - try { - // Update status to processing - let webhookRetry = await this.webhookRetryRepository.findOne({ - where: { id: webhookRetryId }, - }); - - if (!webhookRetry) { - this.logger.warn(`Webhook retry record not found: ${webhookRetryId}`); - return; - } - - webhookRetry.status = WebhookStatus.PROCESSING; - await this.webhookRetryRepository.save(webhookRetry); - - // Process the webhook based on provider - if (provider === WebhookProvider.STRIPE) { - await this.handleStripeWebhook(payload as Buffer, signature, webhookRetryId); - } else if (provider === WebhookProvider.PAYPAL) { - await this.handlePayPalWebhook(payload as Record, headers, webhookRetryId); - } - - // Mark as succeeded - webhookRetry = await this.webhookRetryRepository.findOne({ - where: { id: webhookRetryId }, - }); - webhookRetry.status = WebhookStatus.SUCCEEDED; - webhookRetry.processedAt = new Date(); - await this.webhookRetryRepository.save(webhookRetry); - - this.logger.log(`Webhook processed successfully: ${webhookRetryId}`); - } catch (error: any) { - await this.handleWebhookError(webhookRetryId, error, job); + private readonly webhookRetryRepository: Repository, private readonly providerFactory: ProviderFactoryService, private readonly paymentsService: PaymentsService) { } + @Process(JOB_NAMES.PROCESS_WEBHOOK) + async processWebhook(job: Job) { + const { webhookRetryId, provider, payload, signature, externalEventId: _externalEventId, headers, } = job.data; + try { + // Update status to processing + let webhookRetry = await this.webhookRetryRepository.findOne({ + where: { id: webhookRetryId }, + }); + if (!webhookRetry) { + this.logger.warn(`Webhook retry record not found: ${webhookRetryId}`); + return; + } + webhookRetry.status = WebhookStatus.PROCESSING; + await this.webhookRetryRepository.save(webhookRetry); + // Process the webhook based on provider + if (provider === WebhookProvider.STRIPE) { + await this.handleStripeWebhook(payload as Buffer, signature, webhookRetryId); + } + else if (provider === WebhookProvider.PAYPAL) { + await this.handlePayPalWebhook(payload as Record, headers, webhookRetryId); + } + // Mark as succeeded + webhookRetry = await this.webhookRetryRepository.findOne({ + where: { id: webhookRetryId }, + }); + webhookRetry.status = WebhookStatus.SUCCEEDED; + webhookRetry.processedAt = new Date(); + await this.webhookRetryRepository.save(webhookRetry); + this.logger.log(`Webhook processed successfully: ${webhookRetryId}`); + } + catch (error: unknown) { + await this.handleWebhookError(webhookRetryId, error, job); + } } - } - - private async handleWebhookError( - webhookRetryId: string, - error: Error, - job: Job, - ): Promise { - const webhookRetry = await this.webhookRetryRepository.findOne({ - where: { id: webhookRetryId }, - }); - - if (!webhookRetry) { - this.logger.error(`Webhook retry record not found for error handling: ${webhookRetryId}`); - return; + private async handleWebhookError(webhookRetryId: string, error: Error, job: Job): Promise { + const webhookRetry = await this.webhookRetryRepository.findOne({ + where: { id: webhookRetryId }, + }); + if (!webhookRetry) { + this.logger.error(`Webhook retry record not found for error handling: ${webhookRetryId}`); + return; + } + webhookRetry.retryCount += 1; + webhookRetry.lastError = error.message; + webhookRetry.errorDetails = { + stack: error.stack, + timestamp: new Date().toISOString(), + retryCount: webhookRetry.retryCount, + }; + // Check if we should retry + if (webhookRetry.retryCount < webhookRetry.maxRetries) { + // Calculate next retry time with exponential backoff + const nextRetryTime = this.calculateNextRetryTime(webhookRetry.retryCount); + webhookRetry.nextRetryTime = new Date(Date.now() + nextRetryTime); + webhookRetry.status = WebhookStatus.PENDING; + await this.webhookRetryRepository.save(webhookRetry); + this.logger.warn(`Webhook ${webhookRetryId} will be retried at ${webhookRetry.nextRetryTime}. Retry ${webhookRetry.retryCount}/${webhookRetry.maxRetries}`); + // Re-queue the job with delay (Bull's retry() takes no arguments in this version) + await job.retry(); + } + else { + // All retries exhausted - move to dead letter + webhookRetry.status = WebhookStatus.DEAD_LETTER; + await this.webhookRetryRepository.save(webhookRetry); + this.logger.error(`Webhook ${webhookRetryId} moved to dead letter after ${webhookRetry.retryCount} attempts. Error: ${error.message}`); + // You can emit an event here for alerting operations team + // this.eventEmitter.emit('webhook.dead-letter', { webhookRetryId, error }); + } } - - webhookRetry.retryCount += 1; - webhookRetry.lastError = error.message; - webhookRetry.errorDetails = { - stack: error.stack, - timestamp: new Date().toISOString(), - retryCount: webhookRetry.retryCount, - }; - - // Check if we should retry - if (webhookRetry.retryCount < webhookRetry.maxRetries) { - // Calculate next retry time with exponential backoff - const nextRetryTime = this.calculateNextRetryTime(webhookRetry.retryCount); - webhookRetry.nextRetryTime = new Date(Date.now() + nextRetryTime); - webhookRetry.status = WebhookStatus.PENDING; - - await this.webhookRetryRepository.save(webhookRetry); - this.logger.warn( - `Webhook ${webhookRetryId} will be retried at ${webhookRetry.nextRetryTime}. Retry ${webhookRetry.retryCount}/${webhookRetry.maxRetries}`, - ); - - // Re-queue the job with delay (Bull's retry() takes no arguments in this version) - await job.retry(); - } else { - // All retries exhausted - move to dead letter - webhookRetry.status = WebhookStatus.DEAD_LETTER; - await this.webhookRetryRepository.save(webhookRetry); - - this.logger.error( - `Webhook ${webhookRetryId} moved to dead letter after ${webhookRetry.retryCount} attempts. Error: ${error.message}`, - ); - - // You can emit an event here for alerting operations team - // this.eventEmitter.emit('webhook.dead-letter', { webhookRetryId, error }); + private calculateNextRetryTime(retryCount: number): number { + // Exponential backoff: initialDelay * (backoffMultiplier ^ retryCount) + const delay = this.initialDelayMs * Math.pow(this.backoffMultiplier, retryCount); + // Add jitter to prevent thundering herd + const jitter = Math.random() * 0.1 * delay; + const totalDelay = Math.min(delay + jitter, this.maxDelayMs); + return totalDelay; + } + private async handleStripeWebhook(payload: Buffer, signature: string, _webhookRetryId: string): Promise { + if (!payload) { + throw new Error('Missing payload for Stripe webhook'); + } + let event: unknown; + try { + const stripeProvider = this.providerFactory.getProvider('stripe'); + event = await stripeProvider.handleWebhook(payload, signature); + } + catch (err: unknown) { + this.logger.error(`Stripe webhook signature verification failed: ${err.message}`); + throw new Error(`Webhook signature verification failed: ${err.message}`); + } + this.logger.log(`Processing Stripe webhook: ${event.type}`); + switch (event.type) { + case 'payment_intent.succeeded': + await this.handlePaymentIntentSucceeded(event.data.object as StripePaymentIntent); + break; + case 'payment_intent.payment_failed': + await this.handlePaymentIntentFailed(event.data.object as StripePaymentIntent); + break; + case 'charge.refunded': + await this.handleChargeRefunded(event.data.object as StripeCharge); + break; + case 'customer.subscription.created': + case 'customer.subscription.updated': + case 'customer.subscription.deleted': + await this.handleSubscriptionEvent(event as unknown as SubscriptionWebhookEvent); + break; + default: + this.logger.log(`Unhandled Stripe event type: ${event.type}`); + } + } + private async handlePaymentIntentSucceeded(paymentIntent: StripePaymentIntent): Promise { + await this.paymentsService.updatePaymentStatus(paymentIntent.id, PaymentStatus.COMPLETED, paymentIntent.metadata); + } + private async handlePaymentIntentFailed(paymentIntent: StripePaymentIntent): Promise { + await this.paymentsService.updatePaymentStatus(paymentIntent.id, PaymentStatus.FAILED, paymentIntent.metadata); + } + private async handleChargeRefunded(charge: StripeCharge): Promise { + const refund = charge.refunds.data[0]; + await this.paymentsService.processRefundFromWebhook(charge.payment_intent, refund); } - } - - private calculateNextRetryTime(retryCount: number): number { - // Exponential backoff: initialDelay * (backoffMultiplier ^ retryCount) - const delay = this.initialDelayMs * Math.pow(this.backoffMultiplier, retryCount); - // Add jitter to prevent thundering herd - const jitter = Math.random() * 0.1 * delay; - const totalDelay = Math.min(delay + jitter, this.maxDelayMs); - return totalDelay; - } - - private async handleStripeWebhook( - payload: Buffer, - signature: string, - _webhookRetryId: string, - ): Promise { - if (!payload) { - throw new Error('Missing payload for Stripe webhook'); + private async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { + await this.paymentsService.handleSubscriptionEvent(event); } - - let event: any; - try { - const stripeProvider = this.providerFactory.getProvider('stripe'); - event = await stripeProvider.handleWebhook(payload, signature); - } catch (err: any) { - this.logger.error(`Stripe webhook signature verification failed: ${err.message}`); - throw new Error(`Webhook signature verification failed: ${err.message}`); + private async handlePayPalWebhook(payload: Record, _headers: Record, _webhookRetryId: string): Promise { + const paypalPayload = payload as unknown as PayPalWebhookPayload; + this.logger.log(`Processing PayPal webhook: ${paypalPayload.event_type}`); + switch (paypalPayload.event_type) { + case 'PAYMENT.SALE.COMPLETED': + await this.handlePayPalPaymentCompleted(paypalPayload.resource); + break; + case 'PAYMENT.SALE.REFUNDED': + await this.handlePayPalRefundCompleted(paypalPayload.resource); + break; + default: + this.logger.log(`Unhandled PayPal event type: ${paypalPayload.event_type}`); + } } - - this.logger.log(`Processing Stripe webhook: ${event.type}`); - - switch (event.type) { - case 'payment_intent.succeeded': - await this.handlePaymentIntentSucceeded(event.data.object as StripePaymentIntent); - break; - case 'payment_intent.payment_failed': - await this.handlePaymentIntentFailed(event.data.object as StripePaymentIntent); - break; - case 'charge.refunded': - await this.handleChargeRefunded(event.data.object as StripeCharge); - break; - case 'customer.subscription.created': - case 'customer.subscription.updated': - case 'customer.subscription.deleted': - await this.handleSubscriptionEvent(event as unknown as SubscriptionWebhookEvent); - break; - default: - this.logger.log(`Unhandled Stripe event type: ${event.type}`); + private async handlePayPalPaymentCompleted(resource: PayPalResource): Promise { + await this.paymentsService.updatePaymentStatus(resource.id, PaymentStatus.COMPLETED); } - } - - private async handlePaymentIntentSucceeded(paymentIntent: StripePaymentIntent): Promise { - await this.paymentsService.updatePaymentStatus( - paymentIntent.id, - PaymentStatus.COMPLETED, - paymentIntent.metadata, - ); - } - - private async handlePaymentIntentFailed(paymentIntent: StripePaymentIntent): Promise { - await this.paymentsService.updatePaymentStatus( - paymentIntent.id, - PaymentStatus.FAILED, - paymentIntent.metadata, - ); - } - - private async handleChargeRefunded(charge: StripeCharge): Promise { - const refund = charge.refunds.data[0]; - await this.paymentsService.processRefundFromWebhook(charge.payment_intent, refund); - } - - private async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { - await this.paymentsService.handleSubscriptionEvent(event); - } - - private async handlePayPalWebhook( - payload: Record, - _headers: Record, - _webhookRetryId: string, - ): Promise { - const paypalPayload = payload as unknown as PayPalWebhookPayload; - this.logger.log(`Processing PayPal webhook: ${paypalPayload.event_type}`); - - switch (paypalPayload.event_type) { - case 'PAYMENT.SALE.COMPLETED': - await this.handlePayPalPaymentCompleted(paypalPayload.resource); - break; - case 'PAYMENT.SALE.REFUNDED': - await this.handlePayPalRefundCompleted(paypalPayload.resource); - break; - default: - this.logger.log(`Unhandled PayPal event type: ${paypalPayload.event_type}`); + private async handlePayPalRefundCompleted(resource: PayPalResource): Promise { + await this.paymentsService.processRefundFromWebhook(resource.parent_payment, { + id: resource.id, + amount: resource.amount, + }); } - } - - private async handlePayPalPaymentCompleted(resource: PayPalResource): Promise { - await this.paymentsService.updatePaymentStatus(resource.id, PaymentStatus.COMPLETED); - } - - private async handlePayPalRefundCompleted(resource: PayPalResource): Promise { - await this.paymentsService.processRefundFromWebhook(resource.parent_payment, { - id: resource.id, - amount: resource.amount, - }); - } } diff --git a/src/payments/webhooks/webhook.controller.ts b/src/payments/webhooks/webhook.controller.ts index 79490711..d0e23a2d 100644 --- a/src/payments/webhooks/webhook.controller.ts +++ b/src/payments/webhooks/webhook.controller.ts @@ -1,54 +1,41 @@ -import { - Controller, - Post, - Headers, - Body, - HttpCode, - HttpStatus, - Req, - RawBodyRequest, -} from '@nestjs/common'; +import { Controller, Post, Headers, Body, HttpCode, HttpStatus, Req, RawBodyRequest, } from '@nestjs/common'; import { Request } from 'express'; import { ApiTags, ApiOperation, ApiResponse } from '@nestjs/swagger'; import { SkipThrottle } from '@nestjs/throttler'; import { WebhookService } from './webhook.service'; - @SkipThrottle() @ApiTags('webhooks') @Controller('webhooks') export class WebhookController { - constructor(private readonly webhookService: WebhookService) {} - - @Post('stripe') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Handle Stripe webhook events' }) - @ApiResponse({ status: 200, description: 'Webhook processed' }) - async handleStripeWebhook( - @Headers('stripe-signature') signature: string, - @Req() req: RawBodyRequest, - ): Promise { - return this.webhookService.handleStripeWebhook(req.rawBody, signature); - } - - @Post('paypal') - @HttpCode(HttpStatus.OK) - @ApiOperation({ summary: 'Handle PayPal webhook events' }) - @ApiResponse({ status: 200, description: 'Webhook processed' }) - async handlePayPalWebhook( - @Headers('paypal-transmission-id') transmissionId: string, - @Headers('paypal-transmission-time') transmissionTime: string, - @Headers('paypal-transmission-sig') transmissionSig: string, - @Headers('paypal-cert-url') certUrl: string, - @Headers('paypal-auth-algo') authAlgo: string, - @Body() payload: any, - ): Promise { - return this.webhookService.handlePayPalWebhook( - payload, - transmissionId, - transmissionTime, - transmissionSig, - certUrl, - authAlgo, - ); - } + constructor(private readonly webhookService: WebhookService) { } + @Post('stripe') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Handle Stripe webhook events' }) + @ApiResponse({ status: 200, description: 'Webhook processed' }) + async handleStripeWebhook( + @Headers('stripe-signature') + signature: string, + @Req() + req: RawBodyRequest): Promise { + return this.webhookService.handleStripeWebhook(req.rawBody, signature); + } + @Post('paypal') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: 'Handle PayPal webhook events' }) + @ApiResponse({ status: 200, description: 'Webhook processed' }) + async handlePayPalWebhook( + @Headers('paypal-transmission-id') + transmissionId: string, + @Headers('paypal-transmission-time') + transmissionTime: string, + @Headers('paypal-transmission-sig') + transmissionSig: string, + @Headers('paypal-cert-url') + certUrl: string, + @Headers('paypal-auth-algo') + authAlgo: string, + @Body() + payload: unknown): Promise { + return this.webhookService.handlePayPalWebhook(payload, transmissionId, transmissionTime, transmissionSig, certUrl, authAlgo); + } } diff --git a/src/payments/webhooks/webhook.service.ts b/src/payments/webhooks/webhook.service.ts index 1003f8bd..cfd66601 100644 --- a/src/payments/webhooks/webhook.service.ts +++ b/src/payments/webhooks/webhook.service.ts @@ -4,155 +4,120 @@ import { PaymentsService } from '../payments.service'; import { PaymentStatus } from '../entities/payment.entity'; import { WebhookQueueService } from './webhook-queue.service'; import { WebhookProvider } from './entities/webhook-retry.entity'; -import { - SubscriptionWebhookEvent, - RefundWebhookData, -} from '../interfaces/payment-provider.interface'; - +import { SubscriptionWebhookEvent, RefundWebhookData, } from '../interfaces/payment-provider.interface'; interface StripePaymentIntent { - id: string; - metadata: Record; + id: string; + metadata: Record; } - interface StripeCharge { - payment_intent: string; - refunds: { - data: RefundWebhookData[]; - }; + payment_intent: string; + refunds: { + data: RefundWebhookData[]; + }; } - interface PayPalResource { - id: string; - parent_payment: string; - amount: number; + id: string; + parent_payment: string; + amount: number; } - interface PayPalWebhookPayload { - event_type: string; - resource: PayPalResource; + event_type: string; + resource: PayPalResource; } - @Injectable() export class WebhookService { - private readonly logger = new Logger(WebhookService.name); - - constructor( - private readonly paymentsService: PaymentsService, - private readonly providerFactory: ProviderFactoryService, - private readonly webhookQueueService: WebhookQueueService, - ) {} - - async handleStripeWebhook( - payload: Buffer | undefined, - signature: string, - ): Promise<{ received: boolean; webhookRetryId?: string }> { - if (!payload) { - this.logger.error('Missing raw body in Stripe webhook'); - return { received: false }; + private readonly logger = new Logger(WebhookService.name); + constructor(private readonly paymentsService: PaymentsService, private readonly providerFactory: ProviderFactoryService, private readonly webhookQueueService: WebhookQueueService) { } + async handleStripeWebhook(payload: Buffer | undefined, signature: string): Promise<{ + received: boolean; + webhookRetryId?: string; + }> { + if (!payload) { + this.logger.error('Missing raw body in Stripe webhook'); + return { received: false }; + } + let eventId: string; + try { + // Verify signature first before queuing + const stripeProvider = this.providerFactory.getProvider('stripe'); + const event = await stripeProvider.handleWebhook(payload, signature); + eventId = event.id; + } + catch (err: unknown) { + this.logger.error(`Webhook signature verification failed: ${err.message}`); + throw new BadRequestException('Webhook signature verification failed'); + } + // Queue the webhook for processing with retry logic + try { + const webhookRetryId = await this.webhookQueueService.queueWebhook({ + webhookRetryId: '', // Will be generated by the service + provider: WebhookProvider.STRIPE, + payload, + signature, + externalEventId: eventId, + }); + this.logger.log(`Stripe webhook queued for processing: ${webhookRetryId}`); + return { received: true, webhookRetryId }; + } + catch (error: unknown) { + this.logger.error(`Failed to queue Stripe webhook: ${error.message}`); + throw error; + } } - - let eventId: string; - try { - // Verify signature first before queuing - const stripeProvider = this.providerFactory.getProvider('stripe'); - const event = await stripeProvider.handleWebhook(payload, signature); - eventId = event.id; - } catch (err: any) { - this.logger.error(`Webhook signature verification failed: ${err.message}`); - throw new BadRequestException('Webhook signature verification failed'); + private async handlePaymentIntentSucceeded(paymentIntent: StripePaymentIntent): Promise { + // Update payment status to completed + await this.paymentsService.updatePaymentStatus(paymentIntent.id, PaymentStatus.COMPLETED, paymentIntent.metadata); } - - // Queue the webhook for processing with retry logic - try { - const webhookRetryId = await this.webhookQueueService.queueWebhook({ - webhookRetryId: '', // Will be generated by the service - provider: WebhookProvider.STRIPE, - payload, - signature, - externalEventId: eventId, - }); - - this.logger.log(`Stripe webhook queued for processing: ${webhookRetryId}`); - return { received: true, webhookRetryId }; - } catch (error: any) { - this.logger.error(`Failed to queue Stripe webhook: ${error.message}`); - throw error; + private async handlePaymentIntentFailed(paymentIntent: StripePaymentIntent): Promise { + // Update payment status to failed + await this.paymentsService.updatePaymentStatus(paymentIntent.id, PaymentStatus.FAILED, paymentIntent.metadata); } - } - - private async handlePaymentIntentSucceeded(paymentIntent: StripePaymentIntent): Promise { - // Update payment status to completed - await this.paymentsService.updatePaymentStatus( - paymentIntent.id, - PaymentStatus.COMPLETED, - paymentIntent.metadata, - ); - } - - private async handlePaymentIntentFailed(paymentIntent: StripePaymentIntent): Promise { - // Update payment status to failed - await this.paymentsService.updatePaymentStatus( - paymentIntent.id, - PaymentStatus.FAILED, - paymentIntent.metadata, - ); - } - - private async handleChargeRefunded(charge: StripeCharge): Promise { - // Process refund - const refund = charge.refunds.data[0]; - await this.paymentsService.processRefundFromWebhook(charge.payment_intent, refund); - } - - private async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { - // Handle subscription events - await this.paymentsService.handleSubscriptionEvent(event); - } - - async handlePayPalWebhook( - payload: Record, - _transmissionId: string, - _transmissionTime: string, - _transmissionSig: string, - _certUrl: string, - _authAlgo: string, - ): Promise<{ received: boolean; webhookRetryId?: string }> { - const paypalPayload = payload as unknown as PayPalWebhookPayload; - - try { - // Queue the webhook for processing with retry logic - const webhookRetryId = await this.webhookQueueService.queueWebhook({ - webhookRetryId: '', // Will be generated by the service - provider: WebhookProvider.PAYPAL, - payload, - externalEventId: paypalPayload.event_type, - headers: { - 'paypal-transmission-id': _transmissionId, - 'paypal-transmission-time': _transmissionTime, - 'paypal-transmission-sig': _transmissionSig, - 'paypal-cert-url': _certUrl, - 'paypal-auth-algo': _authAlgo, - }, - }); - - this.logger.log(`PayPal webhook queued for processing: ${webhookRetryId}`); - return { received: true, webhookRetryId }; - } catch (error: any) { - this.logger.error(`Failed to queue PayPal webhook: ${error.message}`); - throw error; + private async handleChargeRefunded(charge: StripeCharge): Promise { + // Process refund + const refund = charge.refunds.data[0]; + await this.paymentsService.processRefundFromWebhook(charge.payment_intent, refund); + } + private async handleSubscriptionEvent(event: SubscriptionWebhookEvent): Promise { + // Handle subscription events + await this.paymentsService.handleSubscriptionEvent(event); + } + async handlePayPalWebhook(payload: Record, _transmissionId: string, _transmissionTime: string, _transmissionSig: string, _certUrl: string, _authAlgo: string): Promise<{ + received: boolean; + webhookRetryId?: string; + }> { + const paypalPayload = payload as unknown as PayPalWebhookPayload; + try { + // Queue the webhook for processing with retry logic + const webhookRetryId = await this.webhookQueueService.queueWebhook({ + webhookRetryId: '', // Will be generated by the service + provider: WebhookProvider.PAYPAL, + payload, + externalEventId: paypalPayload.event_type, + headers: { + 'paypal-transmission-id': _transmissionId, + 'paypal-transmission-time': _transmissionTime, + 'paypal-transmission-sig': _transmissionSig, + 'paypal-cert-url': _certUrl, + 'paypal-auth-algo': _authAlgo, + }, + }); + this.logger.log(`PayPal webhook queued for processing: ${webhookRetryId}`); + return { received: true, webhookRetryId }; + } + catch (error: unknown) { + this.logger.error(`Failed to queue PayPal webhook: ${error.message}`); + throw error; + } + } + private async handlePayPalPaymentCompleted(resource: PayPalResource): Promise { + // Update payment status to completed + await this.paymentsService.updatePaymentStatus(resource.id, PaymentStatus.COMPLETED); + } + private async handlePayPalRefundCompleted(resource: PayPalResource): Promise { + // Process refund + await this.paymentsService.processRefundFromWebhook(resource.parent_payment, { + id: resource.id, + amount: resource.amount, + }); } - } - - private async handlePayPalPaymentCompleted(resource: PayPalResource): Promise { - // Update payment status to completed - await this.paymentsService.updatePaymentStatus(resource.id, PaymentStatus.COMPLETED); - } - - private async handlePayPalRefundCompleted(resource: PayPalResource): Promise { - // Process refund - await this.paymentsService.processRefundFromWebhook(resource.parent_payment, { - id: resource.id, - amount: resource.amount, - }); - } } diff --git a/src/queues/dto/queue.dto.ts b/src/queues/dto/queue.dto.ts index afc17c3d..968579e7 100644 --- a/src/queues/dto/queue.dto.ts +++ b/src/queues/dto/queue.dto.ts @@ -1,242 +1,193 @@ -import { - IsString, - IsOptional, - IsObject, - IsNumber, - IsDateString, - IsEnum, - IsArray, - IsBoolean, - Min, - Max, - ValidateNested, - IsNotEmpty, -} from 'class-validator'; +import { IsString, IsOptional, IsObject, IsNumber, IsDateString, IsEnum, IsArray, IsBoolean, Min, Max, ValidateNested, IsNotEmpty, } from 'class-validator'; import { Type } from 'class-transformer'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { JobPriority } from '../enums/job-priority.enum'; - // ── Sub-DTOs ───────────────────────────────────────────────────────────────── - export class BackoffOptionsDto { - @ApiProperty({ enum: ['fixed', 'exponential'] }) - @IsEnum(['fixed', 'exponential']) - type: 'fixed' | 'exponential'; - - @ApiProperty({ minimum: 0 }) - @IsNumber() - @Min(0) - delay: number; + @ApiProperty({ enum: ['fixed', 'exponential'] }) + @IsEnum(['fixed', 'exponential']) + type: 'fixed' | 'exponential'; + @ApiProperty({ minimum: 0 }) + @IsNumber() + @Min(0) + delay: number; } - export class JobOptionsDto { - @ApiPropertyOptional({ enum: JobPriority }) - @IsOptional() - @IsEnum(JobPriority) - priority?: JobPriority; - - @ApiPropertyOptional({ minimum: 1, maximum: 20 }) - @IsOptional() - @IsNumber() - @Min(1) - @Max(20) - attempts?: number; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => BackoffOptionsDto) - backoff?: BackoffOptionsDto; - - @ApiPropertyOptional({ minimum: 0 }) - @IsOptional() - @IsNumber() - @Min(0) - delay?: number; - - @ApiPropertyOptional({ minimum: 1000 }) - @IsOptional() - @IsNumber() - @Min(1000) - timeout?: number; - - @ApiPropertyOptional() - @IsOptional() - @IsBoolean() - removeOnComplete?: boolean; - - @ApiPropertyOptional() - @IsOptional() - @IsBoolean() - removeOnFail?: boolean; + @ApiPropertyOptional({ enum: JobPriority }) + @IsOptional() + @IsEnum(JobPriority) + priority?: JobPriority; + @ApiPropertyOptional({ minimum: 1, maximum: 20 }) + @IsOptional() + @IsNumber() + @Min(1) + @Max(20) + attempts?: number; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => BackoffOptionsDto) + backoff?: BackoffOptionsDto; + @ApiPropertyOptional({ minimum: 0 }) + @IsOptional() + @IsNumber() + @Min(0) + delay?: number; + @ApiPropertyOptional({ minimum: 1000 }) + @IsOptional() + @IsNumber() + @Min(1000) + timeout?: number; + @ApiPropertyOptional() + @IsOptional() + @IsBoolean() + removeOnComplete?: boolean; + @ApiPropertyOptional() + @IsOptional() + @IsBoolean() + removeOnFail?: boolean; } - export class PriorityFactorsDto { - @ApiPropertyOptional({ enum: ['premium', 'pro', 'basic', 'free'] }) - @IsOptional() - @IsEnum(['premium', 'pro', 'basic', 'free']) - userTier?: 'premium' | 'pro' | 'basic' | 'free'; - - @ApiPropertyOptional({ enum: ['critical', 'high', 'medium', 'low'] }) - @IsOptional() - @IsEnum(['critical', 'high', 'medium', 'low']) - urgency?: 'critical' | 'high' | 'medium' | 'low'; + @ApiPropertyOptional({ enum: ['premium', 'pro', 'basic', 'free'] }) + @IsOptional() + @IsEnum(['premium', 'pro', 'basic', 'free']) + userTier?: 'premium' | 'pro' | 'basic' | 'free'; + @ApiPropertyOptional({ enum: ['critical', 'high', 'medium', 'low'] }) + @IsOptional() + @IsEnum(['critical', 'high', 'medium', 'low']) + urgency?: 'critical' | 'high' | 'medium' | 'low'; } - // ── Request DTOs ────────────────────────────────────────────────────────────── - export class AddJobDto { - @ApiProperty({ description: 'Job type / processor name' }) - @IsString() - @IsNotEmpty() - name: string; - - @ApiProperty({ description: 'Arbitrary job payload' }) - @IsObject() - data: Record; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => JobOptionsDto) - options?: JobOptionsDto; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => PriorityFactorsDto) - priorityFactors?: PriorityFactorsDto; + @ApiProperty({ description: 'Job type / processor name' }) + @IsString() + @IsNotEmpty() + name: string; + @ApiProperty({ description: 'Arbitrary job payload' }) + @IsObject() + data: Record; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => JobOptionsDto) + options?: JobOptionsDto; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => PriorityFactorsDto) + priorityFactors?: PriorityFactorsDto; } - export class BulkJobItemDto { - @ApiProperty() - @IsString() - @IsNotEmpty() - name: string; - - @ApiProperty() - @IsObject() - data: Record; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => JobOptionsDto) - options?: JobOptionsDto; + @ApiProperty() + @IsString() + @IsNotEmpty() + name: string; + @ApiProperty() + @IsObject() + data: Record; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => JobOptionsDto) + options?: JobOptionsDto; } - export class AddBulkJobsDto { - @ApiProperty({ type: [BulkJobItemDto] }) - @IsArray() - @ValidateNested({ each: true }) - @Type(() => BulkJobItemDto) - jobs: BulkJobItemDto[]; + @ApiProperty({ type: [BulkJobItemDto] }) + @IsArray() + @ValidateNested({ each: true }) + @Type(() => BulkJobItemDto) + jobs: BulkJobItemDto[]; } - export class ScheduleJobDto { - @ApiProperty() - @IsString() - @IsNotEmpty() - name: string; - - @ApiProperty() - @IsObject() - data: Record; - - @ApiProperty({ description: 'ISO-8601 datetime string (must be in the future)' }) - @IsDateString() - scheduledTime: string; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => JobOptionsDto) - options?: JobOptionsDto; + @ApiProperty() + @IsString() + @IsNotEmpty() + name: string; + @ApiProperty() + @IsObject() + data: Record; + @ApiProperty({ description: 'ISO-8601 datetime string (must be in the future)' }) + @IsDateString() + scheduledTime: string; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => JobOptionsDto) + options?: JobOptionsDto; } - export class ScheduleDelayedJobDto { - @ApiProperty() - @IsString() - @IsNotEmpty() - name: string; - - @ApiProperty() - @IsObject() - data: Record; - - @ApiProperty({ description: 'Delay in milliseconds', minimum: 0 }) - @IsNumber() - @Min(0) - delayMs: number; - - @ApiPropertyOptional() - @IsOptional() - @ValidateNested() - @Type(() => JobOptionsDto) - options?: JobOptionsDto; + @ApiProperty() + @IsString() + @IsNotEmpty() + name: string; + @ApiProperty() + @IsObject() + data: Record; + @ApiProperty({ description: 'Delay in milliseconds', minimum: 0 }) + @IsNumber() + @Min(0) + delayMs: number; + @ApiPropertyOptional() + @IsOptional() + @ValidateNested() + @Type(() => JobOptionsDto) + options?: JobOptionsDto; } - export class CleanQueueDto { - @ApiPropertyOptional({ description: 'Grace period in ms (default 5000)', minimum: 0 }) - @IsOptional() - @IsNumber() - @Min(0) - grace?: number; - - @ApiPropertyOptional({ enum: ['completed', 'failed'] }) - @IsOptional() - @IsEnum(['completed', 'failed']) - status?: 'completed' | 'failed'; + @ApiPropertyOptional({ description: 'Grace period in ms (default 5000)', minimum: 0 }) + @IsOptional() + @IsNumber() + @Min(0) + grace?: number; + @ApiPropertyOptional({ enum: ['completed', 'failed'] }) + @IsOptional() + @IsEnum(['completed', 'failed']) + status?: 'completed' | 'failed'; } - export class FailedJobsQueryDto { - @ApiPropertyOptional({ - description: 'Maximum number of jobs to return', - minimum: 1, - maximum: 200, - }) - @IsOptional() - @Type(() => Number) - @IsNumber() - @Min(1) - @Max(200) - limit?: number; - - @ApiPropertyOptional({ description: 'Page offset (0-based)', minimum: 0 }) - @IsOptional() - @Type(() => Number) - @IsNumber() - @Min(0) - offset?: number; - - @ApiPropertyOptional({ description: 'Filter by job name / type' }) - @IsOptional() - @IsString() - jobName?: string; + @ApiPropertyOptional({ + description: 'Maximum number of jobs to return', + minimum: 1, + maximum: 200, + }) + @IsOptional() + @Type(() => Number) + @IsNumber() + @Min(1) + @Max(200) + limit?: number; + @ApiPropertyOptional({ description: 'Page offset (0-based)', minimum: 0 }) + @IsOptional() + @Type(() => Number) + @IsNumber() + @Min(0) + offset?: number; + @ApiPropertyOptional({ description: 'Filter by job name / type' }) + @IsOptional() + @IsString() + jobName?: string; } - export class StuckJobsQueryDto { - @ApiPropertyOptional({ - description: 'Threshold in ms — jobs active longer are considered stuck', - minimum: 5000, - }) - @IsOptional() - @Type(() => Number) - @IsNumber() - @Min(5000) - threshold?: number; + @ApiPropertyOptional({ + description: 'Threshold in ms — jobs active longer are considered stuck', + minimum: 5000, + }) + @IsOptional() + @Type(() => Number) + @IsNumber() + @Min(5000) + threshold?: number; } - export class AnalyticsQueryDto { - @ApiPropertyOptional({ - description: 'Lookback window in minutes (default: 60)', - minimum: 1, - maximum: 1440, - }) - @IsOptional() - @Type(() => Number) - @IsNumber() - @Min(1) - @Max(1440) - windowMinutes?: number; + @ApiPropertyOptional({ + description: 'Lookback window in minutes (default: 60)', + minimum: 1, + maximum: 1440, + }) + @IsOptional() + @Type(() => Number) + @IsNumber() + @Min(1) + @Max(1440) + windowMinutes?: number; } diff --git a/src/queues/enums/job-priority.enum.ts b/src/queues/enums/job-priority.enum.ts index efcfc289..e07a71dd 100644 --- a/src/queues/enums/job-priority.enum.ts +++ b/src/queues/enums/job-priority.enum.ts @@ -1,16 +1,15 @@ export enum JobPriority { - CRITICAL = 1, - HIGH = 2, - NORMAL = 3, - LOW = 4, - BACKGROUND = 5, + CRITICAL = 1, + HIGH = 2, + NORMAL = 3, + LOW = 4, + BACKGROUND = 5 } - export enum JobStatus { - WAITING = 'waiting', - ACTIVE = 'active', - COMPLETED = 'completed', - FAILED = 'failed', - DELAYED = 'delayed', - PAUSED = 'paused', + WAITING = 'waiting', + ACTIVE = 'active', + COMPLETED = 'completed', + FAILED = 'failed', + DELAYED = 'delayed', + PAUSED = 'paused' } diff --git a/src/queues/interfaces/queue.interfaces.ts b/src/queues/interfaces/queue.interfaces.ts index bf08bf22..8e80e162 100644 --- a/src/queues/interfaces/queue.interfaces.ts +++ b/src/queues/interfaces/queue.interfaces.ts @@ -1,52 +1,47 @@ import { JobPriority, JobStatus } from '../enums/job-priority.enum'; - export interface JobOptions { - priority?: JobPriority; - attempts?: number; - backoff?: number | BackoffOptions; - delay?: number; - timeout?: number; - removeOnComplete?: boolean | number; - removeOnFail?: boolean | number; + priority?: JobPriority; + attempts?: number; + backoff?: number | BackoffOptions; + delay?: number; + timeout?: number; + removeOnComplete?: boolean | number; + removeOnFail?: boolean | number; } - export interface BackoffOptions { - type: 'fixed' | 'exponential'; - delay: number; + type: 'fixed' | 'exponential'; + delay: number; } - export interface JobMetrics { - jobId: string; - name: string; - status: JobStatus; - priority: JobPriority; - attempts: number; - maxAttempts: number; - progress: number; - createdAt: Date; - processedAt?: Date; - finishedAt?: Date; - failedReason?: string; - data: any; + jobId: string; + name: string; + status: JobStatus; + priority: JobPriority; + attempts: number; + maxAttempts: number; + progress: number; + createdAt: Date; + processedAt?: Date; + finishedAt?: Date; + failedReason?: string; + data: unknown; } - export interface QueueMetrics { - queueName: string; - waiting: number; - active: number; - completed: number; - failed: number; - delayed: number; - paused: number; - total: number; - throughput: number; - avgProcessingTime: number; + queueName: string; + waiting: number; + active: number; + completed: number; + failed: number; + delayed: number; + paused: number; + total: number; + throughput: number; + avgProcessingTime: number; } - export interface RetryStrategy { - maxAttempts: number; - backoffType: 'fixed' | 'exponential'; - initialDelay: number; - maxDelay?: number; - multiplier?: number; + maxAttempts: number; + backoffType: 'fixed' | 'exponential'; + initialDelay: number; + maxDelay?: number; + multiplier?: number; } diff --git a/src/queues/monitoring/queue-monitoring.service.ts b/src/queues/monitoring/queue-monitoring.service.ts index aa6f52e9..2d1647b5 100644 --- a/src/queues/monitoring/queue-monitoring.service.ts +++ b/src/queues/monitoring/queue-monitoring.service.ts @@ -3,55 +3,48 @@ import { InjectQueue } from '@nestjs/bull'; import { Queue, Job } from 'bull'; import { Cron, CronExpression } from '@nestjs/schedule'; import { QueueMetrics } from '../interfaces/queue.interfaces'; - // ── Exported interfaces - export interface TimestampedMetrics extends QueueMetrics { - /** Unix timestamp (ms) when this snapshot was taken */ - capturedAt: number; + /** Unix timestamp (ms) when this snapshot was taken */ + capturedAt: number; } - export interface QueueHealthStatus { - status: 'healthy' | 'warning' | 'critical'; - issues: string[]; - metrics: QueueMetrics; - timestamp: Date; + status: 'healthy' | 'warning' | 'critical'; + issues: string[]; + metrics: QueueMetrics; + timestamp: Date; } - export interface QueueStatistics { - current: QueueMetrics; - trends: { - completed: 'up' | 'down' | 'stable'; - failed: 'up' | 'down' | 'stable'; - throughput: 'up' | 'down' | 'stable'; - }; - health: QueueHealthStatus; + current: QueueMetrics; + trends: { + completed: 'up' | 'down' | 'stable'; + failed: 'up' | 'down' | 'stable'; + throughput: 'up' | 'down' | 'stable'; + }; + health: QueueHealthStatus; } - export interface RetryAnalytics { - windowMinutes: number; - totalFailed: number; - totalRetried: number; - successAfterRetry: number; - permanentlyFailed: number; - retryRate: number; // fraction of failed jobs that were retried - successAfterRetryRate: number; - byJobType: Record< - string, - { - failed: number; - retried: number; - avgAttempts: number; - } - >; + windowMinutes: number; + totalFailed: number; + totalRetried: number; + successAfterRetry: number; + permanentlyFailed: number; + retryRate: number; // fraction of failed jobs that were retried + successAfterRetryRate: number; + byJobType: Record; } - export interface BulkRetryResult { - requeued: number; - skipped: number; - errors: Array<{ jobId: string | number; reason: string }>; + requeued: number; + skipped: number; + errors: Array<{ + jobId: string | number; + reason: string; + }>; } - /** * Queue Monitoring Servic * Responsibilities: @@ -64,353 +57,317 @@ export interface BulkRetryResult { */ @Injectable() export class QueueMonitoringService { - private readonly logger = new Logger(QueueMonitoringService.name); - - /** Sliding window of metric snapshots (default: last 100 = ~100 minutes) */ - private readonly metricsHistory: TimestampedMetrics[] = []; - private readonly MAX_HISTORY_SIZE = 100; - - // Health thresholds — tune per environment - private readonly THRESHOLDS = { - failureRateCritical: 0.2, // 20 % - failureRateWarning: 0.1, // 10 % - backlogCritical: 5_000, - backlogWarning: 1_000, - activeJobsCritical: 500, - activeJobsWarning: 100, - delayedJobsWarning: 500, - stuckThresholdMs: 300_000, // 5 min - } as const; - - constructor(@InjectQueue('default') private readonly defaultQueue: Queue) {} - - /** - * Capture and return the current queue metrics. - * Appends a timestamped snapshot to the in-memory history window. - */ - async getQueueMetrics(): Promise { - const [waiting, active, completed, failed, delayed, paused] = await Promise.all([ - this.defaultQueue.getWaitingCount(), - this.defaultQueue.getActiveCount(), - this.defaultQueue.getCompletedCount(), - this.defaultQueue.getFailedCount(), - this.defaultQueue.getDelayedCount(), - this.defaultQueue.getPausedCount(), - ]); - - const total = waiting + active + completed + failed + delayed + paused; - const capturedAt = Date.now(); - - const throughput = this.calculateThroughput(completed, capturedAt); - const avgProcessingTime = await this.calculateAvgProcessingTime(); - - const metrics: TimestampedMetrics = { - queueName: 'default', - waiting, - active, - completed, - failed, - delayed, - paused, - total, - throughput, - avgProcessingTime, - capturedAt, - }; - - this.appendToHistory(metrics); - return metrics; - } - - /** - * Return the in-memory metrics history. - * Each entry includes a `capturedAt` timestamp so callers can draw charts. - */ - getMetricsHistory(): TimestampedMetrics[] { - return [...this.metricsHistory]; - } - - async checkQueueHealth(): Promise { - const metrics = await this.getQueueMetrics(); - const issues: string[] = []; - let status: 'healthy' | 'warning' | 'critical' = 'healthy'; - - const failureRate = metrics.total > 0 ? metrics.failed / metrics.total : 0; - - if (failureRate > this.THRESHOLDS.failureRateCritical) { - issues.push(`Critical failure rate: ${(failureRate * 100).toFixed(1)}%`); - status = 'critical'; - } else if (failureRate > this.THRESHOLDS.failureRateWarning) { - issues.push(`Elevated failure rate: ${(failureRate * 100).toFixed(1)}%`); - this.escalate(status, 'warning'); - status = 'warning'; + private readonly logger = new Logger(QueueMonitoringService.name); + /** Sliding window of metric snapshots (default: last 100 = ~100 minutes) */ + private readonly metricsHistory: TimestampedMetrics[] = []; + private readonly MAX_HISTORY_SIZE = 100; + // Health thresholds — tune per environment + private readonly THRESHOLDS = { + failureRateCritical: 0.2, // 20 % + failureRateWarning: 0.1, // 10 % + backlogCritical: 5000, + backlogWarning: 1000, + activeJobsCritical: 500, + activeJobsWarning: 100, + delayedJobsWarning: 500, + stuckThresholdMs: 300000, // 5 min + } as const; + constructor( + @InjectQueue('default') + private readonly defaultQueue: Queue) { } + /** + * Capture and return the current queue metrics. + * Appends a timestamped snapshot to the in-memory history window. + */ + async getQueueMetrics(): Promise { + const [waiting, active, completed, failed, delayed, paused] = await Promise.all([ + this.defaultQueue.getWaitingCount(), + this.defaultQueue.getActiveCount(), + this.defaultQueue.getCompletedCount(), + this.defaultQueue.getFailedCount(), + this.defaultQueue.getDelayedCount(), + this.defaultQueue.getPausedCount(), + ]); + const total = waiting + active + completed + failed + delayed + paused; + const capturedAt = Date.now(); + const throughput = this.calculateThroughput(completed, capturedAt); + const avgProcessingTime = await this.calculateAvgProcessingTime(); + const metrics: TimestampedMetrics = { + queueName: 'default', + waiting, + active, + completed, + failed, + delayed, + paused, + total, + throughput, + avgProcessingTime, + capturedAt, + }; + this.appendToHistory(metrics); + return metrics; } - - if (metrics.waiting > this.THRESHOLDS.backlogCritical) { - issues.push(`Critical backlog: ${metrics.waiting} waiting jobs`); - status = 'critical'; - } else if (metrics.waiting > this.THRESHOLDS.backlogWarning) { - issues.push(`Elevated backlog: ${metrics.waiting} waiting jobs`); - if (status === 'healthy') status = 'warning'; + /** + * Return the in-memory metrics history. + * Each entry includes a `capturedAt` timestamp so callers can draw charts. + */ + getMetricsHistory(): TimestampedMetrics[] { + return [...this.metricsHistory]; } - - if (metrics.active > this.THRESHOLDS.activeJobsCritical) { - issues.push(`Critical active-job count: ${metrics.active}`); - status = 'critical'; - } else if (metrics.active > this.THRESHOLDS.activeJobsWarning) { - issues.push(`High active-job count: ${metrics.active}`); - if (status === 'healthy') status = 'warning'; + async checkQueueHealth(): Promise { + const metrics = await this.getQueueMetrics(); + const issues: string[] = []; + let status: 'healthy' | 'warning' | 'critical' = 'healthy'; + const failureRate = metrics.total > 0 ? metrics.failed / metrics.total : 0; + if (failureRate > this.THRESHOLDS.failureRateCritical) { + issues.push(`Critical failure rate: ${(failureRate * 100).toFixed(1)}%`); + status = 'critical'; + } + else if (failureRate > this.THRESHOLDS.failureRateWarning) { + issues.push(`Elevated failure rate: ${(failureRate * 100).toFixed(1)}%`); + this.escalate(status, 'warning'); + status = 'warning'; + } + if (metrics.waiting > this.THRESHOLDS.backlogCritical) { + issues.push(`Critical backlog: ${metrics.waiting} waiting jobs`); + status = 'critical'; + } + else if (metrics.waiting > this.THRESHOLDS.backlogWarning) { + issues.push(`Elevated backlog: ${metrics.waiting} waiting jobs`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.active > this.THRESHOLDS.activeJobsCritical) { + issues.push(`Critical active-job count: ${metrics.active}`); + status = 'critical'; + } + else if (metrics.active > this.THRESHOLDS.activeJobsWarning) { + issues.push(`High active-job count: ${metrics.active}`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.delayed > this.THRESHOLDS.delayedJobsWarning) { + issues.push(`Many delayed jobs: ${metrics.delayed}`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.throughput === 0 && metrics.waiting > 0 && metrics.active === 0) { + issues.push('Queue appears stalled: jobs waiting but none active and throughput is zero'); + if (status === 'healthy') + status = 'warning'; + } + return { status, issues, metrics, timestamp: new Date() }; } - - if (metrics.delayed > this.THRESHOLDS.delayedJobsWarning) { - issues.push(`Many delayed jobs: ${metrics.delayed}`); - if (status === 'healthy') status = 'warning'; + async getQueueStatistics(): Promise { + const metrics = await this.getQueueMetrics(); + const history = this.metricsHistory; + return { + current: metrics, + trends: { + completed: this.calculateTrend(history.map((m) => m.completed)), + failed: this.calculateTrend(history.map((m) => m.failed)), + throughput: this.calculateTrend(history.map((m) => m.throughput)), + }, + health: await this.checkQueueHealth(), + }; } - - if (metrics.throughput === 0 && metrics.waiting > 0 && metrics.active === 0) { - issues.push('Queue appears stalled: jobs waiting but none active and throughput is zero'); - if (status === 'healthy') status = 'warning'; + // ── Public API: failed jobs + /** + * Paginated failed-job list with optional name filter. + */ + async getFailedJobs(limit: number = 50, offset: number = 0, jobName?: string): Promise { + if (jobName) { + const all = await this.defaultQueue.getFailed(0, 5000); + const filtered = all.filter((j) => j.name === jobName); + return filtered.slice(offset, offset + limit); + } + return this.defaultQueue.getFailed(offset, offset + limit - 1); } - - return { status, issues, metrics, timestamp: new Date() }; - } - - async getQueueStatistics(): Promise { - const metrics = await this.getQueueMetrics(); - const history = this.metricsHistory; - - return { - current: metrics, - trends: { - completed: this.calculateTrend(history.map((m) => m.completed)), - failed: this.calculateTrend(history.map((m) => m.failed)), - throughput: this.calculateTrend(history.map((m) => m.throughput)), - }, - health: await this.checkQueueHealth(), - }; - } - - // ── Public API: failed jobs - - /** - * Paginated failed-job list with optional name filter. - */ - async getFailedJobs(limit: number = 50, offset: number = 0, jobName?: string): Promise { - if (jobName) { - const all = await this.defaultQueue.getFailed(0, 5_000); - const filtered = all.filter((j) => j.name === jobName); - return filtered.slice(offset, offset + limit); + /** + * Retry every currently-failed job. + * Returns a summary so callers know what happened. + */ + async retryAllFailedJobs(): Promise { + const failed = await this.defaultQueue.getFailed(0, 10000); + const result: BulkRetryResult = { requeued: 0, skipped: 0, errors: [] }; + for (const job of failed) { + try { + await job.retry(); + result.requeued++; + } + catch (err) { + result.errors.push({ + jobId: job.id, + reason: (err as Error).message, + }); + result.skipped++; + } + } + this.logger.log(`Bulk retry complete: ${result.requeued} requeued, ${result.skipped} skipped`); + return result; } - - return this.defaultQueue.getFailed(offset, offset + limit - 1); - } - - /** - * Retry every currently-failed job. - * Returns a summary so callers know what happened. - */ - async retryAllFailedJobs(): Promise { - const failed = await this.defaultQueue.getFailed(0, 10_000); - const result: BulkRetryResult = { requeued: 0, skipped: 0, errors: [] }; - - for (const job of failed) { - try { - await job.retry(); - result.requeued++; - } catch (err) { - result.errors.push({ - jobId: job.id, - reason: (err as Error).message, + // ── Public API: stuck jobs + async getStuckJobs(thresholdMs: number = 300000): Promise { + const activeJobs = await this.defaultQueue.getActive(); + const now = Date.now(); + return activeJobs.filter((job) => { + const startedAt = job.processedOn ?? job.timestamp; + return now - startedAt > thresholdMs; }); - result.skipped++; - } } - - this.logger.log(`Bulk retry complete: ${result.requeued} requeued, ${result.skipped} skipped`); - return result; - } - - // ── Public API: stuck jobs - - async getStuckJobs(thresholdMs: number = 300_000): Promise { - const activeJobs = await this.defaultQueue.getActive(); - const now = Date.now(); - return activeJobs.filter((job) => { - const startedAt = job.processedOn ?? job.timestamp; - return now - startedAt > thresholdMs; - }); - } - - // ── Public API: retry analytics - - /** - * Analyse retry behaviour for jobs that finished within the given window. - * - * Examines the last N completed+failed jobs and produces per-job-type - * breakdowns so engineers can tune `maxAttempts` per queue type. - */ - async getRetryAnalytics(windowMinutes: number = 60): Promise { - const windowMs = windowMinutes * 60 * 1_000; - const cutoff = Date.now() - windowMs; - - const [failedJobs, completedJobs] = await Promise.all([ - this.defaultQueue.getFailed(0, 2_000), - this.defaultQueue.getCompleted(0, 2_000), - ]); - - // Filter to window - const recentFailed = failedJobs.filter((j) => (j.finishedOn ?? j.timestamp) >= cutoff); - const recentCompleted = completedJobs.filter((j) => (j.finishedOn ?? j.timestamp) >= cutoff); - - // Jobs that succeeded after ≥1 retry - const successAfterRetry = recentCompleted.filter((j) => j.attemptsMade > 1); - // Jobs that failed after all retries (attemptsMade >= maxAttempts) - const permanentlyFailed = recentFailed.filter((j) => j.attemptsMade >= (j.opts?.attempts ?? 3)); - const retried = recentFailed.filter((j) => j.attemptsMade > 1); - - // Per-job-type breakdown - const byJobType: RetryAnalytics['byJobType'] = {}; - const allRecent = [...recentFailed, ...recentCompleted]; - for (const job of allRecent) { - const bucket = byJobType[job.name] ?? { failed: 0, retried: 0, avgAttempts: 0 }; - const isFailed = recentFailed.includes(job); - if (isFailed) bucket.failed++; - if (isFailed && job.attemptsMade > 1) bucket.retried++; - // Rolling average of attempts - bucket.avgAttempts = - (bucket.avgAttempts * (bucket.failed + bucket.retried - 1) + job.attemptsMade) / - Math.max(bucket.failed + bucket.retried, 1); - byJobType[job.name] = bucket; + // ── Public API: retry analytics + /** + * Analyse retry behaviour for jobs that finished within the given window. + * + * Examines the last N completed+failed jobs and produces per-job-type + * breakdowns so engineers can tune `maxAttempts` per queue type. + */ + async getRetryAnalytics(windowMinutes: number = 60): Promise { + const windowMs = windowMinutes * 60 * 1000; + const cutoff = Date.now() - windowMs; + const [failedJobs, completedJobs] = await Promise.all([ + this.defaultQueue.getFailed(0, 2000), + this.defaultQueue.getCompleted(0, 2000), + ]); + // Filter to window + const recentFailed = failedJobs.filter((j) => (j.finishedOn ?? j.timestamp) >= cutoff); + const recentCompleted = completedJobs.filter((j) => (j.finishedOn ?? j.timestamp) >= cutoff); + // Jobs that succeeded after ≥1 retry + const successAfterRetry = recentCompleted.filter((j) => j.attemptsMade > 1); + // Jobs that failed after all retries (attemptsMade >= maxAttempts) + const permanentlyFailed = recentFailed.filter((j) => j.attemptsMade >= (j.opts?.attempts ?? 3)); + const retried = recentFailed.filter((j) => j.attemptsMade > 1); + // Per-job-type breakdown + const byJobType: RetryAnalytics['byJobType'] = {}; + const allRecent = [...recentFailed, ...recentCompleted]; + for (const job of allRecent) { + const bucket = byJobType[job.name] ?? { failed: 0, retried: 0, avgAttempts: 0 }; + const isFailed = recentFailed.includes(job); + if (isFailed) + bucket.failed++; + if (isFailed && job.attemptsMade > 1) + bucket.retried++; + // Rolling average of attempts + bucket.avgAttempts = + (bucket.avgAttempts * (bucket.failed + bucket.retried - 1) + job.attemptsMade) / + Math.max(bucket.failed + bucket.retried, 1); + byJobType[job.name] = bucket; + } + const totalFailed = recentFailed.length; + return { + windowMinutes, + totalFailed, + totalRetried: retried.length, + successAfterRetry: successAfterRetry.length, + permanentlyFailed: permanentlyFailed.length, + retryRate: totalFailed > 0 ? retried.length / totalFailed : 0, + successAfterRetryRate: retried.length > 0 ? successAfterRetry.length / retried.length : 0, + byJobType, + }; } - - const totalFailed = recentFailed.length; - - return { - windowMinutes, - totalFailed, - totalRetried: retried.length, - successAfterRetry: successAfterRetry.length, - permanentlyFailed: permanentlyFailed.length, - retryRate: totalFailed > 0 ? retried.length / totalFailed : 0, - successAfterRetryRate: retried.length > 0 ? successAfterRetry.length / retried.length : 0, - byJobType, - }; - } - - // ── Scheduled tasks - - /** - * Periodic health check — every minute. - * Logs warnings/criticals and attempts to recover stuck jobs automatically. - */ - @Cron(CronExpression.EVERY_MINUTE) - async periodicHealthCheck(): Promise { - try { - const health = await this.checkQueueHealth(); - - if (health.status === 'critical') { - this.logger.error(`Queue health CRITICAL: ${health.issues.join(' | ')}`); - await this.sendAlert(health); - } else if (health.status === 'warning') { - this.logger.warn(`Queue health WARNING: ${health.issues.join(' | ')}`); - } else { - this.logger.debug('Queue health: OK'); - } - - const stuckJobs = await this.getStuckJobs(this.THRESHOLDS.stuckThresholdMs); - if (stuckJobs.length > 0) { - this.logger.warn(`Found ${stuckJobs.length} stuck job(s), moving to failed for retry`); - await this.recoverStuckJobs(stuckJobs); - } - } catch (err) { - this.logger.error('Error during periodic health check:', (err as Error).message); + // ── Scheduled tasks + /** + * Periodic health check — every minute. + * Logs warnings/criticals and attempts to recover stuck jobs automatically. + */ + @Cron(CronExpression.EVERY_MINUTE) + async periodicHealthCheck(): Promise { + try { + const health = await this.checkQueueHealth(); + if (health.status === 'critical') { + this.logger.error(`Queue health CRITICAL: ${health.issues.join(' | ')}`); + await this.sendAlert(health); + } + else if (health.status === 'warning') { + this.logger.warn(`Queue health WARNING: ${health.issues.join(' | ')}`); + } + else { + this.logger.debug('Queue health: OK'); + } + const stuckJobs = await this.getStuckJobs(this.THRESHOLDS.stuckThresholdMs); + if (stuckJobs.length > 0) { + this.logger.warn(`Found ${stuckJobs.length} stuck job(s), moving to failed for retry`); + await this.recoverStuckJobs(stuckJobs); + } + } + catch (err) { + this.logger.error('Error during periodic health check:', (err as Error).message); + } + } + // ── Private helpers + /** + * Calculate jobs completed per minute using the last two timestamped + * snapshots. Falls back to 0 if there is insufficient history. + */ + private calculateThroughput(currentCompleted: number, capturedAt: number): number { + if (this.metricsHistory.length === 0) + return 0; + const previous = this.metricsHistory[this.metricsHistory.length - 1]; + const deltaCompleted = Math.max(0, currentCompleted - previous.completed); + const deltaMs = capturedAt - previous.capturedAt; + if (deltaMs <= 0) + return 0; + // Jobs per minute + return Math.round((deltaCompleted / deltaMs) * 60000); + } + private async calculateAvgProcessingTime(): Promise { + try { + const completed = await this.defaultQueue.getCompleted(0, 100); + if (completed.length === 0) + return 0; + const times = completed + .filter((j) => j.finishedOn != null && j.processedOn != null) + .map((j) => (j.finishedOn ?? 0) - (j.processedOn ?? 0)); + if (times.length === 0) + return 0; + return Math.round(times.reduce((a, b) => a + b, 0) / times.length); + } + catch (err) { + this.logger.error('Error calculating avg processing time:', (err as Error).message); + return 0; + } + } + private appendToHistory(metrics: TimestampedMetrics): void { + this.metricsHistory.push(metrics); + if (this.metricsHistory.length > this.MAX_HISTORY_SIZE) { + this.metricsHistory.shift(); + } } - } - - // ── Private helpers - - /** - * Calculate jobs completed per minute using the last two timestamped - * snapshots. Falls back to 0 if there is insufficient history. - */ - private calculateThroughput(currentCompleted: number, capturedAt: number): number { - if (this.metricsHistory.length === 0) return 0; - - const previous = this.metricsHistory[this.metricsHistory.length - 1]; - const deltaCompleted = Math.max(0, currentCompleted - previous.completed); - const deltaMs = capturedAt - previous.capturedAt; - - if (deltaMs <= 0) return 0; - - // Jobs per minute - return Math.round((deltaCompleted / deltaMs) * 60_000); - } - - private async calculateAvgProcessingTime(): Promise { - try { - const completed = await this.defaultQueue.getCompleted(0, 100); - if (completed.length === 0) return 0; - - const times = completed - .filter((j) => j.finishedOn != null && j.processedOn != null) - .map((j) => (j.finishedOn ?? 0) - (j.processedOn ?? 0)); - - if (times.length === 0) return 0; - return Math.round(times.reduce((a, b) => a + b, 0) / times.length); - } catch (err) { - this.logger.error('Error calculating avg processing time:', (err as Error).message); - return 0; + private calculateTrend(values: number[]): 'up' | 'down' | 'stable' { + if (values.length < 2) + return 'stable'; + const recent = values.slice(-10); + const avg = recent.reduce((a, b) => a + b, 0) / recent.length; + const latest = recent[recent.length - 1]; + if (avg === 0) + return 'stable'; + const changePct = ((latest - avg) / avg) * 100; + if (changePct > 10) + return 'up'; + if (changePct < -10) + return 'down'; + return 'stable'; } - } - - private appendToHistory(metrics: TimestampedMetrics): void { - this.metricsHistory.push(metrics); - if (this.metricsHistory.length > this.MAX_HISTORY_SIZE) { - this.metricsHistory.shift(); + private async recoverStuckJobs(jobs: Job[]): Promise { + for (const job of jobs) { + try { + this.logger.log(`Moving stuck job ${job.id} (${job.name}) to failed for retry`); + await job.moveToFailed({ message: 'Automatically moved: job exceeded stuck threshold' }, true); + } + catch (err) { + this.logger.error(`Failed to recover job ${job.id}:`, (err as Error).message); + } + } } - } - - private calculateTrend(values: number[]): 'up' | 'down' | 'stable' { - if (values.length < 2) return 'stable'; - const recent = values.slice(-10); - const avg = recent.reduce((a, b) => a + b, 0) / recent.length; - const latest = recent[recent.length - 1]; - if (avg === 0) return 'stable'; - const changePct = ((latest - avg) / avg) * 100; - if (changePct > 10) return 'up'; - if (changePct < -10) return 'down'; - return 'stable'; - } - - private async recoverStuckJobs(jobs: Job[]): Promise { - for (const job of jobs) { - try { - this.logger.log(`Moving stuck job ${job.id} (${job.name}) to failed for retry`); - await job.moveToFailed( - { message: 'Automatically moved: job exceeded stuck threshold' }, - true, - ); - } catch (err) { - this.logger.error(`Failed to recover job ${job.id}:`, (err as Error).message); - } + private async sendAlert(health: QueueHealthStatus): Promise { + this.logger.error('QUEUE_ALERT', JSON.stringify({ + status: health.status, + issues: health.issues, + metrics: health.metrics, + timestamp: health.timestamp, + })); } - } - - private async sendAlert(health: QueueHealthStatus): Promise { - this.logger.error( - 'QUEUE_ALERT', - JSON.stringify({ - status: health.status, - issues: health.issues, - metrics: health.metrics, - timestamp: health.timestamp, - }), - ); - } - - /** Helper — no-op used for readable escalation logic above */ - private escalate( - _current: 'healthy' | 'warning' | 'critical', - _next: 'warning' | 'critical', - ): void {} + /** Helper — no-op used for readable escalation logic above */ + private escalate(_current: 'healthy' | 'warning' | 'critical', _next: 'warning' | 'critical'): void { } } diff --git a/src/queues/prioritization/prioritization.service.ts b/src/queues/prioritization/prioritization.service.ts index 70ce1c28..d67b2645 100644 --- a/src/queues/prioritization/prioritization.service.ts +++ b/src/queues/prioritization/prioritization.service.ts @@ -1,175 +1,163 @@ import { Injectable, Logger } from '@nestjs/common'; import { JobPriority } from '../enums/job-priority.enum'; import { JobOptions } from '../interfaces/queue.interfaces'; - /** * Job Prioritization Service * Determines job priority based on various factors */ @Injectable() export class PrioritizationService { - private readonly logger = new Logger(PrioritizationService.name); - - /** - * Calculate job priority based on multiple factors - */ - calculatePriority(factors: PriorityFactors): JobPriority { - let score = 0; - - // User tier weight (0-30 points) - if (factors.userTier) { - const tierScores = { - premium: 30, - pro: 20, - basic: 10, - free: 0, - }; - score += tierScores[factors.userTier] || 0; + private readonly logger = new Logger(PrioritizationService.name); + /** + * Calculate job priority based on multiple factors + */ + calculatePriority(factors: PriorityFactors): JobPriority { + let score = 0; + // User tier weight (0-30 points) + if (factors.userTier) { + const tierScores = { + premium: 30, + pro: 20, + basic: 10, + free: 0, + }; + score += tierScores[factors.userTier] || 0; + } + // Urgency weight (0-25 points) + if (factors.urgency) { + const urgencyScores = { + critical: 25, + high: 20, + medium: 10, + low: 0, + }; + score += urgencyScores[factors.urgency] || 0; + } + // Business impact weight (0-25 points) + if (factors.businessImpact) { + const impactScores = { + revenue: 25, + customer: 20, + operational: 15, + internal: 5, + }; + score += impactScores[factors.businessImpact] || 0; + } + // Time sensitivity (0-20 points) + if (factors.deadline) { + const now = Date.now(); + const deadline = new Date(factors.deadline).getTime(); + const timeLeft = deadline - now; + const hoursLeft = timeLeft / (1000 * 60 * 60); + if (hoursLeft < 1) + score += 20; + else if (hoursLeft < 6) + score += 15; + else if (hoursLeft < 24) + score += 10; + else if (hoursLeft < 72) + score += 5; + } + // Map score to priority level + return this.scoreToPriority(score); } - - // Urgency weight (0-25 points) - if (factors.urgency) { - const urgencyScores = { - critical: 25, - high: 20, - medium: 10, - low: 0, - }; - score += urgencyScores[factors.urgency] || 0; + /** + * Convert numeric score to JobPriority enum + */ + private scoreToPriority(score: number): JobPriority { + if (score >= 70) + return JobPriority.CRITICAL; + if (score >= 50) + return JobPriority.HIGH; + if (score >= 30) + return JobPriority.NORMAL; + if (score >= 15) + return JobPriority.LOW; + return JobPriority.BACKGROUND; } - - // Business impact weight (0-25 points) - if (factors.businessImpact) { - const impactScores = { - revenue: 25, - customer: 20, - operational: 15, - internal: 5, - }; - score += impactScores[factors.businessImpact] || 0; + /** + * Get recommended job options based on priority + */ + getJobOptions(priority: JobPriority): Partial { + const optionsMap: Record> = { + [JobPriority.CRITICAL]: { + priority: JobPriority.CRITICAL, + attempts: 5, + timeout: 60000, + backoff: { + type: 'exponential', + delay: 1000, + }, + removeOnComplete: false, + removeOnFail: false, + }, + [JobPriority.HIGH]: { + priority: JobPriority.HIGH, + attempts: 4, + timeout: 45000, + backoff: { + type: 'exponential', + delay: 2000, + }, + removeOnComplete: true, + removeOnFail: false, + }, + [JobPriority.NORMAL]: { + priority: JobPriority.NORMAL, + attempts: 3, + timeout: 30000, + backoff: { + type: 'exponential', + delay: 3000, + }, + removeOnComplete: true, + removeOnFail: false, + }, + [JobPriority.LOW]: { + priority: JobPriority.LOW, + attempts: 2, + timeout: 20000, + backoff: { + type: 'fixed', + delay: 5000, + }, + removeOnComplete: true, + removeOnFail: true, + }, + [JobPriority.BACKGROUND]: { + priority: JobPriority.BACKGROUND, + attempts: 1, + timeout: 15000, + backoff: { + type: 'fixed', + delay: 10000, + }, + removeOnComplete: true, + removeOnFail: true, + }, + }; + return optionsMap[priority]; } - - // Time sensitivity (0-20 points) - if (factors.deadline) { - const now = Date.now(); - const deadline = new Date(factors.deadline).getTime(); - const timeLeft = deadline - now; - const hoursLeft = timeLeft / (1000 * 60 * 60); - - if (hoursLeft < 1) score += 20; - else if (hoursLeft < 6) score += 15; - else if (hoursLeft < 24) score += 10; - else if (hoursLeft < 72) score += 5; + /** + * Adjust priority dynamically based on job age + */ + adjustPriorityByAge(currentPriority: JobPriority, createdAt: Date): JobPriority { + const ageInHours = (Date.now() - createdAt.getTime()) / (1000 * 60 * 60); + // Increase priority for jobs waiting too long + if (ageInHours > 24 && currentPriority > JobPriority.CRITICAL) { + this.logger.log(`Increasing priority for job older than 24 hours: ${currentPriority} -> ${currentPriority - 1}`); + return (currentPriority - 1) as JobPriority; + } + if (ageInHours > 12 && currentPriority > JobPriority.HIGH) { + this.logger.log(`Increasing priority for job older than 12 hours: ${currentPriority} -> ${currentPriority - 1}`); + return (currentPriority - 1) as JobPriority; + } + return currentPriority; } - - // Map score to priority level - return this.scoreToPriority(score); - } - - /** - * Convert numeric score to JobPriority enum - */ - private scoreToPriority(score: number): JobPriority { - if (score >= 70) return JobPriority.CRITICAL; - if (score >= 50) return JobPriority.HIGH; - if (score >= 30) return JobPriority.NORMAL; - if (score >= 15) return JobPriority.LOW; - return JobPriority.BACKGROUND; - } - - /** - * Get recommended job options based on priority - */ - getJobOptions(priority: JobPriority): Partial { - const optionsMap: Record> = { - [JobPriority.CRITICAL]: { - priority: JobPriority.CRITICAL, - attempts: 5, - timeout: 60000, - backoff: { - type: 'exponential', - delay: 1000, - }, - removeOnComplete: false, - removeOnFail: false, - }, - [JobPriority.HIGH]: { - priority: JobPriority.HIGH, - attempts: 4, - timeout: 45000, - backoff: { - type: 'exponential', - delay: 2000, - }, - removeOnComplete: true, - removeOnFail: false, - }, - [JobPriority.NORMAL]: { - priority: JobPriority.NORMAL, - attempts: 3, - timeout: 30000, - backoff: { - type: 'exponential', - delay: 3000, - }, - removeOnComplete: true, - removeOnFail: false, - }, - [JobPriority.LOW]: { - priority: JobPriority.LOW, - attempts: 2, - timeout: 20000, - backoff: { - type: 'fixed', - delay: 5000, - }, - removeOnComplete: true, - removeOnFail: true, - }, - [JobPriority.BACKGROUND]: { - priority: JobPriority.BACKGROUND, - attempts: 1, - timeout: 15000, - backoff: { - type: 'fixed', - delay: 10000, - }, - removeOnComplete: true, - removeOnFail: true, - }, - }; - - return optionsMap[priority]; - } - - /** - * Adjust priority dynamically based on job age - */ - adjustPriorityByAge(currentPriority: JobPriority, createdAt: Date): JobPriority { - const ageInHours = (Date.now() - createdAt.getTime()) / (1000 * 60 * 60); - - // Increase priority for jobs waiting too long - if (ageInHours > 24 && currentPriority > JobPriority.CRITICAL) { - this.logger.log( - `Increasing priority for job older than 24 hours: ${currentPriority} -> ${currentPriority - 1}`, - ); - return (currentPriority - 1) as JobPriority; - } - - if (ageInHours > 12 && currentPriority > JobPriority.HIGH) { - this.logger.log( - `Increasing priority for job older than 12 hours: ${currentPriority} -> ${currentPriority - 1}`, - ); - return (currentPriority - 1) as JobPriority; - } - - return currentPriority; - } } - export interface PriorityFactors { - userTier?: 'premium' | 'pro' | 'basic' | 'free'; - urgency?: 'critical' | 'high' | 'medium' | 'low'; - businessImpact?: 'revenue' | 'customer' | 'operational' | 'internal'; - deadline?: Date | string; + userTier?: 'premium' | 'pro' | 'basic' | 'free'; + urgency?: 'critical' | 'high' | 'medium' | 'low'; + businessImpact?: 'revenue' | 'customer' | 'operational' | 'internal'; + deadline?: Date | string; } diff --git a/src/queues/processors/default-queue.processor.ts b/src/queues/processors/default-queue.processor.ts index 328db8bc..192d1a0e 100644 --- a/src/queues/processors/default-queue.processor.ts +++ b/src/queues/processors/default-queue.processor.ts @@ -4,138 +4,99 @@ import { Logger } from '@nestjs/common'; import { Job } from 'bull'; import { RetryLogicService } from '../retry/retry-logic.service'; import { sanitizeEmail, sanitizePii } from '../../common/utils/pii-sanitizer.utils'; - /** * Default Queue Processor * Processes jobs from the default queue */ @Processor(QUEUE_NAMES.DEFAULT) export class DefaultQueueProcessor { - private readonly logger = new Logger(DefaultQueueProcessor.name); - - constructor(private readonly retryLogicService: RetryLogicService) {} - - @Process('*') - async handleJob(job: Job): Promise { - this.logger.log(`Processing job ${job.name} (ID: ${job.id}) - Attempt ${job.attemptsMade + 1}`); - - try { - // Update progress - await job.progress(10); - - // Simulate job processing based on job name - const result = await this.processJobByType(job); - - await job.progress(100); - return result; - } catch (error) { - this.logger.error(`Error processing job ${job.id}:`, error.message); - throw error; + private readonly logger = new Logger(DefaultQueueProcessor.name); + constructor(private readonly retryLogicService: RetryLogicService) { } + @Process('*') + async handleJob(job: Job): Promise { + this.logger.log(`Processing job ${job.name} (ID: ${job.id}) - Attempt ${job.attemptsMade + 1}`); + try { + // Update progress + await job.progress(10); + // Simulate job processing based on job name + const result = await this.processJobByType(job); + await job.progress(100); + return result; + } + catch (error) { + this.logger.error(`Error processing job ${job.id}:`, error.message); + throw error; + } } - } - - /** - * Process job based on its type/name - */ - private async processJobByType(job: Job): Promise { - switch (job.name) { - case 'send-email': - return this.processSendEmail(job); - case 'generate-report': - return this.processGenerateReport(job); - case 'process-payment': - return this.processPayment(job); - case 'backup-data': - return this.processBackup(job); - default: - this.logger.warn(`Unknown job type: ${job.name}`); - return { status: 'completed', message: 'Job processed' }; + /** + * Process job based on its type/name + */ + private async processJobByType(job: Job): Promise { + switch (job.name) { + case 'send-email': + return this.processSendEmail(job); + case 'generate-report': + return this.processGenerateReport(job); + case 'process-payment': + return this.processPayment(job); + case 'backup-data': + return this.processBackup(job); + default: + this.logger.warn(`Unknown job type: ${job.name}`); + return { status: 'completed', message: 'Job processed' }; + } } - } - - private async processSendEmail(job: Job): Promise { - await job.progress(30); - // Email sending logic here - this.logger.log(`Sending email to ${sanitizeEmail(job.data.to)}`); - await this.simulateWork(2000); - await job.progress(80); - return { status: 'sent', recipient: job.data.to }; - } - - private async processGenerateReport(job: Job): Promise { - await job.progress(20); - this.logger.log(`Generating report: ${job.data.reportType}`); - await this.simulateWork(5000); - await job.progress(90); - return { status: 'generated', reportId: `report-${Date.now()}` }; - } - - private async processPayment(job: Job): Promise { - await job.progress(40); - this.logger.log(`Processing payment: ${job.data.amount}`); - await this.simulateWork(3000); - await job.progress(95); - return { status: 'processed', transactionId: `txn-${Date.now()}` }; - } - - private async processBackup(job: Job): Promise { - await job.progress(25); - this.logger.log(`Backing up data: ${job.data.dataType}`); - await this.simulateWork(10000); - await job.progress(90); - return { status: 'backed-up', backupId: `backup-${Date.now()}` }; - } - - private simulateWork(ms: number): Promise { - return new Promise((resolve) => setTimeout(resolve, ms)); - } - - @OnQueueActive() - onActive(job: Job) { - this.logger.log( - `Job ${job.name} (${job.id}) started processing - Priority: ${job.opts.priority}`, - ); - } - - @OnQueueCompleted() - onCompleted(job: Job, result: any) { - const processingTime = (job.finishedOn ?? Date.now()) - (job.processedOn ?? Date.now()); - this.logger.log( - `Job ${job.name} (${job.id}) completed in ${processingTime}ms - Result: ${JSON.stringify( - sanitizePii(result), - )}`, - ); - } - - @OnQueueFailed() - async onFailed(job: Job, error: Error) { - const strategy = this.retryLogicService.getDefaultStrategy(job.name); - const shouldRetry = this.retryLogicService.shouldRetry( - error, - job.attemptsMade, - strategy.maxAttempts, - ); - - if (shouldRetry) { - const nextDelay = this.retryLogicService.calculateBackoffDelay( - job.attemptsMade + 1, - strategy, - ); - - this.retryLogicService.logRetryAttempt( - job.id.toString(), - job.name, - job.attemptsMade, - error, - nextDelay, - ); - } else { - this.retryLogicService.handleFinalFailure( - job.id.toString(), - job.name, - error, - job.attemptsMade, - ); + private async processSendEmail(job: Job): Promise { + await job.progress(30); + // Email sending logic here + this.logger.log(`Sending email to ${sanitizeEmail(job.data.to)}`); + await this.simulateWork(2000); + await job.progress(80); + return { status: 'sent', recipient: job.data.to }; + } + private async processGenerateReport(job: Job): Promise { + await job.progress(20); + this.logger.log(`Generating report: ${job.data.reportType}`); + await this.simulateWork(5000); + await job.progress(90); + return { status: 'generated', reportId: `report-${Date.now()}` }; + } + private async processPayment(job: Job): Promise { + await job.progress(40); + this.logger.log(`Processing payment: ${job.data.amount}`); + await this.simulateWork(3000); + await job.progress(95); + return { status: 'processed', transactionId: `txn-${Date.now()}` }; + } + private async processBackup(job: Job): Promise { + await job.progress(25); + this.logger.log(`Backing up data: ${job.data.dataType}`); + await this.simulateWork(10000); + await job.progress(90); + return { status: 'backed-up', backupId: `backup-${Date.now()}` }; + } + private simulateWork(ms: number): Promise { + return new Promise((resolve) => setTimeout(resolve, ms)); + } + @OnQueueActive() + onActive(job: Job) { + this.logger.log(`Job ${job.name} (${job.id}) started processing - Priority: ${job.opts.priority}`); + } + @OnQueueCompleted() + onCompleted(job: Job, result: unknown) { + const processingTime = (job.finishedOn ?? Date.now()) - (job.processedOn ?? Date.now()); + this.logger.log(`Job ${job.name} (${job.id}) completed in ${processingTime}ms - Result: ${JSON.stringify(sanitizePii(result))}`); + } + @OnQueueFailed() + async onFailed(job: Job, error: Error) { + const strategy = this.retryLogicService.getDefaultStrategy(job.name); + const shouldRetry = this.retryLogicService.shouldRetry(error, job.attemptsMade, strategy.maxAttempts); + if (shouldRetry) { + const nextDelay = this.retryLogicService.calculateBackoffDelay(job.attemptsMade + 1, strategy); + this.retryLogicService.logRetryAttempt(job.id.toString(), job.name, job.attemptsMade, error, nextDelay); + } + else { + this.retryLogicService.handleFinalFailure(job.id.toString(), job.name, error, job.attemptsMade); + } } - } } diff --git a/src/queues/queue.controller.ts b/src/queues/queue.controller.ts index 99ef4477..4fc89666 100644 --- a/src/queues/queue.controller.ts +++ b/src/queues/queue.controller.ts @@ -1,370 +1,317 @@ -import { - Body, - Controller, - Delete, - Get, - HttpCode, - HttpStatus, - NotFoundException, - Param, - Post, - Query, - UseGuards, - ValidationPipe, -} from '@nestjs/common'; +import { Body, Controller, Delete, Get, HttpCode, HttpStatus, NotFoundException, Param, Post, Query, UseGuards, ValidationPipe, } from '@nestjs/common'; import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiParam } from '@nestjs/swagger'; import { Throttle } from '@nestjs/throttler'; - import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RolesGuard } from '../common/guards/roles.guard'; import { Roles } from '../common/decorators/roles.decorator'; - import { QueueService } from './queue.service'; import { PrioritizationService } from './prioritization/prioritization.service'; import { JobSchedulerService } from './scheduler/job-scheduler.service'; import { QueueMonitoringService } from './monitoring/queue-monitoring.service'; import { JobOptions } from './interfaces/queue.interfaces'; - -import { - AddJobDto, - AddBulkJobsDto, - ScheduleJobDto, - ScheduleDelayedJobDto, - CleanQueueDto, - FailedJobsQueryDto, - StuckJobsQueryDto, - AnalyticsQueryDto, -} from './dto/queue.dto'; - +import { AddJobDto, AddBulkJobsDto, ScheduleJobDto, ScheduleDelayedJobDto, CleanQueueDto, FailedJobsQueryDto, StuckJobsQueryDto, AnalyticsQueryDto, } from './dto/queue.dto'; @ApiTags('queues') @ApiBearerAuth() @UseGuards(JwtAuthGuard, RolesGuard) -@Throttle({ default: { limit: 60, ttl: 60_000 } }) +@Throttle({ default: { limit: 60, ttl: 60000 } }) @Controller('queues') export class QueueController { - constructor( - private readonly queueService: QueueService, - private readonly prioritizationService: PrioritizationService, - private readonly schedulerService: JobSchedulerService, - private readonly monitoringService: QueueMonitoringService, - ) {} - - // ── Monitoring (read-only) - /** - * Live queue metrics: counts per state, throughput, avg processing time. - */ - @Get('metrics') - @ApiOperation({ summary: 'Live queue metrics' }) - @ApiResponse({ status: 200, description: 'Current queue metrics snapshot' }) - async getMetrics() { - return this.monitoringService.getQueueMetrics(); - } - - /** - * Aggregated statistics with historical trends. - */ - @Get('statistics') - @ApiOperation({ summary: 'Queue statistics with trends' }) - async getStatistics() { - return this.monitoringService.getQueueStatistics(); - } - - /** - * Health status: healthy | warning | critical + issue list. - */ - @Get('health') - @ApiOperation({ summary: 'Queue health status' }) - async getHealth() { - return this.monitoringService.checkQueueHealth(); - } - - /** - * Raw job-state counts (lightweight — no processing time calculation). - */ - @Get('counts') - @ApiOperation({ summary: 'Job counts by state' }) - async getCounts() { - return this.queueService.getQueueCounts(); - } - - /** - * In-memory metrics history (sliding window). - */ - @Get('metrics/history') - @ApiOperation({ summary: 'Historical metrics snapshots' }) - async getMetricsHistory() { - return { history: this.monitoringService.getMetricsHistory() }; - } - - /** - * Retry-rate analytics: failure rates, retry counts, success-after-retry %. - * Supports an optional windowMinutes parameter (default 60, max 1440). - */ - @Get('metrics/retries') - @ApiOperation({ summary: 'Retry analytics for the given time window' }) - async getRetryAnalytics( - @Query(new ValidationPipe({ transform: true })) query: AnalyticsQueryDto, - ) { - return this.monitoringService.getRetryAnalytics(query.windowMinutes ?? 60); - } - - // ── Static job-collection routes (MUST precede /:id) - - /** - * Failed jobs — paginated, optionally filtered by job name. - * GET /queues/jobs/failed?limit=50&offset=0&jobName=send-email - */ - @Get('jobs/failed') - @ApiOperation({ summary: 'List failed jobs (paginated)' }) - async getFailedJobs(@Query(new ValidationPipe({ transform: true })) query: FailedJobsQueryDto) { - const limit = query.limit ?? 50; - const offset = query.offset ?? 0; - - const jobs = await this.monitoringService.getFailedJobs(limit, offset, query.jobName); - - return { - data: jobs.map((job) => ({ - id: job.id, - name: job.name, - data: job.data, - failedReason: job.failedReason, - attemptsMade: job.attemptsMade, - maxAttempts: job.opts?.attempts ?? 3, - stackTrace: job.stacktrace ?? [], - timestamp: new Date(job.timestamp).toISOString(), - processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, - finishedOn: job.finishedOn ? new Date(job.finishedOn).toISOString() : null, - })), - meta: { limit, offset, total: jobs.length }, - }; - } - - /** - * Retry all currently-failed jobs (admin only). - * Returns a summary of how many were requeued and how many failed to retry. - */ - @Post('jobs/failed/retry-all') - @Roles('admin') - @ApiOperation({ summary: 'Retry all failed jobs (admin)' }) - @ApiResponse({ status: 200, description: 'Retry summary' }) - async retryAllFailedJobs() { - return this.monitoringService.retryAllFailedJobs(); - } - - /** - * Stuck jobs — active jobs that have exceeded the processing threshold. - * GET /queues/jobs/stuck?threshold=300000 - */ - @Get('jobs/stuck') - @ApiOperation({ summary: 'List stuck (over-threshold active) jobs' }) - async getStuckJobs(@Query(new ValidationPipe({ transform: true })) query: StuckJobsQueryDto) { - const threshold = query.threshold ?? 300_000; - const jobs = await this.monitoringService.getStuckJobs(threshold); - - return { - data: jobs.map((job) => ({ - id: job.id, - name: job.name, - data: job.data, - processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, - activeForMs: job.processedOn ? Date.now() - job.processedOn : null, - attemptsMade: job.attemptsMade, - priority: job.opts?.priority, - })), - thresholdMs: threshold, - }; - } - - /** - * All jobs currently in the delayed state (scheduled for future execution). - */ - @Get('jobs/scheduled') - @ApiOperation({ summary: 'List delayed (scheduled) jobs' }) - async getScheduledJobs() { - return this.schedulerService.getScheduledJobs(); - } - - // ── Per-job routes (/jobs/:id — after static routes) - /** - * Full details + metrics for a single job. - */ - @Get('jobs/:id') - @ApiOperation({ summary: 'Get job details' }) - @ApiParam({ name: 'id', description: 'Bull job ID' }) - async getJob(@Param('id') id: string) { - const metrics = await this.queueService.getJobMetrics(id); - if (!metrics) { - throw new NotFoundException(`Job ${id} not found`); + constructor(private readonly queueService: QueueService, private readonly prioritizationService: PrioritizationService, private readonly schedulerService: JobSchedulerService, private readonly monitoringService: QueueMonitoringService) { } + // ── Monitoring (read-only) + /** + * Live queue metrics: counts per state, throughput, avg processing time. + */ + @Get('metrics') + @ApiOperation({ summary: 'Live queue metrics' }) + @ApiResponse({ status: 200, description: 'Current queue metrics snapshot' }) + async getMetrics() { + return this.monitoringService.getQueueMetrics(); } - return metrics; - } - - /** - * Retry a specific failed job. - */ - @Post('jobs/:id/retry') - @Roles('admin') - @ApiOperation({ summary: 'Retry a failed job (admin)' }) - async retryJob(@Param('id') id: string) { - const job = await this.queueService.getJob(id); - if (!job) throw new NotFoundException(`Job ${id} not found`); - - await this.queueService.retryJob(id); - return { message: 'Job retry initiated', jobId: id }; - } - - /** - * Remove a specific job (any state). - */ - @Delete('jobs/:id') - @Roles('admin') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Delete a job (admin)' }) - async removeJob(@Param('id') id: string) { - const job = await this.queueService.getJob(id); - if (!job) throw new NotFoundException(`Job ${id} not found`); - await this.queueService.removeJob(id); - } - - // ── Job submission - - /** - * Enqueue a single job, optionally with priority factor calculation. - */ - @Post('jobs') - @Roles('admin') - @ApiOperation({ summary: 'Add a job to the queue (admin)' }) - async addJob(@Body(ValidationPipe) body: AddJobDto) { - let options: JobOptions = body.options ?? {}; - - if (body.priorityFactors) { - const priority = this.prioritizationService.calculatePriority(body.priorityFactors as any); - options = { ...options, ...this.prioritizationService.getJobOptions(priority) }; + /** + * Aggregated statistics with historical trends. + */ + @Get('statistics') + @ApiOperation({ summary: 'Queue statistics with trends' }) + async getStatistics() { + return this.monitoringService.getQueueStatistics(); } - - const job = await this.queueService.addJob(body.name, body.data, options); - - return { - jobId: job.id, - name: job.name, - priority: job.opts.priority, - status: 'queued', - }; - } - - /** - * Enqueue multiple jobs atomically. - */ - @Post('jobs/bulk') - @Roles('admin') - @ApiOperation({ summary: 'Bulk-enqueue jobs (admin)' }) - async addBulkJobs(@Body(ValidationPipe) body: AddBulkJobsDto) { - const jobs = await this.queueService.addBulkJobs(body.jobs); - return { - count: jobs.length, - jobIds: jobs.map((j) => j.id), - }; - } - - /** - * Schedule a job at an absolute future timestamp. - */ - @Post('jobs/schedule') - @Roles('admin') - @ApiOperation({ summary: 'Schedule a job at a specific time (admin)' }) - async scheduleJob(@Body(ValidationPipe) body: ScheduleJobDto) { - const scheduledTime = new Date(body.scheduledTime); - if (scheduledTime <= new Date()) { - throw new NotFoundException('scheduledTime must be in the future'); + /** + * Health status: healthy | warning | critical + issue list. + */ + @Get('health') + @ApiOperation({ summary: 'Queue health status' }) + async getHealth() { + return this.monitoringService.checkQueueHealth(); + } + /** + * Raw job-state counts (lightweight — no processing time calculation). + */ + @Get('counts') + @ApiOperation({ summary: 'Job counts by state' }) + async getCounts() { + return this.queueService.getQueueCounts(); + } + /** + * In-memory metrics history (sliding window). + */ + @Get('metrics/history') + @ApiOperation({ summary: 'Historical metrics snapshots' }) + async getMetricsHistory() { + return { history: this.monitoringService.getMetricsHistory() }; + } + /** + * Retry-rate analytics: failure rates, retry counts, success-after-retry %. + * Supports an optional windowMinutes parameter (default 60, max 1440). + */ + @Get('metrics/retries') + @ApiOperation({ summary: 'Retry analytics for the given time window' }) + async getRetryAnalytics( + @Query(new ValidationPipe({ transform: true })) + query: AnalyticsQueryDto) { + return this.monitoringService.getRetryAnalytics(query.windowMinutes ?? 60); + } + // ── Static job-collection routes (MUST precede /:id) + /** + * Failed jobs — paginated, optionally filtered by job name. + * GET /queues/jobs/failed?limit=50&offset=0&jobName=send-email + */ + @Get('jobs/failed') + @ApiOperation({ summary: 'List failed jobs (paginated)' }) + async getFailedJobs( + @Query(new ValidationPipe({ transform: true })) + query: FailedJobsQueryDto) { + const limit = query.limit ?? 50; + const offset = query.offset ?? 0; + const jobs = await this.monitoringService.getFailedJobs(limit, offset, query.jobName); + return { + data: jobs.map((job) => ({ + id: job.id, + name: job.name, + data: job.data, + failedReason: job.failedReason, + attemptsMade: job.attemptsMade, + maxAttempts: job.opts?.attempts ?? 3, + stackTrace: job.stacktrace ?? [], + timestamp: new Date(job.timestamp).toISOString(), + processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, + finishedOn: job.finishedOn ? new Date(job.finishedOn).toISOString() : null, + })), + meta: { limit, offset, total: jobs.length }, + }; + } + /** + * Retry all currently-failed jobs (admin only). + * Returns a summary of how many were requeued and how many failed to retry. + */ + @Post('jobs/failed/retry-all') + @Roles('admin') + @ApiOperation({ summary: 'Retry all failed jobs (admin)' }) + @ApiResponse({ status: 200, description: 'Retry summary' }) + async retryAllFailedJobs() { + return this.monitoringService.retryAllFailedJobs(); + } + /** + * Stuck jobs — active jobs that have exceeded the processing threshold. + * GET /queues/jobs/stuck?threshold=300000 + */ + @Get('jobs/stuck') + @ApiOperation({ summary: 'List stuck (over-threshold active) jobs' }) + async getStuckJobs( + @Query(new ValidationPipe({ transform: true })) + query: StuckJobsQueryDto) { + const threshold = query.threshold ?? 300000; + const jobs = await this.monitoringService.getStuckJobs(threshold); + return { + data: jobs.map((job) => ({ + id: job.id, + name: job.name, + data: job.data, + processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, + activeForMs: job.processedOn ? Date.now() - job.processedOn : null, + attemptsMade: job.attemptsMade, + priority: job.opts?.priority, + })), + thresholdMs: threshold, + }; + } + /** + * All jobs currently in the delayed state (scheduled for future execution). + */ + @Get('jobs/scheduled') + @ApiOperation({ summary: 'List delayed (scheduled) jobs' }) + async getScheduledJobs() { + return this.schedulerService.getScheduledJobs(); + } + // ── Per-job routes (/jobs/:id — after static routes) + /** + * Full details + metrics for a single job. + */ + @Get('jobs/:id') + @ApiOperation({ summary: 'Get job details' }) + @ApiParam({ name: 'id', description: 'Bull job ID' }) + async getJob( + @Param('id') + id: string) { + const metrics = await this.queueService.getJobMetrics(id); + if (!metrics) { + throw new NotFoundException(`Job ${id} not found`); + } + return metrics; + } + /** + * Retry a specific failed job. + */ + @Post('jobs/:id/retry') + @Roles('admin') + @ApiOperation({ summary: 'Retry a failed job (admin)' }) + async retryJob( + @Param('id') + id: string) { + const job = await this.queueService.getJob(id); + if (!job) + throw new NotFoundException(`Job ${id} not found`); + await this.queueService.retryJob(id); + return { message: 'Job retry initiated', jobId: id }; + } + /** + * Remove a specific job (any state). + */ + @Delete('jobs/:id') + @Roles('admin') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Delete a job (admin)' }) + async removeJob( + @Param('id') + id: string) { + const job = await this.queueService.getJob(id); + if (!job) + throw new NotFoundException(`Job ${id} not found`); + await this.queueService.removeJob(id); + } + // ── Job submission + /** + * Enqueue a single job, optionally with priority factor calculation. + */ + @Post('jobs') + @Roles('admin') + @ApiOperation({ summary: 'Add a job to the queue (admin)' }) + async addJob( + @Body(ValidationPipe) + body: AddJobDto) { + let options: JobOptions = body.options ?? {}; + if (body.priorityFactors) { + const priority = this.prioritizationService.calculatePriority(body.priorityFactors as unknown); + options = { ...options, ...this.prioritizationService.getJobOptions(priority) }; + } + const job = await this.queueService.addJob(body.name, body.data, options); + return { + jobId: job.id, + name: job.name, + priority: job.opts.priority, + status: 'queued', + }; + } + /** + * Enqueue multiple jobs atomically. + */ + @Post('jobs/bulk') + @Roles('admin') + @ApiOperation({ summary: 'Bulk-enqueue jobs (admin)' }) + async addBulkJobs( + @Body(ValidationPipe) + body: AddBulkJobsDto) { + const jobs = await this.queueService.addBulkJobs(body.jobs); + return { + count: jobs.length, + jobIds: jobs.map((j) => j.id), + }; + } + /** + * Schedule a job at an absolute future timestamp. + */ + @Post('jobs/schedule') + @Roles('admin') + @ApiOperation({ summary: 'Schedule a job at a specific time (admin)' }) + async scheduleJob( + @Body(ValidationPipe) + body: ScheduleJobDto) { + const scheduledTime = new Date(body.scheduledTime); + if (scheduledTime <= new Date()) { + throw new NotFoundException('scheduledTime must be in the future'); + } + const jobId = await this.schedulerService.scheduleJob(body.name, body.data, scheduledTime, body.options); + return { jobId, scheduledFor: body.scheduledTime, status: 'scheduled' }; + } + /** + * Schedule a job with a relative delay in milliseconds. + */ + @Post('jobs/delay') + @Roles('admin') + @ApiOperation({ summary: 'Schedule a job with a delay (admin)' }) + async scheduleDelayedJob( + @Body(ValidationPipe) + body: ScheduleDelayedJobDto) { + const jobId = await this.schedulerService.scheduleDelayedJob(body.name, body.data, body.delayMs, body.options); + return { jobId, delayMs: body.delayMs, status: 'scheduled' }; + } + /** + * Cancel a specific scheduled (delayed) job. + */ + @Delete('jobs/scheduled/:id') + @Roles('admin') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Cancel a scheduled job (admin)' }) + async cancelScheduledJob( + @Param('id') + id: string) { + await this.schedulerService.cancelScheduledJob(id); + } + // ── Queue-level controls (admin only) + @Post('pause') + @Roles('admin') + @ApiOperation({ summary: 'Pause the queue (admin)' }) + async pauseQueue() { + await this.queueService.pauseQueue(); + return { message: 'Queue paused', timestamp: new Date().toISOString() }; + } + @Post('resume') + @Roles('admin') + @ApiOperation({ summary: 'Resume a paused queue (admin)' }) + async resumeQueue() { + await this.queueService.resumeQueue(); + return { message: 'Queue resumed', timestamp: new Date().toISOString() }; + } + /** + * Remove completed/failed jobs older than the grace period. + */ + @Post('clean') + @Roles('admin') + @ApiOperation({ summary: 'Clean old jobs (admin)' }) + async cleanQueue( + @Body(ValidationPipe) + body: CleanQueueDto) { + await this.queueService.cleanQueue(body.grace, body.status); + return { + message: 'Queue cleaned', + grace: body.grace ?? 5000, + status: body.status ?? 'all', + }; + } + /** + * Drain the entire queue (removes ALL jobs in all states). + * Use with extreme care — irreversible. + */ + @Delete('empty') + @Roles('admin') + @HttpCode(HttpStatus.NO_CONTENT) + @ApiOperation({ summary: 'Empty the queue — removes ALL jobs (admin)' }) + async emptyQueue() { + await this.queueService.emptyQueue(); + } + // ── Cron / scheduler info + @Get('cron/jobs') + @ApiOperation({ summary: 'List active cron job names' }) + async getActiveCronJobs() { + return { jobs: this.schedulerService.getActiveCronJobs() }; } - - const jobId = await this.schedulerService.scheduleJob( - body.name, - body.data, - scheduledTime, - body.options, - ); - - return { jobId, scheduledFor: body.scheduledTime, status: 'scheduled' }; - } - - /** - * Schedule a job with a relative delay in milliseconds. - */ - @Post('jobs/delay') - @Roles('admin') - @ApiOperation({ summary: 'Schedule a job with a delay (admin)' }) - async scheduleDelayedJob(@Body(ValidationPipe) body: ScheduleDelayedJobDto) { - const jobId = await this.schedulerService.scheduleDelayedJob( - body.name, - body.data, - body.delayMs, - body.options, - ); - - return { jobId, delayMs: body.delayMs, status: 'scheduled' }; - } - - /** - * Cancel a specific scheduled (delayed) job. - */ - @Delete('jobs/scheduled/:id') - @Roles('admin') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Cancel a scheduled job (admin)' }) - async cancelScheduledJob(@Param('id') id: string) { - await this.schedulerService.cancelScheduledJob(id); - } - - // ── Queue-level controls (admin only) - @Post('pause') - @Roles('admin') - @ApiOperation({ summary: 'Pause the queue (admin)' }) - async pauseQueue() { - await this.queueService.pauseQueue(); - return { message: 'Queue paused', timestamp: new Date().toISOString() }; - } - - @Post('resume') - @Roles('admin') - @ApiOperation({ summary: 'Resume a paused queue (admin)' }) - async resumeQueue() { - await this.queueService.resumeQueue(); - return { message: 'Queue resumed', timestamp: new Date().toISOString() }; - } - - /** - * Remove completed/failed jobs older than the grace period. - */ - @Post('clean') - @Roles('admin') - @ApiOperation({ summary: 'Clean old jobs (admin)' }) - async cleanQueue(@Body(ValidationPipe) body: CleanQueueDto) { - await this.queueService.cleanQueue(body.grace, body.status); - return { - message: 'Queue cleaned', - grace: body.grace ?? 5000, - status: body.status ?? 'all', - }; - } - - /** - * Drain the entire queue (removes ALL jobs in all states). - * Use with extreme care — irreversible. - */ - @Delete('empty') - @Roles('admin') - @HttpCode(HttpStatus.NO_CONTENT) - @ApiOperation({ summary: 'Empty the queue — removes ALL jobs (admin)' }) - async emptyQueue() { - await this.queueService.emptyQueue(); - } - - // ── Cron / scheduler info - - @Get('cron/jobs') - @ApiOperation({ summary: 'List active cron job names' }) - async getActiveCronJobs() { - return { jobs: this.schedulerService.getActiveCronJobs() }; - } } diff --git a/src/queues/queue.module.ts b/src/queues/queue.module.ts index ae4cf0cf..5c7154a9 100644 --- a/src/queues/queue.module.ts +++ b/src/queues/queue.module.ts @@ -9,42 +9,42 @@ import { RetryLogicService } from './retry/retry-logic.service'; import { QueueMonitoringService } from './monitoring/queue-monitoring.service'; import { JobSchedulerService } from './scheduler/job-scheduler.service'; import { DefaultQueueProcessor } from './processors/default-queue.processor'; - /** * Queue Module * Comprehensive queue management with prioritization, retry logic, and monitoring */ @Module({ - imports: [ - BullModule.registerQueue({ - name: QUEUE_NAMES.DEFAULT, - defaultJobOptions: { - attempts: 3, - backoff: { - type: 'exponential', - delay: 2000, - }, - removeOnComplete: true, - removeOnFail: false, - }, - }), - ScheduleModule.forRoot(), - ], - controllers: [QueueController], - providers: [ - QueueService, - PrioritizationService, - RetryLogicService, - QueueMonitoringService, - JobSchedulerService, - DefaultQueueProcessor, - ], - exports: [ - QueueService, - PrioritizationService, - RetryLogicService, - QueueMonitoringService, - JobSchedulerService, - ], + imports: [ + BullModule.registerQueue({ + name: QUEUE_NAMES.DEFAULT, + defaultJobOptions: { + attempts: 3, + backoff: { + type: 'exponential', + delay: 2000, + }, + removeOnComplete: true, + removeOnFail: false, + }, + }), + ScheduleModule.forRoot(), + ], + controllers: [QueueController], + providers: [ + QueueService, + PrioritizationService, + RetryLogicService, + QueueMonitoringService, + JobSchedulerService, + DefaultQueueProcessor, + ], + exports: [ + QueueService, + PrioritizationService, + RetryLogicService, + QueueMonitoringService, + JobSchedulerService, + ], }) -export class QueueModule {} +export class QueueModule { +} diff --git a/src/queues/queue.service.ts b/src/queues/queue.service.ts index 4b16f0e1..f144a3b9 100644 --- a/src/queues/queue.service.ts +++ b/src/queues/queue.service.ts @@ -4,178 +4,166 @@ import { Queue, Job } from 'bull'; import { QUEUE_NAMES } from '../common/constants/queue.constants'; import { JobOptions, JobMetrics } from './interfaces/queue.interfaces'; import { JobPriority, JobStatus } from './enums/job-priority.enum'; - /** * Core Queue Service * Provides centralized queue management with priority support */ @Injectable() export class QueueService { - private readonly logger = new Logger(QueueService.name); - - constructor(@InjectQueue(QUEUE_NAMES.DEFAULT) private readonly defaultQueue: Queue) {} - - /** - * Add a job to the queue with priority and options - */ - async addJob(name: string, data: T, options?: JobOptions): Promise> { - try { - const job = await this.defaultQueue.add(name, data, { - priority: options?.priority || JobPriority.NORMAL, - attempts: options?.attempts || 3, - backoff: options?.backoff || { - type: 'exponential', - delay: 2000, - }, - delay: options?.delay, - timeout: options?.timeout || 30000, - removeOnComplete: options?.removeOnComplete ?? true, - removeOnFail: options?.removeOnFail ?? false, - }); - - this.logger.log( - `Job ${name} added to queue with ID: ${job.id}, Priority: ${options?.priority || JobPriority.NORMAL}`, - ); - - return job; - } catch (error) { - this.logger.error(`Failed to add job ${name}:`, error); - throw error; + private readonly logger = new Logger(QueueService.name); + constructor( + @InjectQueue(QUEUE_NAMES.DEFAULT) + private readonly defaultQueue: Queue) { } + /** + * Add a job to the queue with priority and options + */ + async addJob(name: string, data: T, options?: JobOptions): Promise> { + try { + const job = await this.defaultQueue.add(name, data, { + priority: options?.priority || JobPriority.NORMAL, + attempts: options?.attempts || 3, + backoff: options?.backoff || { + type: 'exponential', + delay: 2000, + }, + delay: options?.delay, + timeout: options?.timeout || 30000, + removeOnComplete: options?.removeOnComplete ?? true, + removeOnFail: options?.removeOnFail ?? false, + }); + this.logger.log(`Job ${name} added to queue with ID: ${job.id}, Priority: ${options?.priority || JobPriority.NORMAL}`); + return job; + } + catch (error) { + this.logger.error(`Failed to add job ${name}:`, error); + throw error; + } } - } - - /** - * Add multiple jobs in bulk - */ - async addBulkJobs( - jobs: Array<{ name: string; data: T; options?: JobOptions }>, - ): Promise>> { - try { - const bulkJobs = jobs.map((job) => ({ - name: job.name, - data: job.data, - opts: { - priority: job.options?.priority || JobPriority.NORMAL, - attempts: job.options?.attempts || 3, - backoff: job.options?.backoff || { - type: 'exponential', - delay: 2000, - }, - }, - })); - - const addedJobs = await this.defaultQueue.addBulk(bulkJobs); - this.logger.log(`Added ${addedJobs.length} jobs in bulk`); - return addedJobs; - } catch (error) { - this.logger.error('Failed to add bulk jobs:', error); - throw error; + /** + * Add multiple jobs in bulk + */ + async addBulkJobs(jobs: Array<{ + name: string; + data: T; + options?: JobOptions; + }>): Promise>> { + try { + const bulkJobs = jobs.map((job) => ({ + name: job.name, + data: job.data, + opts: { + priority: job.options?.priority || JobPriority.NORMAL, + attempts: job.options?.attempts || 3, + backoff: job.options?.backoff || { + type: 'exponential', + delay: 2000, + }, + }, + })); + const addedJobs = await this.defaultQueue.addBulk(bulkJobs); + this.logger.log(`Added ${addedJobs.length} jobs in bulk`); + return addedJobs; + } + catch (error) { + this.logger.error('Failed to add bulk jobs:', error); + throw error; + } } - } - - /** - * Get job by ID - */ - async getJob(jobId: string): Promise { - return this.defaultQueue.getJob(jobId); - } - - /** - * Get job metrics - */ - async getJobMetrics(jobId: string): Promise { - const job = await this.getJob(jobId); - if (!job) return null; - - const state = await job.getState(); - - return { - jobId: job.id.toString(), - name: job.name, - status: state as JobStatus, - priority: job.opts.priority as JobPriority, - attempts: job.attemptsMade, - maxAttempts: job.opts.attempts || 3, - progress: await job.progress(), - createdAt: new Date(job.timestamp), - processedAt: job.processedOn ? new Date(job.processedOn) : undefined, - finishedAt: job.finishedOn ? new Date(job.finishedOn) : undefined, - failedReason: job.failedReason, - data: job.data, - }; - } - - /** - * Remove a job from the queue - */ - async removeJob(jobId: string): Promise { - const job = await this.getJob(jobId); - if (job) { - await job.remove(); - this.logger.log(`Job ${jobId} removed from queue`); + /** + * Get job by ID + */ + async getJob(jobId: string): Promise { + return this.defaultQueue.getJob(jobId); } - } - - /** - * Retry a failed job - */ - async retryJob(jobId: string): Promise { - const job = await this.getJob(jobId); - if (job) { - await job.retry(); - this.logger.log(`Job ${jobId} retried`); + /** + * Get job metrics + */ + async getJobMetrics(jobId: string): Promise { + const job = await this.getJob(jobId); + if (!job) + return null; + const state = await job.getState(); + return { + jobId: job.id.toString(), + name: job.name, + status: state as JobStatus, + priority: job.opts.priority as JobPriority, + attempts: job.attemptsMade, + maxAttempts: job.opts.attempts || 3, + progress: await job.progress(), + createdAt: new Date(job.timestamp), + processedAt: job.processedOn ? new Date(job.processedOn) : undefined, + finishedAt: job.finishedOn ? new Date(job.finishedOn) : undefined, + failedReason: job.failedReason, + data: job.data, + }; } - } - - /** - * Pause the queue - */ - async pauseQueue(): Promise { - await this.defaultQueue.pause(); - this.logger.log('Queue paused'); - } - - /** - * Resume the queue - */ - async resumeQueue(): Promise { - await this.defaultQueue.resume(); - this.logger.log('Queue resumed'); - } - - /** - * Clean old jobs from the queue - */ - async cleanQueue(grace: number = 5000, status?: 'completed' | 'failed'): Promise { - if (status) { - await this.defaultQueue.clean(grace, status); - this.logger.log(`Cleaned ${status} jobs older than ${grace}ms`); - } else { - await this.defaultQueue.clean(grace, 'completed'); - await this.defaultQueue.clean(grace, 'failed'); - this.logger.log(`Cleaned all jobs older than ${grace}ms`); + /** + * Remove a job from the queue + */ + async removeJob(jobId: string): Promise { + const job = await this.getJob(jobId); + if (job) { + await job.remove(); + this.logger.log(`Job ${jobId} removed from queue`); + } + } + /** + * Retry a failed job + */ + async retryJob(jobId: string): Promise { + const job = await this.getJob(jobId); + if (job) { + await job.retry(); + this.logger.log(`Job ${jobId} retried`); + } + } + /** + * Pause the queue + */ + async pauseQueue(): Promise { + await this.defaultQueue.pause(); + this.logger.log('Queue paused'); + } + /** + * Resume the queue + */ + async resumeQueue(): Promise { + await this.defaultQueue.resume(); + this.logger.log('Queue resumed'); + } + /** + * Clean old jobs from the queue + */ + async cleanQueue(grace: number = 5000, status?: 'completed' | 'failed'): Promise { + if (status) { + await this.defaultQueue.clean(grace, status); + this.logger.log(`Cleaned ${status} jobs older than ${grace}ms`); + } + else { + await this.defaultQueue.clean(grace, 'completed'); + await this.defaultQueue.clean(grace, 'failed'); + this.logger.log(`Cleaned all jobs older than ${grace}ms`); + } + } + /** + * Get queue counts + */ + async getQueueCounts() { + return { + waiting: await this.defaultQueue.getWaitingCount(), + active: await this.defaultQueue.getActiveCount(), + completed: await this.defaultQueue.getCompletedCount(), + failed: await this.defaultQueue.getFailedCount(), + delayed: await this.defaultQueue.getDelayedCount(), + paused: await this.defaultQueue.getPausedCount(), + }; + } + /** + * Empty the queue (remove all jobs) + */ + async emptyQueue(): Promise { + await this.defaultQueue.empty(); + this.logger.warn('Queue emptied - all jobs removed'); } - } - - /** - * Get queue counts - */ - async getQueueCounts() { - return { - waiting: await this.defaultQueue.getWaitingCount(), - active: await this.defaultQueue.getActiveCount(), - completed: await this.defaultQueue.getCompletedCount(), - failed: await this.defaultQueue.getFailedCount(), - delayed: await this.defaultQueue.getDelayedCount(), - paused: await this.defaultQueue.getPausedCount(), - }; - } - - /** - * Empty the queue (remove all jobs) - */ - async emptyQueue(): Promise { - await this.defaultQueue.empty(); - this.logger.warn('Queue emptied - all jobs removed'); - } } diff --git a/src/queues/queue.spec.ts b/src/queues/queue.spec.ts index 0667ab8f..bfc943a0 100644 --- a/src/queues/queue.spec.ts +++ b/src/queues/queue.spec.ts @@ -2,1208 +2,1074 @@ * Self-contained tests for QueueMonitoringService logic. * All Bull queue methods are mocked inline. */ - // ─── Inline type definitions (no NestJS imports needed) interface TimestampedMetrics { - queueName: string; - waiting: number; - active: number; - completed: number; - failed: number; - delayed: number; - paused: number; - total: number; - throughput: number; - avgProcessingTime: number; - capturedAt: number; + queueName: string; + waiting: number; + active: number; + completed: number; + failed: number; + delayed: number; + paused: number; + total: number; + throughput: number; + avgProcessingTime: number; + capturedAt: number; } - interface QueueHealthStatus { - status: 'healthy' | 'warning' | 'critical'; - issues: string[]; - metrics: any; - timestamp: Date; + status: 'healthy' | 'warning' | 'critical'; + issues: string[]; + metrics: unknown; + timestamp: Date; } - interface BulkRetryResult { - requeued: number; - skipped: number; - errors: Array<{ jobId: string | number; reason: string }>; + requeued: number; + skipped: number; + errors: Array<{ + jobId: string | number; + reason: string; + }>; } - // ─── Minimal monitoring service (extracted logic, no decorators) - class QueueMonitoringService { - private readonly metricsHistory: TimestampedMetrics[] = []; - private readonly MAX_HISTORY_SIZE = 100; - private readonly THRESHOLDS = { - failureRateCritical: 0.2, - failureRateWarning: 0.1, - backlogCritical: 5_000, - backlogWarning: 1_000, - activeJobsCritical: 500, - activeJobsWarning: 100, - delayedJobsWarning: 500, - stuckThresholdMs: 300_000, - }; - - constructor(private readonly queue: any) {} - - async getQueueMetrics(): Promise { - const [waiting, active, completed, failed, delayed, paused] = await Promise.all([ - this.queue.getWaitingCount(), - this.queue.getActiveCount(), - this.queue.getCompletedCount(), - this.queue.getFailedCount(), - this.queue.getDelayedCount(), - this.queue.getPausedCount(), - ]); - - const total = waiting + active + completed + failed + delayed + paused; - const capturedAt = Date.now(); - const throughput = this.calculateThroughput(completed, capturedAt); - const avgProcessingTime = await this.calculateAvgProcessingTime(); - - const metrics: TimestampedMetrics = { - queueName: 'default', - waiting, - active, - completed, - failed, - delayed, - paused, - total, - throughput, - avgProcessingTime, - capturedAt, + private readonly metricsHistory: TimestampedMetrics[] = []; + private readonly MAX_HISTORY_SIZE = 100; + private readonly THRESHOLDS = { + failureRateCritical: 0.2, + failureRateWarning: 0.1, + backlogCritical: 5000, + backlogWarning: 1000, + activeJobsCritical: 500, + activeJobsWarning: 100, + delayedJobsWarning: 500, + stuckThresholdMs: 300000, }; - - this.appendToHistory(metrics); - return metrics; - } - - getMetricsHistory(): TimestampedMetrics[] { - return [...this.metricsHistory]; - } - - async checkQueueHealth(): Promise { - const metrics = await this.getQueueMetrics(); - const issues: string[] = []; - let status: 'healthy' | 'warning' | 'critical' = 'healthy'; - - const failureRate = metrics.total > 0 ? metrics.failed / metrics.total : 0; - if (failureRate > this.THRESHOLDS.failureRateCritical) { - issues.push(`Critical failure rate: ${(failureRate * 100).toFixed(1)}%`); - status = 'critical'; - } else if (failureRate > this.THRESHOLDS.failureRateWarning) { - issues.push(`Elevated failure rate: ${(failureRate * 100).toFixed(1)}%`); - if (status === 'healthy') status = 'warning'; + constructor(private readonly queue: unknown) { } + async getQueueMetrics(): Promise { + const [waiting, active, completed, failed, delayed, paused] = await Promise.all([ + this.queue.getWaitingCount(), + this.queue.getActiveCount(), + this.queue.getCompletedCount(), + this.queue.getFailedCount(), + this.queue.getDelayedCount(), + this.queue.getPausedCount(), + ]); + const total = waiting + active + completed + failed + delayed + paused; + const capturedAt = Date.now(); + const throughput = this.calculateThroughput(completed, capturedAt); + const avgProcessingTime = await this.calculateAvgProcessingTime(); + const metrics: TimestampedMetrics = { + queueName: 'default', + waiting, + active, + completed, + failed, + delayed, + paused, + total, + throughput, + avgProcessingTime, + capturedAt, + }; + this.appendToHistory(metrics); + return metrics; } - - if (metrics.waiting > this.THRESHOLDS.backlogCritical) { - issues.push(`Critical backlog: ${metrics.waiting} waiting jobs`); - status = 'critical'; - } else if (metrics.waiting > this.THRESHOLDS.backlogWarning) { - issues.push(`Elevated backlog: ${metrics.waiting} waiting jobs`); - if (status === 'healthy') status = 'warning'; + getMetricsHistory(): TimestampedMetrics[] { + return [...this.metricsHistory]; } - - if (metrics.active > this.THRESHOLDS.activeJobsCritical) { - issues.push(`Critical active-job count: ${metrics.active}`); - status = 'critical'; - } else if (metrics.active > this.THRESHOLDS.activeJobsWarning) { - issues.push(`High active-job count: ${metrics.active}`); - if (status === 'healthy') status = 'warning'; + async checkQueueHealth(): Promise { + const metrics = await this.getQueueMetrics(); + const issues: string[] = []; + let status: 'healthy' | 'warning' | 'critical' = 'healthy'; + const failureRate = metrics.total > 0 ? metrics.failed / metrics.total : 0; + if (failureRate > this.THRESHOLDS.failureRateCritical) { + issues.push(`Critical failure rate: ${(failureRate * 100).toFixed(1)}%`); + status = 'critical'; + } + else if (failureRate > this.THRESHOLDS.failureRateWarning) { + issues.push(`Elevated failure rate: ${(failureRate * 100).toFixed(1)}%`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.waiting > this.THRESHOLDS.backlogCritical) { + issues.push(`Critical backlog: ${metrics.waiting} waiting jobs`); + status = 'critical'; + } + else if (metrics.waiting > this.THRESHOLDS.backlogWarning) { + issues.push(`Elevated backlog: ${metrics.waiting} waiting jobs`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.active > this.THRESHOLDS.activeJobsCritical) { + issues.push(`Critical active-job count: ${metrics.active}`); + status = 'critical'; + } + else if (metrics.active > this.THRESHOLDS.activeJobsWarning) { + issues.push(`High active-job count: ${metrics.active}`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.delayed > this.THRESHOLDS.delayedJobsWarning) { + issues.push(`Many delayed jobs: ${metrics.delayed}`); + if (status === 'healthy') + status = 'warning'; + } + if (metrics.throughput === 0 && metrics.waiting > 0 && metrics.active === 0) { + issues.push('Queue appears stalled: jobs waiting but none active and throughput is zero'); + if (status === 'healthy') + status = 'warning'; + } + return { status, issues, metrics, timestamp: new Date() }; } - - if (metrics.delayed > this.THRESHOLDS.delayedJobsWarning) { - issues.push(`Many delayed jobs: ${metrics.delayed}`); - if (status === 'healthy') status = 'warning'; + async getQueueStatistics() { + const metrics = await this.getQueueMetrics(); + const history = this.metricsHistory; + return { + current: metrics, + trends: { + completed: this.calculateTrend(history.map((m) => m.completed)), + failed: this.calculateTrend(history.map((m) => m.failed)), + throughput: this.calculateTrend(history.map((m) => m.throughput)), + }, + health: await this.checkQueueHealth(), + }; } - - if (metrics.throughput === 0 && metrics.waiting > 0 && metrics.active === 0) { - issues.push('Queue appears stalled: jobs waiting but none active and throughput is zero'); - if (status === 'healthy') status = 'warning'; + async getFailedJobs(limit = 50, offset = 0, jobName?: string) { + if (jobName) { + const all = await this.queue.getFailed(0, 5000); + const filtered = all.filter((j: unknown) => j.name === jobName); + return filtered.slice(offset, offset + limit); + } + return this.queue.getFailed(offset, offset + limit - 1); } - - return { status, issues, metrics, timestamp: new Date() }; - } - - async getQueueStatistics() { - const metrics = await this.getQueueMetrics(); - const history = this.metricsHistory; - return { - current: metrics, - trends: { - completed: this.calculateTrend(history.map((m) => m.completed)), - failed: this.calculateTrend(history.map((m) => m.failed)), - throughput: this.calculateTrend(history.map((m) => m.throughput)), - }, - health: await this.checkQueueHealth(), - }; - } - - async getFailedJobs(limit = 50, offset = 0, jobName?: string) { - if (jobName) { - const all = await this.queue.getFailed(0, 5_000); - const filtered = all.filter((j: any) => j.name === jobName); - return filtered.slice(offset, offset + limit); + async retryAllFailedJobs(): Promise { + const failed = await this.queue.getFailed(0, 10000); + const result: BulkRetryResult = { requeued: 0, skipped: 0, errors: [] }; + for (const job of failed) { + try { + await job.retry(); + result.requeued++; + } + catch (err) { + result.errors.push({ jobId: job.id, reason: (err as Error).message }); + result.skipped++; + } + } + return result; } - return this.queue.getFailed(offset, offset + limit - 1); - } - - async retryAllFailedJobs(): Promise { - const failed = await this.queue.getFailed(0, 10_000); - const result: BulkRetryResult = { requeued: 0, skipped: 0, errors: [] }; - for (const job of failed) { - try { - await job.retry(); - result.requeued++; - } catch (err) { - result.errors.push({ jobId: job.id, reason: (err as Error).message }); - result.skipped++; - } + async getStuckJobs(thresholdMs = 300000) { + const active = await this.queue.getActive(); + const now = Date.now(); + return active.filter((job: unknown) => { + const startedAt = job.processedOn ?? job.timestamp; + return now - startedAt > thresholdMs; + }); } - return result; - } - - async getStuckJobs(thresholdMs = 300_000) { - const active = await this.queue.getActive(); - const now = Date.now(); - return active.filter((job: any) => { - const startedAt = job.processedOn ?? job.timestamp; - return now - startedAt > thresholdMs; - }); - } - - async getRetryAnalytics(windowMinutes = 60) { - const windowMs = windowMinutes * 60 * 1_000; - const cutoff = Date.now() - windowMs; - - const [failedJobs, completedJobs] = await Promise.all([ - this.queue.getFailed(0, 2_000), - this.queue.getCompleted(0, 2_000), - ]); - - const recentFailed = failedJobs.filter((j: any) => (j.finishedOn ?? j.timestamp) >= cutoff); - const recentCompleted = completedJobs.filter( - (j: any) => (j.finishedOn ?? j.timestamp) >= cutoff, - ); - - const successAfterRetry = recentCompleted.filter((j: any) => j.attemptsMade > 1); - const permanentlyFailed = recentFailed.filter( - (j: any) => j.attemptsMade >= (j.opts?.attempts ?? 3), - ); - const retried = recentFailed.filter((j: any) => j.attemptsMade > 1); - - const byJobType: Record = {}; - for (const job of [...recentFailed, ...recentCompleted]) { - const bucket = byJobType[job.name] ?? { failed: 0, retried: 0, avgAttempts: 0 }; - const isFailed = recentFailed.includes(job); - if (isFailed) bucket.failed++; - if (isFailed && job.attemptsMade > 1) bucket.retried++; - bucket.avgAttempts = - (bucket.avgAttempts * (bucket.failed + bucket.retried - 1) + job.attemptsMade) / - Math.max(bucket.failed + bucket.retried, 1); - byJobType[job.name] = bucket; + async getRetryAnalytics(windowMinutes = 60) { + const windowMs = windowMinutes * 60 * 1000; + const cutoff = Date.now() - windowMs; + const [failedJobs, completedJobs] = await Promise.all([ + this.queue.getFailed(0, 2000), + this.queue.getCompleted(0, 2000), + ]); + const recentFailed = failedJobs.filter((j: unknown) => (j.finishedOn ?? j.timestamp) >= cutoff); + const recentCompleted = completedJobs.filter((j: unknown) => (j.finishedOn ?? j.timestamp) >= cutoff); + const successAfterRetry = recentCompleted.filter((j: unknown) => j.attemptsMade > 1); + const permanentlyFailed = recentFailed.filter((j: unknown) => j.attemptsMade >= (j.opts?.attempts ?? 3)); + const retried = recentFailed.filter((j: unknown) => j.attemptsMade > 1); + const byJobType: Record = {}; + for (const job of [...recentFailed, ...recentCompleted]) { + const bucket = byJobType[job.name] ?? { failed: 0, retried: 0, avgAttempts: 0 }; + const isFailed = recentFailed.includes(job); + if (isFailed) + bucket.failed++; + if (isFailed && job.attemptsMade > 1) + bucket.retried++; + bucket.avgAttempts = + (bucket.avgAttempts * (bucket.failed + bucket.retried - 1) + job.attemptsMade) / + Math.max(bucket.failed + bucket.retried, 1); + byJobType[job.name] = bucket; + } + const totalFailed = recentFailed.length; + return { + windowMinutes, + totalFailed, + totalRetried: retried.length, + successAfterRetry: successAfterRetry.length, + permanentlyFailed: permanentlyFailed.length, + retryRate: totalFailed > 0 ? retried.length / totalFailed : 0, + successAfterRetryRate: retried.length > 0 ? successAfterRetry.length / retried.length : 0, + byJobType, + }; } - - const totalFailed = recentFailed.length; - return { - windowMinutes, - totalFailed, - totalRetried: retried.length, - successAfterRetry: successAfterRetry.length, - permanentlyFailed: permanentlyFailed.length, - retryRate: totalFailed > 0 ? retried.length / totalFailed : 0, - successAfterRetryRate: retried.length > 0 ? successAfterRetry.length / retried.length : 0, - byJobType, - }; - } - - private calculateThroughput(currentCompleted: number, capturedAt: number): number { - if (this.metricsHistory.length === 0) return 0; - const previous = this.metricsHistory[this.metricsHistory.length - 1]; - const deltaCompleted = Math.max(0, currentCompleted - previous.completed); - const deltaMs = capturedAt - previous.capturedAt; - if (deltaMs <= 0) return 0; - return Math.round((deltaCompleted / deltaMs) * 60_000); - } - - private async calculateAvgProcessingTime(): Promise { - try { - const completed = await this.queue.getCompleted(0, 100); - if (!completed.length) return 0; - const times = completed - .filter((j: any) => j.finishedOn != null && j.processedOn != null) - .map((j: any) => j.finishedOn - j.processedOn); - if (!times.length) return 0; - return Math.round(times.reduce((a: number, b: number) => a + b, 0) / times.length); - } catch { - return 0; + private calculateThroughput(currentCompleted: number, capturedAt: number): number { + if (this.metricsHistory.length === 0) + return 0; + const previous = this.metricsHistory[this.metricsHistory.length - 1]; + const deltaCompleted = Math.max(0, currentCompleted - previous.completed); + const deltaMs = capturedAt - previous.capturedAt; + if (deltaMs <= 0) + return 0; + return Math.round((deltaCompleted / deltaMs) * 60000); + } + private async calculateAvgProcessingTime(): Promise { + try { + const completed = await this.queue.getCompleted(0, 100); + if (!completed.length) + return 0; + const times = completed + .filter((j: unknown) => j.finishedOn != null && j.processedOn != null) + .map((j: unknown) => j.finishedOn - j.processedOn); + if (!times.length) + return 0; + return Math.round(times.reduce((a: number, b: number) => a + b, 0) / times.length); + } + catch { + return 0; + } + } + private appendToHistory(metrics: TimestampedMetrics): void { + this.metricsHistory.push(metrics); + if (this.metricsHistory.length > this.MAX_HISTORY_SIZE) + this.metricsHistory.shift(); + } + private calculateTrend(values: number[]): 'up' | 'down' | 'stable' { + if (values.length < 2) + return 'stable'; + const recent = values.slice(-10); + const avg = recent.reduce((a, b) => a + b, 0) / recent.length; + const latest = recent[recent.length - 1]; + if (avg === 0) + return 'stable'; + const changePct = ((latest - avg) / avg) * 100; + if (changePct > 10) + return 'up'; + if (changePct < -10) + return 'down'; + return 'stable'; } - } - - private appendToHistory(metrics: TimestampedMetrics): void { - this.metricsHistory.push(metrics); - if (this.metricsHistory.length > this.MAX_HISTORY_SIZE) this.metricsHistory.shift(); - } - - private calculateTrend(values: number[]): 'up' | 'down' | 'stable' { - if (values.length < 2) return 'stable'; - const recent = values.slice(-10); - const avg = recent.reduce((a, b) => a + b, 0) / recent.length; - const latest = recent[recent.length - 1]; - if (avg === 0) return 'stable'; - const changePct = ((latest - avg) / avg) * 100; - if (changePct > 10) return 'up'; - if (changePct < -10) return 'down'; - return 'stable'; - } } - // ─── Controller logic (extracted, no NestJS decorators) - class QueueController { - constructor( - private readonly queueService: any, - private readonly prioritizationService: any, - private readonly schedulerService: any, - private readonly monitoringService: QueueMonitoringService, - ) {} - - async getMetrics() { - return this.monitoringService.getQueueMetrics(); - } - async getStatistics() { - return this.monitoringService.getQueueStatistics(); - } - async getHealth() { - return this.monitoringService.checkQueueHealth(); - } - async getCounts() { - return this.queueService.getQueueCounts(); - } - async getMetricsHistory() { - return { history: this.monitoringService.getMetricsHistory() }; - } - async getRetryAnalytics(query: any) { - return this.monitoringService.getRetryAnalytics(query.windowMinutes ?? 60); - } - - async getFailedJobs(query: any) { - const limit = query.limit ?? 50; - const offset = query.offset ?? 0; - const jobs = await this.monitoringService.getFailedJobs(limit, offset, query.jobName); - return { - data: jobs.map((job: any) => ({ - id: job.id, - name: job.name, - data: job.data, - failedReason: job.failedReason, - attemptsMade: job.attemptsMade, - maxAttempts: job.opts?.attempts ?? 3, - stackTrace: job.stacktrace ?? [], - timestamp: new Date(job.timestamp).toISOString(), - processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, - finishedOn: job.finishedOn ? new Date(job.finishedOn).toISOString() : null, - })), - meta: { limit, offset, total: jobs.length }, - }; - } - - async retryAllFailedJobs() { - return this.monitoringService.retryAllFailedJobs(); - } - - async getStuckJobs(query: any) { - const threshold = query.threshold ?? 300_000; - const jobs = await this.monitoringService.getStuckJobs(threshold); - return { - data: jobs.map((job: any) => ({ - id: job.id, - name: job.name, - data: job.data, - processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, - activeForMs: job.processedOn ? Date.now() - job.processedOn : null, - attemptsMade: job.attemptsMade, - priority: job.opts?.priority, - })), - thresholdMs: threshold, - }; - } - - async getJob(id: string) { - const metrics = await this.queueService.getJobMetrics(id); - if (!metrics) throw new Error(`Job ${id} not found`); - return metrics; - } - - async retryJob(id: string) { - const job = await this.queueService.getJob(id); - if (!job) throw new Error(`Job ${id} not found`); - await this.queueService.retryJob(id); - return { message: 'Job retry initiated', jobId: id }; - } - - async removeJob(id: string) { - const job = await this.queueService.getJob(id); - if (!job) throw new Error(`Job ${id} not found`); - await this.queueService.removeJob(id); - } - - async addJob(body: any) { - let options = body.options ?? {}; - if (body.priorityFactors) { - const priority = this.prioritizationService.calculatePriority(body.priorityFactors); - options = { ...options, ...this.prioritizationService.getJobOptions(priority) }; + constructor(private readonly queueService: unknown, private readonly prioritizationService: unknown, private readonly schedulerService: unknown, private readonly monitoringService: QueueMonitoringService) { } + async getMetrics() { + return this.monitoringService.getQueueMetrics(); + } + async getStatistics() { + return this.monitoringService.getQueueStatistics(); + } + async getHealth() { + return this.monitoringService.checkQueueHealth(); + } + async getCounts() { + return this.queueService.getQueueCounts(); + } + async getMetricsHistory() { + return { history: this.monitoringService.getMetricsHistory() }; + } + async getRetryAnalytics(query: unknown) { + return this.monitoringService.getRetryAnalytics(query.windowMinutes ?? 60); + } + async getFailedJobs(query: unknown) { + const limit = query.limit ?? 50; + const offset = query.offset ?? 0; + const jobs = await this.monitoringService.getFailedJobs(limit, offset, query.jobName); + return { + data: jobs.map((job: unknown) => ({ + id: job.id, + name: job.name, + data: job.data, + failedReason: job.failedReason, + attemptsMade: job.attemptsMade, + maxAttempts: job.opts?.attempts ?? 3, + stackTrace: job.stacktrace ?? [], + timestamp: new Date(job.timestamp).toISOString(), + processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, + finishedOn: job.finishedOn ? new Date(job.finishedOn).toISOString() : null, + })), + meta: { limit, offset, total: jobs.length }, + }; + } + async retryAllFailedJobs() { + return this.monitoringService.retryAllFailedJobs(); + } + async getStuckJobs(query: unknown) { + const threshold = query.threshold ?? 300000; + const jobs = await this.monitoringService.getStuckJobs(threshold); + return { + data: jobs.map((job: unknown) => ({ + id: job.id, + name: job.name, + data: job.data, + processedOn: job.processedOn ? new Date(job.processedOn).toISOString() : null, + activeForMs: job.processedOn ? Date.now() - job.processedOn : null, + attemptsMade: job.attemptsMade, + priority: job.opts?.priority, + })), + thresholdMs: threshold, + }; + } + async getJob(id: string) { + const metrics = await this.queueService.getJobMetrics(id); + if (!metrics) + throw new Error(`Job ${id} not found`); + return metrics; + } + async retryJob(id: string) { + const job = await this.queueService.getJob(id); + if (!job) + throw new Error(`Job ${id} not found`); + await this.queueService.retryJob(id); + return { message: 'Job retry initiated', jobId: id }; + } + async removeJob(id: string) { + const job = await this.queueService.getJob(id); + if (!job) + throw new Error(`Job ${id} not found`); + await this.queueService.removeJob(id); + } + async addJob(body: unknown) { + let options = body.options ?? {}; + if (body.priorityFactors) { + const priority = this.prioritizationService.calculatePriority(body.priorityFactors); + options = { ...options, ...this.prioritizationService.getJobOptions(priority) }; + } + const job = await this.queueService.addJob(body.name, body.data, options); + return { jobId: job.id, name: job.name, priority: job.opts.priority, status: 'queued' }; + } + async addBulkJobs(body: unknown) { + const jobs = await this.queueService.addBulkJobs(body.jobs); + return { count: jobs.length, jobIds: jobs.map((j: unknown) => j.id) }; + } + async scheduleJob(body: unknown) { + const scheduledTime = new Date(body.scheduledTime); + if (scheduledTime <= new Date()) + throw new Error('scheduledTime must be in the future'); + const jobId = await this.schedulerService.scheduleJob(body.name, body.data, scheduledTime, body.options); + return { jobId, scheduledFor: body.scheduledTime, status: 'scheduled' }; + } + async scheduleDelayedJob(body: unknown) { + const jobId = await this.schedulerService.scheduleDelayedJob(body.name, body.data, body.delayMs, body.options); + return { jobId, delayMs: body.delayMs, status: 'scheduled' }; + } + async pauseQueue() { + await this.queueService.pauseQueue(); + return { message: 'Queue paused', timestamp: new Date().toISOString() }; + } + async resumeQueue() { + await this.queueService.resumeQueue(); + return { message: 'Queue resumed', timestamp: new Date().toISOString() }; + } + async cleanQueue(body: unknown) { + await this.queueService.cleanQueue(body.grace, body.status); + return { message: 'Queue cleaned', grace: body.grace ?? 5000, status: body.status ?? 'all' }; + } + async getActiveCronJobs() { + return { jobs: this.schedulerService.getActiveCronJobs() }; } - const job = await this.queueService.addJob(body.name, body.data, options); - return { jobId: job.id, name: job.name, priority: job.opts.priority, status: 'queued' }; - } - - async addBulkJobs(body: any) { - const jobs = await this.queueService.addBulkJobs(body.jobs); - return { count: jobs.length, jobIds: jobs.map((j: any) => j.id) }; - } - - async scheduleJob(body: any) { - const scheduledTime = new Date(body.scheduledTime); - if (scheduledTime <= new Date()) throw new Error('scheduledTime must be in the future'); - const jobId = await this.schedulerService.scheduleJob( - body.name, - body.data, - scheduledTime, - body.options, - ); - return { jobId, scheduledFor: body.scheduledTime, status: 'scheduled' }; - } - - async scheduleDelayedJob(body: any) { - const jobId = await this.schedulerService.scheduleDelayedJob( - body.name, - body.data, - body.delayMs, - body.options, - ); - return { jobId, delayMs: body.delayMs, status: 'scheduled' }; - } - - async pauseQueue() { - await this.queueService.pauseQueue(); - return { message: 'Queue paused', timestamp: new Date().toISOString() }; - } - - async resumeQueue() { - await this.queueService.resumeQueue(); - return { message: 'Queue resumed', timestamp: new Date().toISOString() }; - } - - async cleanQueue(body: any) { - await this.queueService.cleanQueue(body.grace, body.status); - return { message: 'Queue cleaned', grace: body.grace ?? 5000, status: body.status ?? 'all' }; - } - - async getActiveCronJobs() { - return { jobs: this.schedulerService.getActiveCronJobs() }; - } } - // ─── Mock factories - -function buildQueue(overrides: any = {}) { - return { - getWaitingCount: jest.fn().mockResolvedValue(0), - getActiveCount: jest.fn().mockResolvedValue(0), - getCompletedCount: jest.fn().mockResolvedValue(0), - getFailedCount: jest.fn().mockResolvedValue(0), - getDelayedCount: jest.fn().mockResolvedValue(0), - getPausedCount: jest.fn().mockResolvedValue(0), - getCompleted: jest.fn().mockResolvedValue([]), - getFailed: jest.fn().mockResolvedValue([]), - getActive: jest.fn().mockResolvedValue([]), - ...overrides, - }; +function buildQueue(overrides: unknown = {}) { + return { + getWaitingCount: jest.fn().mockResolvedValue(0), + getActiveCount: jest.fn().mockResolvedValue(0), + getCompletedCount: jest.fn().mockResolvedValue(0), + getFailedCount: jest.fn().mockResolvedValue(0), + getDelayedCount: jest.fn().mockResolvedValue(0), + getPausedCount: jest.fn().mockResolvedValue(0), + getCompleted: jest.fn().mockResolvedValue([]), + getFailed: jest.fn().mockResolvedValue([]), + getActive: jest.fn().mockResolvedValue([]), + ...overrides, + }; } - -function buildQueueService(overrides: any = {}) { - return { - addJob: jest.fn(), - addBulkJobs: jest.fn(), - getJob: jest.fn(), - getJobMetrics: jest.fn(), - removeJob: jest.fn(), - retryJob: jest.fn(), - pauseQueue: jest.fn(), - resumeQueue: jest.fn(), - cleanQueue: jest.fn(), - emptyQueue: jest.fn(), - getQueueCounts: jest.fn(), - ...overrides, - }; +function buildQueueService(overrides: unknown = {}) { + return { + addJob: jest.fn(), + addBulkJobs: jest.fn(), + getJob: jest.fn(), + getJobMetrics: jest.fn(), + removeJob: jest.fn(), + retryJob: jest.fn(), + pauseQueue: jest.fn(), + resumeQueue: jest.fn(), + cleanQueue: jest.fn(), + emptyQueue: jest.fn(), + getQueueCounts: jest.fn(), + ...overrides, + }; } - function buildPrioritizationService() { - return { - calculatePriority: jest.fn().mockReturnValue(3), - getJobOptions: jest.fn().mockReturnValue({ priority: 3 }), - }; + return { + calculatePriority: jest.fn().mockReturnValue(3), + getJobOptions: jest.fn().mockReturnValue({ priority: 3 }), + }; } - -function buildSchedulerService(overrides: any = {}) { - return { - scheduleJob: jest.fn(), - scheduleDelayedJob: jest.fn(), - getScheduledJobs: jest.fn(), - cancelScheduledJob: jest.fn(), - getActiveCronJobs: jest.fn().mockReturnValue([]), - ...overrides, - }; +function buildSchedulerService(overrides: unknown = {}) { + return { + scheduleJob: jest.fn(), + scheduleDelayedJob: jest.fn(), + getScheduledJobs: jest.fn(), + cancelScheduledJob: jest.fn(), + getActiveCronJobs: jest.fn().mockReturnValue([]), + ...overrides, + }; } - // ─── Tests: QueueMonitoringService - describe('QueueMonitoringService', () => { - let queue: any; - let service: QueueMonitoringService; - - beforeEach(() => { - queue = buildQueue(); - service = new QueueMonitoringService(queue); - }); - - // getQueueMetrics - describe('getQueueMetrics', () => { - it('sums all state counts into total', async () => { - queue.getWaitingCount.mockResolvedValue(10); - queue.getActiveCount.mockResolvedValue(3); - queue.getCompletedCount.mockResolvedValue(100); - queue.getFailedCount.mockResolvedValue(5); - queue.getDelayedCount.mockResolvedValue(2); - queue.getPausedCount.mockResolvedValue(1); - - const m = await service.getQueueMetrics(); - expect(m.waiting).toBe(10); - expect(m.active).toBe(3); - expect(m.total).toBe(121); - }); - - it('includes a capturedAt unix timestamp', async () => { - const before = Date.now(); - const m = await service.getQueueMetrics(); - expect(m.capturedAt).toBeGreaterThanOrEqual(before); - }); - - it('appends to history on each call', async () => { - await service.getQueueMetrics(); - await service.getQueueMetrics(); - expect(service.getMetricsHistory()).toHaveLength(2); - }); - - it('caps history at 100 entries', async () => { - for (let i = 0; i < 110; i++) await service.getQueueMetrics(); - expect(service.getMetricsHistory().length).toBeLessThanOrEqual(100); - }); - - it('calculates avgProcessingTime from completed jobs', async () => { - const now = Date.now(); - queue.getCompleted.mockResolvedValue([ - { processedOn: now - 2000, finishedOn: now - 0 }, - { processedOn: now - 4000, finishedOn: now - 2000 }, - ]); - const m = await service.getQueueMetrics(); - expect(m.avgProcessingTime).toBe(2000); - }); - - it('returns 0 avgProcessingTime when no completed jobs', async () => { - queue.getCompleted.mockResolvedValue([]); - const m = await service.getQueueMetrics(); - expect(m.avgProcessingTime).toBe(0); - }); - }); - - // Throughput calculation - describe('throughput', () => { - it('returns 0 on the very first snapshot', async () => { - queue.getCompletedCount.mockResolvedValue(50); - const m = await service.getQueueMetrics(); - expect(m.throughput).toBe(0); - }); - - it('computes jobs/min correctly between snapshots', async () => { - queue.getCompletedCount.mockResolvedValue(0); - await service.getQueueMetrics(); - - // Wind back the first snapshot's timestamp by 60 seconds - (service as any).metricsHistory[0].capturedAt = Date.now() - 60_000; - - queue.getCompletedCount.mockResolvedValue(60); - const m2 = await service.getQueueMetrics(); - // ~60 jobs in ~60 s → ~60 jobs/min - expect(m2.throughput).toBeGreaterThanOrEqual(55); - expect(m2.throughput).toBeLessThanOrEqual(65); - }); - - it('never returns negative throughput if completed count drops', async () => { - queue.getCompletedCount.mockResolvedValue(200); - await service.getQueueMetrics(); - - queue.getCompletedCount.mockResolvedValue(0); // reset after clean - const m2 = await service.getQueueMetrics(); - expect(m2.throughput).toBeGreaterThanOrEqual(0); - }); - }); - - // checkQueueHealth - describe('checkQueueHealth', () => { - it('returns healthy when all counts are zero', async () => { - const h = await service.checkQueueHealth(); - expect(h.status).toBe('healthy'); - expect(h.issues).toHaveLength(0); - }); - - it('returns warning for failure rate between 10% and 20%', async () => { - queue.getCompletedCount.mockResolvedValue(85); - queue.getFailedCount.mockResolvedValue(15); // 15% of 100 - const h = await service.checkQueueHealth(); - expect(h.status).toBe('warning'); - expect(h.issues.some((i) => i.includes('failure rate'))).toBe(true); - }); - - it('returns critical for failure rate above 20%', async () => { - queue.getCompletedCount.mockResolvedValue(70); - queue.getFailedCount.mockResolvedValue(30); // 30% - const h = await service.checkQueueHealth(); - expect(h.status).toBe('critical'); - }); - - it('returns warning for backlog between 1000 and 5000', async () => { - queue.getWaitingCount.mockResolvedValue(2000); - const h = await service.checkQueueHealth(); - expect(h.status).toBe('warning'); - expect(h.issues.some((i) => i.includes('backlog'))).toBe(true); - }); - - it('returns critical for backlog > 5000', async () => { - queue.getWaitingCount.mockResolvedValue(6000); - const h = await service.checkQueueHealth(); - expect(h.status).toBe('critical'); - }); - - it('detects stalled queue (waiting > 0, active = 0, throughput = 0)', async () => { - queue.getWaitingCount.mockResolvedValue(50); - queue.getActiveCount.mockResolvedValue(0); - const h = await service.checkQueueHealth(); - expect(h.issues.some((i) => i.toLowerCase().includes('stall'))).toBe(true); - }); - - it('includes timestamp in every response', async () => { - const h = await service.checkQueueHealth(); - expect(h.timestamp).toBeInstanceOf(Date); - }); - - it('includes metrics snapshot in response', async () => { - const h = await service.checkQueueHealth(); - expect(h.metrics).toBeDefined(); - expect(h.metrics.queueName).toBe('default'); - }); - }); - - // getFailedJobs - describe('getFailedJobs', () => { - it('calls queue.getFailed with correct start/end for pagination', async () => { - queue.getFailed.mockResolvedValue([]); - await service.getFailedJobs(10, 20); - expect(queue.getFailed).toHaveBeenCalledWith(20, 29); - }); - - it('filters by jobName when provided', async () => { - queue.getFailed.mockResolvedValue([ - { id: '1', name: 'email', finishedOn: Date.now() }, - { id: '2', name: 'backup', finishedOn: Date.now() }, - { id: '3', name: 'email', finishedOn: Date.now() }, - ]); - const results = await service.getFailedJobs(50, 0, 'email'); - expect(results).toHaveLength(2); - expect(results.every((r: any) => r.name === 'email')).toBe(true); - }); - - it('applies pagination after filtering', async () => { - const jobs = Array.from({ length: 10 }, (_, i) => ({ - id: String(i), - name: 'email', - finishedOn: Date.now(), - })); - queue.getFailed.mockResolvedValue(jobs); - const results = await service.getFailedJobs(3, 2, 'email'); - expect(results).toHaveLength(3); - expect(results[0].id).toBe('2'); - }); - - it('uses default limit=50 offset=0', async () => { - queue.getFailed.mockResolvedValue([]); - await service.getFailedJobs(); - expect(queue.getFailed).toHaveBeenCalledWith(0, 49); - }); - }); - - // retryAllFailedJobs - describe('retryAllFailedJobs', () => { - it('returns correct counts when all retries succeed', async () => { - const retryFn = jest.fn().mockResolvedValue(undefined); - queue.getFailed.mockResolvedValue([ - { id: '1', retry: retryFn }, - { id: '2', retry: retryFn }, - ]); - const result = await service.retryAllFailedJobs(); - expect(result.requeued).toBe(2); - expect(result.skipped).toBe(0); - expect(result.errors).toHaveLength(0); - }); - - it('tracks errors for jobs that fail to retry', async () => { - queue.getFailed.mockResolvedValue([ - { id: '1', retry: jest.fn().mockResolvedValue(undefined) }, - { id: '2', retry: jest.fn().mockRejectedValue(new Error('locked')) }, - { id: '3', retry: jest.fn().mockRejectedValue(new Error('gone')) }, - ]); - const result = await service.retryAllFailedJobs(); - expect(result.requeued).toBe(1); - expect(result.skipped).toBe(2); - expect(result.errors[0].jobId).toBe('2'); - expect(result.errors[1].reason).toBe('gone'); - }); - - it('handles empty failed list gracefully', async () => { - queue.getFailed.mockResolvedValue([]); - const result = await service.retryAllFailedJobs(); - expect(result.requeued).toBe(0); - expect(result.skipped).toBe(0); - expect(result.errors).toHaveLength(0); - }); - }); - - // getStuckJobs - describe('getStuckJobs', () => { - it('returns jobs active longer than threshold', async () => { - const now = Date.now(); - queue.getActive.mockResolvedValue([ - { id: '1', processedOn: now - 400_000 }, // stuck - { id: '2', processedOn: now - 100_000 }, // fine - { id: '3', processedOn: now - 600_000 }, // stuck - ]); - const stuck = await service.getStuckJobs(300_000); - expect(stuck).toHaveLength(2); - const ids = stuck.map((j: any) => j.id); - expect(ids).toContain('1'); - expect(ids).toContain('3'); - }); - - it('returns empty array when all active jobs are within threshold', async () => { - queue.getActive.mockResolvedValue([{ id: '1', processedOn: Date.now() - 1_000 }]); - const stuck = await service.getStuckJobs(300_000); - expect(stuck).toHaveLength(0); - }); - - it('falls back to job.timestamp when processedOn is null', async () => { - const now = Date.now(); - queue.getActive.mockResolvedValue([ - { id: '99', processedOn: null, timestamp: now - 999_000 }, - ]); - const stuck = await service.getStuckJobs(300_000); - expect(stuck).toHaveLength(1); - expect(stuck[0].id).toBe('99'); - }); - - it('returns zero stuck jobs for an empty active list', async () => { - queue.getActive.mockResolvedValue([]); - const stuck = await service.getStuckJobs(300_000); - expect(stuck).toHaveLength(0); - }); - }); - - // getRetryAnalytics - describe('getRetryAnalytics', () => { - const now = Date.now(); - - const recentFailed = [ - { - id: 'f1', - name: 'send-email', - finishedOn: now - 1_000, - timestamp: now - 5_000, - attemptsMade: 3, - opts: { attempts: 3 }, - }, - { - id: 'f2', - name: 'send-email', - finishedOn: now - 2_000, - timestamp: now - 6_000, - attemptsMade: 2, - opts: { attempts: 3 }, - }, - { - id: 'f3', - name: 'backup', - finishedOn: now - 3_000, - timestamp: now - 7_000, - attemptsMade: 1, - opts: { attempts: 3 }, - }, - ]; - const recentCompleted = [ - { - id: 'c1', - name: 'send-email', - finishedOn: now - 500, - timestamp: now - 3_000, - attemptsMade: 2, - opts: { attempts: 3 }, - }, - ]; - + let queue: unknown; + let service: QueueMonitoringService; beforeEach(() => { - queue.getFailed.mockResolvedValue(recentFailed); - queue.getCompleted.mockResolvedValue(recentCompleted); - }); - - it('counts total failed correctly', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.totalFailed).toBe(3); - }); - - it('counts retried jobs (attemptsMade > 1)', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.totalRetried).toBe(2); // f1 and f2 - }); - - it('counts permanently failed jobs (attempts >= maxAttempts)', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.permanentlyFailed).toBe(1); // only f1 (3/3) - }); - - it('counts success-after-retry from completed jobs', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.successAfterRetry).toBe(1); // c1 had attemptsMade=2 - }); - - it('computes retry rate as fraction of failed that were retried', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.retryRate).toBeCloseTo(2 / 3, 2); - }); - - it('produces per-job-type breakdown', async () => { - const a = await service.getRetryAnalytics(60); - expect(a.byJobType['send-email']).toBeDefined(); - expect(a.byJobType['backup']).toBeDefined(); - expect(a.byJobType['send-email'].failed).toBeGreaterThan(0); - }); - - it('excludes jobs outside the time window', async () => { - const old = { - id: 'old', - name: 'stale-job', - finishedOn: now - 7_200_000, - timestamp: now - 7_200_000, - attemptsMade: 1, - opts: {}, - }; - queue.getFailed.mockResolvedValue([...recentFailed, old]); - const a = await service.getRetryAnalytics(60); - expect(a.totalFailed).toBe(3); // old excluded - expect(a.byJobType['stale-job']).toBeUndefined(); - }); - - it('returns all-zeros when there are no recent failures', async () => { - queue.getFailed.mockResolvedValue([]); - queue.getCompleted.mockResolvedValue([]); - const a = await service.getRetryAnalytics(60); - expect(a.totalFailed).toBe(0); - expect(a.retryRate).toBe(0); - expect(a.successAfterRetryRate).toBe(0); + queue = buildQueue(); + service = new QueueMonitoringService(queue); + }); + // getQueueMetrics + describe('getQueueMetrics', () => { + it('sums all state counts into total', async () => { + queue.getWaitingCount.mockResolvedValue(10); + queue.getActiveCount.mockResolvedValue(3); + queue.getCompletedCount.mockResolvedValue(100); + queue.getFailedCount.mockResolvedValue(5); + queue.getDelayedCount.mockResolvedValue(2); + queue.getPausedCount.mockResolvedValue(1); + const m = await service.getQueueMetrics(); + expect(m.waiting).toBe(10); + expect(m.active).toBe(3); + expect(m.total).toBe(121); + }); + it('includes a capturedAt unix timestamp', async () => { + const before = Date.now(); + const m = await service.getQueueMetrics(); + expect(m.capturedAt).toBeGreaterThanOrEqual(before); + }); + it('appends to history on each call', async () => { + await service.getQueueMetrics(); + await service.getQueueMetrics(); + expect(service.getMetricsHistory()).toHaveLength(2); + }); + it('caps history at 100 entries', async () => { + for (let i = 0; i < 110; i++) + await service.getQueueMetrics(); + expect(service.getMetricsHistory().length).toBeLessThanOrEqual(100); + }); + it('calculates avgProcessingTime from completed jobs', async () => { + const now = Date.now(); + queue.getCompleted.mockResolvedValue([ + { processedOn: now - 2000, finishedOn: now - 0 }, + { processedOn: now - 4000, finishedOn: now - 2000 }, + ]); + const m = await service.getQueueMetrics(); + expect(m.avgProcessingTime).toBe(2000); + }); + it('returns 0 avgProcessingTime when no completed jobs', async () => { + queue.getCompleted.mockResolvedValue([]); + const m = await service.getQueueMetrics(); + expect(m.avgProcessingTime).toBe(0); + }); }); - - it('respects custom windowMinutes', async () => { - const a = await service.getRetryAnalytics(30); - expect(a.windowMinutes).toBe(30); + // Throughput calculation + describe('throughput', () => { + it('returns 0 on the very first snapshot', async () => { + queue.getCompletedCount.mockResolvedValue(50); + const m = await service.getQueueMetrics(); + expect(m.throughput).toBe(0); + }); + it('computes jobs/min correctly between snapshots', async () => { + queue.getCompletedCount.mockResolvedValue(0); + await service.getQueueMetrics(); + // Wind back the first snapshot's timestamp by 60 seconds + (service as unknown).metricsHistory[0].capturedAt = Date.now() - 60000; + queue.getCompletedCount.mockResolvedValue(60); + const m2 = await service.getQueueMetrics(); + // ~60 jobs in ~60 s → ~60 jobs/min + expect(m2.throughput).toBeGreaterThanOrEqual(55); + expect(m2.throughput).toBeLessThanOrEqual(65); + }); + it('never returns negative throughput if completed count drops', async () => { + queue.getCompletedCount.mockResolvedValue(200); + await service.getQueueMetrics(); + queue.getCompletedCount.mockResolvedValue(0); // reset after clean + const m2 = await service.getQueueMetrics(); + expect(m2.throughput).toBeGreaterThanOrEqual(0); + }); }); - }); - - // getMetricsHistory - describe('getMetricsHistory', () => { - it('returns empty array before any snapshots', () => { - expect(service.getMetricsHistory()).toHaveLength(0); + // checkQueueHealth + describe('checkQueueHealth', () => { + it('returns healthy when all counts are zero', async () => { + const h = await service.checkQueueHealth(); + expect(h.status).toBe('healthy'); + expect(h.issues).toHaveLength(0); + }); + it('returns warning for failure rate between 10% and 20%', async () => { + queue.getCompletedCount.mockResolvedValue(85); + queue.getFailedCount.mockResolvedValue(15); // 15% of 100 + const h = await service.checkQueueHealth(); + expect(h.status).toBe('warning'); + expect(h.issues.some((i) => i.includes('failure rate'))).toBe(true); + }); + it('returns critical for failure rate above 20%', async () => { + queue.getCompletedCount.mockResolvedValue(70); + queue.getFailedCount.mockResolvedValue(30); // 30% + const h = await service.checkQueueHealth(); + expect(h.status).toBe('critical'); + }); + it('returns warning for backlog between 1000 and 5000', async () => { + queue.getWaitingCount.mockResolvedValue(2000); + const h = await service.checkQueueHealth(); + expect(h.status).toBe('warning'); + expect(h.issues.some((i) => i.includes('backlog'))).toBe(true); + }); + it('returns critical for backlog > 5000', async () => { + queue.getWaitingCount.mockResolvedValue(6000); + const h = await service.checkQueueHealth(); + expect(h.status).toBe('critical'); + }); + it('detects stalled queue (waiting > 0, active = 0, throughput = 0)', async () => { + queue.getWaitingCount.mockResolvedValue(50); + queue.getActiveCount.mockResolvedValue(0); + const h = await service.checkQueueHealth(); + expect(h.issues.some((i) => i.toLowerCase().includes('stall'))).toBe(true); + }); + it('includes timestamp in every response', async () => { + const h = await service.checkQueueHealth(); + expect(h.timestamp).toBeInstanceOf(Date); + }); + it('includes metrics snapshot in response', async () => { + const h = await service.checkQueueHealth(); + expect(h.metrics).toBeDefined(); + expect(h.metrics.queueName).toBe('default'); + }); }); - - it('returns a copy — mutations do not affect internal state', async () => { - await service.getQueueMetrics(); - const h = service.getMetricsHistory(); - h.push({} as any); - expect(service.getMetricsHistory()).toHaveLength(1); + // getFailedJobs + describe('getFailedJobs', () => { + it('calls queue.getFailed with correct start/end for pagination', async () => { + queue.getFailed.mockResolvedValue([]); + await service.getFailedJobs(10, 20); + expect(queue.getFailed).toHaveBeenCalledWith(20, 29); + }); + it('filters by jobName when provided', async () => { + queue.getFailed.mockResolvedValue([ + { id: '1', name: 'email', finishedOn: Date.now() }, + { id: '2', name: 'backup', finishedOn: Date.now() }, + { id: '3', name: 'email', finishedOn: Date.now() }, + ]); + const results = await service.getFailedJobs(50, 0, 'email'); + expect(results).toHaveLength(2); + expect(results.every((r: unknown) => r.name === 'email')).toBe(true); + }); + it('applies pagination after filtering', async () => { + const jobs = Array.from({ length: 10 }, (_, i) => ({ + id: String(i), + name: 'email', + finishedOn: Date.now(), + })); + queue.getFailed.mockResolvedValue(jobs); + const results = await service.getFailedJobs(3, 2, 'email'); + expect(results).toHaveLength(3); + expect(results[0].id).toBe('2'); + }); + it('uses default limit=50 offset=0', async () => { + queue.getFailed.mockResolvedValue([]); + await service.getFailedJobs(); + expect(queue.getFailed).toHaveBeenCalledWith(0, 49); + }); }); - - it('each entry has capturedAt', async () => { - await service.getQueueMetrics(); - await service.getQueueMetrics(); - service.getMetricsHistory().forEach((m) => { - expect(typeof m.capturedAt).toBe('number'); - expect(m.capturedAt).toBeGreaterThan(0); - }); + // retryAllFailedJobs + describe('retryAllFailedJobs', () => { + it('returns correct counts when all retries succeed', async () => { + const retryFn = jest.fn().mockResolvedValue(undefined); + queue.getFailed.mockResolvedValue([ + { id: '1', retry: retryFn }, + { id: '2', retry: retryFn }, + ]); + const result = await service.retryAllFailedJobs(); + expect(result.requeued).toBe(2); + expect(result.skipped).toBe(0); + expect(result.errors).toHaveLength(0); + }); + it('tracks errors for jobs that fail to retry', async () => { + queue.getFailed.mockResolvedValue([ + { id: '1', retry: jest.fn().mockResolvedValue(undefined) }, + { id: '2', retry: jest.fn().mockRejectedValue(new Error('locked')) }, + { id: '3', retry: jest.fn().mockRejectedValue(new Error('gone')) }, + ]); + const result = await service.retryAllFailedJobs(); + expect(result.requeued).toBe(1); + expect(result.skipped).toBe(2); + expect(result.errors[0].jobId).toBe('2'); + expect(result.errors[1].reason).toBe('gone'); + }); + it('handles empty failed list gracefully', async () => { + queue.getFailed.mockResolvedValue([]); + const result = await service.retryAllFailedJobs(); + expect(result.requeued).toBe(0); + expect(result.skipped).toBe(0); + expect(result.errors).toHaveLength(0); + }); }); - }); - - // getQueueStatistics - describe('getQueueStatistics', () => { - it('returns current, trends, and health keys', async () => { - const s = await service.getQueueStatistics(); - expect(s).toHaveProperty('current'); - expect(s).toHaveProperty('trends'); - expect(s).toHaveProperty('health'); - expect(s.trends).toHaveProperty('completed'); - expect(s.trends).toHaveProperty('failed'); - expect(s.trends).toHaveProperty('throughput'); + // getStuckJobs + describe('getStuckJobs', () => { + it('returns jobs active longer than threshold', async () => { + const now = Date.now(); + queue.getActive.mockResolvedValue([ + { id: '1', processedOn: now - 400000 }, // stuck + { id: '2', processedOn: now - 100000 }, // fine + { id: '3', processedOn: now - 600000 }, // stuck + ]); + const stuck = await service.getStuckJobs(300000); + expect(stuck).toHaveLength(2); + const ids = stuck.map((j: unknown) => j.id); + expect(ids).toContain('1'); + expect(ids).toContain('3'); + }); + it('returns empty array when all active jobs are within threshold', async () => { + queue.getActive.mockResolvedValue([{ id: '1', processedOn: Date.now() - 1000 }]); + const stuck = await service.getStuckJobs(300000); + expect(stuck).toHaveLength(0); + }); + it('falls back to job.timestamp when processedOn is null', async () => { + const now = Date.now(); + queue.getActive.mockResolvedValue([ + { id: '99', processedOn: null, timestamp: now - 999000 }, + ]); + const stuck = await service.getStuckJobs(300000); + expect(stuck).toHaveLength(1); + expect(stuck[0].id).toBe('99'); + }); + it('returns zero stuck jobs for an empty active list', async () => { + queue.getActive.mockResolvedValue([]); + const stuck = await service.getStuckJobs(300000); + expect(stuck).toHaveLength(0); + }); }); - - it('trend is stable with a single data point', async () => { - const s = await service.getQueueStatistics(); - expect(s.trends.completed).toBe('stable'); + // getRetryAnalytics + describe('getRetryAnalytics', () => { + const now = Date.now(); + const recentFailed = [ + { + id: 'f1', + name: 'send-email', + finishedOn: now - 1000, + timestamp: now - 5000, + attemptsMade: 3, + opts: { attempts: 3 }, + }, + { + id: 'f2', + name: 'send-email', + finishedOn: now - 2000, + timestamp: now - 6000, + attemptsMade: 2, + opts: { attempts: 3 }, + }, + { + id: 'f3', + name: 'backup', + finishedOn: now - 3000, + timestamp: now - 7000, + attemptsMade: 1, + opts: { attempts: 3 }, + }, + ]; + const recentCompleted = [ + { + id: 'c1', + name: 'send-email', + finishedOn: now - 500, + timestamp: now - 3000, + attemptsMade: 2, + opts: { attempts: 3 }, + }, + ]; + beforeEach(() => { + queue.getFailed.mockResolvedValue(recentFailed); + queue.getCompleted.mockResolvedValue(recentCompleted); + }); + it('counts total failed correctly', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.totalFailed).toBe(3); + }); + it('counts retried jobs (attemptsMade > 1)', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.totalRetried).toBe(2); // f1 and f2 + }); + it('counts permanently failed jobs (attempts >= maxAttempts)', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.permanentlyFailed).toBe(1); // only f1 (3/3) + }); + it('counts success-after-retry from completed jobs', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.successAfterRetry).toBe(1); // c1 had attemptsMade=2 + }); + it('computes retry rate as fraction of failed that were retried', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.retryRate).toBeCloseTo(2 / 3, 2); + }); + it('produces per-job-type breakdown', async () => { + const a = await service.getRetryAnalytics(60); + expect(a.byJobType['send-email']).toBeDefined(); + expect(a.byJobType['backup']).toBeDefined(); + expect(a.byJobType['send-email'].failed).toBeGreaterThan(0); + }); + it('excludes jobs outside the time window', async () => { + const old = { + id: 'old', + name: 'stale-job', + finishedOn: now - 7200000, + timestamp: now - 7200000, + attemptsMade: 1, + opts: {}, + }; + queue.getFailed.mockResolvedValue([...recentFailed, old]); + const a = await service.getRetryAnalytics(60); + expect(a.totalFailed).toBe(3); // old excluded + expect(a.byJobType['stale-job']).toBeUndefined(); + }); + it('returns all-zeros when there are no recent failures', async () => { + queue.getFailed.mockResolvedValue([]); + queue.getCompleted.mockResolvedValue([]); + const a = await service.getRetryAnalytics(60); + expect(a.totalFailed).toBe(0); + expect(a.retryRate).toBe(0); + expect(a.successAfterRetryRate).toBe(0); + }); + it('respects custom windowMinutes', async () => { + const a = await service.getRetryAnalytics(30); + expect(a.windowMinutes).toBe(30); + }); }); - - it('trend is up when completed counts rise', async () => { - // Pre-seed history with a clear upward ramp: 0, 100, 200 ... 1000 - const history = (service as any).metricsHistory as any[]; - for (let i = 0; i < 11; i++) { - history.push({ - queueName: 'default', - completed: i * 100, - failed: 0, - throughput: 0, - capturedAt: Date.now() - (10 - i) * 60_000, - waiting: 0, - active: 0, - delayed: 0, - paused: 0, - total: 0, - avgProcessingTime: 0, - }); - } - // The live snapshot appended by getQueueMetrics must continue the rise. - // Mock a value clearly above the 10-entry window average (~500). - queue.getCompletedCount.mockResolvedValue(1500); - const s = await service.getQueueStatistics(); - expect(s.trends.completed).toBe('up'); + // getMetricsHistory + describe('getMetricsHistory', () => { + it('returns empty array before any snapshots', () => { + expect(service.getMetricsHistory()).toHaveLength(0); + }); + it('returns a copy — mutations do not affect internal state', async () => { + await service.getQueueMetrics(); + const h = service.getMetricsHistory(); + h.push({} as unknown); + expect(service.getMetricsHistory()).toHaveLength(1); + }); + it('each entry has capturedAt', async () => { + await service.getQueueMetrics(); + await service.getQueueMetrics(); + service.getMetricsHistory().forEach((m) => { + expect(typeof m.capturedAt).toBe('number'); + expect(m.capturedAt).toBeGreaterThan(0); + }); + }); }); - - it('trend is down when completed counts fall', async () => { - const history = (service as any).metricsHistory as any[]; - for (let i = 0; i < 11; i++) { - history.push({ - queueName: 'default', - completed: (10 - i) * 100, - failed: 0, - throughput: 0, - capturedAt: Date.now() - (10 - i) * 60_000, - waiting: 0, - active: 0, - delayed: 0, - paused: 0, - total: 0, - avgProcessingTime: 0, - }); - } - const s = await service.getQueueStatistics(); - expect(s.trends.completed).toBe('down'); + // getQueueStatistics + describe('getQueueStatistics', () => { + it('returns current, trends, and health keys', async () => { + const s = await service.getQueueStatistics(); + expect(s).toHaveProperty('current'); + expect(s).toHaveProperty('trends'); + expect(s).toHaveProperty('health'); + expect(s.trends).toHaveProperty('completed'); + expect(s.trends).toHaveProperty('failed'); + expect(s.trends).toHaveProperty('throughput'); + }); + it('trend is stable with a single data point', async () => { + const s = await service.getQueueStatistics(); + expect(s.trends.completed).toBe('stable'); + }); + it('trend is up when completed counts rise', async () => { + // Pre-seed history with a clear upward ramp: 0, 100, 200 ... 1000 + const history = (service as unknown).metricsHistory as unknown[]; + for (let i = 0; i < 11; i++) { + history.push({ + queueName: 'default', + completed: i * 100, + failed: 0, + throughput: 0, + capturedAt: Date.now() - (10 - i) * 60000, + waiting: 0, + active: 0, + delayed: 0, + paused: 0, + total: 0, + avgProcessingTime: 0, + }); + } + // The live snapshot appended by getQueueMetrics must continue the rise. + // Mock a value clearly above the 10-entry window average (~500). + queue.getCompletedCount.mockResolvedValue(1500); + const s = await service.getQueueStatistics(); + expect(s.trends.completed).toBe('up'); + }); + it('trend is down when completed counts fall', async () => { + const history = (service as unknown).metricsHistory as unknown[]; + for (let i = 0; i < 11; i++) { + history.push({ + queueName: 'default', + completed: (10 - i) * 100, + failed: 0, + throughput: 0, + capturedAt: Date.now() - (10 - i) * 60000, + waiting: 0, + active: 0, + delayed: 0, + paused: 0, + total: 0, + avgProcessingTime: 0, + }); + } + const s = await service.getQueueStatistics(); + expect(s.trends.completed).toBe('down'); + }); }); - }); }); - // ─── Tests: QueueController - describe('QueueController', () => { - let queueService: any; - let prioritizationService: any; - let schedulerService: any; - let monitoringService: QueueMonitoringService; - let controller: QueueController; - let queue: any; - - beforeEach(() => { - queue = buildQueue(); - queueService = buildQueueService(); - prioritizationService = buildPrioritizationService(); - schedulerService = buildSchedulerService(); - monitoringService = new QueueMonitoringService(queue); - controller = new QueueController( - queueService, - prioritizationService, - schedulerService, - monitoringService, - ); - }); - - // Monitoring endpoints - describe('getMetrics', () => { - it('returns metrics from monitoring service', async () => { - const m = await controller.getMetrics(); - expect(m.queueName).toBe('default'); - }); - }); - - describe('getHealth', () => { - it('returns health status object', async () => { - const h = await controller.getHealth(); - expect(['healthy', 'warning', 'critical']).toContain(h.status); - expect(Array.isArray(h.issues)).toBe(true); - }); - }); - - describe('getCounts', () => { - it('delegates to queueService.getQueueCounts', async () => { - queueService.getQueueCounts.mockResolvedValue({ waiting: 5 }); - const result = await controller.getCounts(); - expect(result).toEqual({ waiting: 5 }); - }); - }); - - describe('getMetricsHistory', () => { - it('returns wrapped history array', async () => { - const result = await controller.getMetricsHistory(); - expect(result).toHaveProperty('history'); - expect(Array.isArray(result.history)).toBe(true); - }); - }); - - describe('getRetryAnalytics', () => { - it('uses 60 as default windowMinutes', async () => { - queue.getFailed.mockResolvedValue([]); - queue.getCompleted.mockResolvedValue([]); - const result = await controller.getRetryAnalytics({ windowMinutes: undefined }); - expect(result.windowMinutes).toBe(60); - }); - - it('passes custom windowMinutes', async () => { - queue.getFailed.mockResolvedValue([]); - queue.getCompleted.mockResolvedValue([]); - const result = await controller.getRetryAnalytics({ windowMinutes: 30 }); - expect(result.windowMinutes).toBe(30); - }); - }); - - // Failed jobs - describe('getFailedJobs', () => { - const now = Date.now(); - const failedJob = { - id: '1', - name: 'email', - data: {}, - failedReason: 'timeout', - attemptsMade: 2, - opts: { attempts: 3 }, - stacktrace: ['at ...'], - timestamp: now - 10000, - processedOn: now - 8000, - finishedOn: now - 5000, - }; - + let queueService: unknown; + let prioritizationService: unknown; + let schedulerService: unknown; + let monitoringService: QueueMonitoringService; + let controller: QueueController; + let queue: unknown; beforeEach(() => { - queue.getFailed.mockResolvedValue([failedJob]); - }); - - it('shapes response with data and meta', async () => { - const result = await controller.getFailedJobs({ limit: 50, offset: 0 }); - expect(result.data).toHaveLength(1); - expect(result.data[0].id).toBe('1'); - expect(result.data[0].maxAttempts).toBe(3); - expect(result.data[0].stackTrace).toEqual(['at ...']); - expect(result.meta.limit).toBe(50); - }); - - it('null processedOn/finishedOn when absent', async () => { - queue.getFailed.mockResolvedValue([{ ...failedJob, processedOn: null, finishedOn: null }]); - const result = await controller.getFailedJobs({}); - expect(result.data[0].processedOn).toBeNull(); - expect(result.data[0].finishedOn).toBeNull(); - }); - }); - - describe('retryAllFailedJobs', () => { - it('returns bulk retry summary', async () => { - queue.getFailed.mockResolvedValue([ - { id: '1', retry: jest.fn().mockResolvedValue(undefined) }, - ]); - const result = await controller.retryAllFailedJobs(); - expect(result.requeued).toBe(1); - }); - }); - - // Stuck jobs - describe('getStuckJobs', () => { - it('uses 300000 as default threshold', async () => { - queue.getActive.mockResolvedValue([]); - const result = await controller.getStuckJobs({}); - expect(result.thresholdMs).toBe(300_000); + queue = buildQueue(); + queueService = buildQueueService(); + prioritizationService = buildPrioritizationService(); + schedulerService = buildSchedulerService(); + monitoringService = new QueueMonitoringService(queue); + controller = new QueueController(queueService, prioritizationService, schedulerService, monitoringService); + }); + // Monitoring endpoints + describe('getMetrics', () => { + it('returns metrics from monitoring service', async () => { + const m = await controller.getMetrics(); + expect(m.queueName).toBe('default'); + }); }); - - it('shapes stuck-job entries correctly', async () => { - const now = Date.now(); - queue.getActive.mockResolvedValue([ - { - id: 'x', - name: 'backup', - data: {}, - processedOn: now - 400_000, - attemptsMade: 1, - opts: { priority: 3 }, - }, - ]); - const result = await controller.getStuckJobs({ threshold: 300_000 }); - expect(result.data).toHaveLength(1); - expect(result.data[0].id).toBe('x'); - expect(result.data[0].activeForMs).toBeGreaterThan(0); + describe('getHealth', () => { + it('returns health status object', async () => { + const h = await controller.getHealth(); + expect(['healthy', 'warning', 'critical']).toContain(h.status); + expect(Array.isArray(h.issues)).toBe(true); + }); }); - }); - - // Per-job CRUD - describe('getJob', () => { - it('returns metrics for existing job', async () => { - queueService.getJobMetrics.mockResolvedValue({ jobId: '42', status: 'completed' }); - const result = await controller.getJob('42'); - expect(result.jobId).toBe('42'); + describe('getCounts', () => { + it('delegates to queueService.getQueueCounts', async () => { + queueService.getQueueCounts.mockResolvedValue({ waiting: 5 }); + const result = await controller.getCounts(); + expect(result).toEqual({ waiting: 5 }); + }); }); - - it('throws when job not found', async () => { - queueService.getJobMetrics.mockResolvedValue(null); - await expect(controller.getJob('missing')).rejects.toThrow('Job missing not found'); + describe('getMetricsHistory', () => { + it('returns wrapped history array', async () => { + const result = await controller.getMetricsHistory(); + expect(result).toHaveProperty('history'); + expect(Array.isArray(result.history)).toBe(true); + }); }); - }); - - describe('retryJob', () => { - it('retries an existing failed job', async () => { - queueService.getJob.mockResolvedValue({ id: '1' }); - queueService.retryJob.mockResolvedValue(undefined); - const result = await controller.retryJob('1'); - expect(result.message).toBe('Job retry initiated'); - expect(result.jobId).toBe('1'); + describe('getRetryAnalytics', () => { + it('uses 60 as default windowMinutes', async () => { + queue.getFailed.mockResolvedValue([]); + queue.getCompleted.mockResolvedValue([]); + const result = await controller.getRetryAnalytics({ windowMinutes: undefined }); + expect(result.windowMinutes).toBe(60); + }); + it('passes custom windowMinutes', async () => { + queue.getFailed.mockResolvedValue([]); + queue.getCompleted.mockResolvedValue([]); + const result = await controller.getRetryAnalytics({ windowMinutes: 30 }); + expect(result.windowMinutes).toBe(30); + }); }); - - it('throws when job not found', async () => { - queueService.getJob.mockResolvedValue(null); - await expect(controller.retryJob('ghost')).rejects.toThrow(); + // Failed jobs + describe('getFailedJobs', () => { + const now = Date.now(); + const failedJob = { + id: '1', + name: 'email', + data: {}, + failedReason: 'timeout', + attemptsMade: 2, + opts: { attempts: 3 }, + stacktrace: ['at ...'], + timestamp: now - 10000, + processedOn: now - 8000, + finishedOn: now - 5000, + }; + beforeEach(() => { + queue.getFailed.mockResolvedValue([failedJob]); + }); + it('shapes response with data and meta', async () => { + const result = await controller.getFailedJobs({ limit: 50, offset: 0 }); + expect(result.data).toHaveLength(1); + expect(result.data[0].id).toBe('1'); + expect(result.data[0].maxAttempts).toBe(3); + expect(result.data[0].stackTrace).toEqual(['at ...']); + expect(result.meta.limit).toBe(50); + }); + it('null processedOn/finishedOn when absent', async () => { + queue.getFailed.mockResolvedValue([{ ...failedJob, processedOn: null, finishedOn: null }]); + const result = await controller.getFailedJobs({}); + expect(result.data[0].processedOn).toBeNull(); + expect(result.data[0].finishedOn).toBeNull(); + }); }); - }); - - describe('removeJob', () => { - it('removes an existing job', async () => { - queueService.getJob.mockResolvedValue({ id: '1' }); - queueService.removeJob.mockResolvedValue(undefined); - await expect(controller.removeJob('1')).resolves.toBeUndefined(); + describe('retryAllFailedJobs', () => { + it('returns bulk retry summary', async () => { + queue.getFailed.mockResolvedValue([ + { id: '1', retry: jest.fn().mockResolvedValue(undefined) }, + ]); + const result = await controller.retryAllFailedJobs(); + expect(result.requeued).toBe(1); + }); }); - - it('throws when job not found', async () => { - queueService.getJob.mockResolvedValue(null); - await expect(controller.removeJob('ghost')).rejects.toThrow(); + // Stuck jobs + describe('getStuckJobs', () => { + it('uses 300000 as default threshold', async () => { + queue.getActive.mockResolvedValue([]); + const result = await controller.getStuckJobs({}); + expect(result.thresholdMs).toBe(300000); + }); + it('shapes stuck-job entries correctly', async () => { + const now = Date.now(); + queue.getActive.mockResolvedValue([ + { + id: 'x', + name: 'backup', + data: {}, + processedOn: now - 400000, + attemptsMade: 1, + opts: { priority: 3 }, + }, + ]); + const result = await controller.getStuckJobs({ threshold: 300000 }); + expect(result.data).toHaveLength(1); + expect(result.data[0].id).toBe('x'); + expect(result.data[0].activeForMs).toBeGreaterThan(0); + }); }); - }); - - // Job submission - describe('addJob', () => { - it('adds job without priority factors', async () => { - queueService.addJob.mockResolvedValue({ id: 'j1', name: 'email', opts: { priority: 3 } }); - const result = await controller.addJob({ name: 'email', data: {} }); - expect(result.jobId).toBe('j1'); - expect(result.status).toBe('queued'); - expect(prioritizationService.calculatePriority).not.toHaveBeenCalled(); + // Per-job CRUD + describe('getJob', () => { + it('returns metrics for existing job', async () => { + queueService.getJobMetrics.mockResolvedValue({ jobId: '42', status: 'completed' }); + const result = await controller.getJob('42'); + expect(result.jobId).toBe('42'); + }); + it('throws when job not found', async () => { + queueService.getJobMetrics.mockResolvedValue(null); + await expect(controller.getJob('missing')).rejects.toThrow('Job missing not found'); + }); }); - - it('calculates priority when factors are supplied', async () => { - queueService.addJob.mockResolvedValue({ id: 'j2', name: 'payment', opts: { priority: 1 } }); - await controller.addJob({ - name: 'payment', - data: {}, - priorityFactors: { urgency: 'critical' }, - }); - expect(prioritizationService.calculatePriority).toHaveBeenCalledWith({ urgency: 'critical' }); - expect(prioritizationService.getJobOptions).toHaveBeenCalledWith(3); + describe('retryJob', () => { + it('retries an existing failed job', async () => { + queueService.getJob.mockResolvedValue({ id: '1' }); + queueService.retryJob.mockResolvedValue(undefined); + const result = await controller.retryJob('1'); + expect(result.message).toBe('Job retry initiated'); + expect(result.jobId).toBe('1'); + }); + it('throws when job not found', async () => { + queueService.getJob.mockResolvedValue(null); + await expect(controller.retryJob('ghost')).rejects.toThrow(); + }); }); - }); - - describe('addBulkJobs', () => { - it('returns count and jobIds', async () => { - queueService.addBulkJobs.mockResolvedValue([{ id: 'a' }, { id: 'b' }]); - const result = await controller.addBulkJobs({ jobs: [{}, {}] }); - expect(result.count).toBe(2); - expect(result.jobIds).toEqual(['a', 'b']); + describe('removeJob', () => { + it('removes an existing job', async () => { + queueService.getJob.mockResolvedValue({ id: '1' }); + queueService.removeJob.mockResolvedValue(undefined); + await expect(controller.removeJob('1')).resolves.toBeUndefined(); + }); + it('throws when job not found', async () => { + queueService.getJob.mockResolvedValue(null); + await expect(controller.removeJob('ghost')).rejects.toThrow(); + }); }); - }); - - describe('scheduleJob', () => { - it('throws for a past scheduledTime', async () => { - const past = new Date(Date.now() - 10_000).toISOString(); - await expect( - controller.scheduleJob({ name: 'x', data: {}, scheduledTime: past }), - ).rejects.toThrow(); + // Job submission + describe('addJob', () => { + it('adds job without priority factors', async () => { + queueService.addJob.mockResolvedValue({ id: 'j1', name: 'email', opts: { priority: 3 } }); + const result = await controller.addJob({ name: 'email', data: {} }); + expect(result.jobId).toBe('j1'); + expect(result.status).toBe('queued'); + expect(prioritizationService.calculatePriority).not.toHaveBeenCalled(); + }); + it('calculates priority when factors are supplied', async () => { + queueService.addJob.mockResolvedValue({ id: 'j2', name: 'payment', opts: { priority: 1 } }); + await controller.addJob({ + name: 'payment', + data: {}, + priorityFactors: { urgency: 'critical' }, + }); + expect(prioritizationService.calculatePriority).toHaveBeenCalledWith({ urgency: 'critical' }); + expect(prioritizationService.getJobOptions).toHaveBeenCalledWith(3); + }); }); - - it('schedules a future job', async () => { - schedulerService.scheduleJob.mockResolvedValue('sched-1'); - const future = new Date(Date.now() + 60_000).toISOString(); - const result = await controller.scheduleJob({ - name: 'report', - data: {}, - scheduledTime: future, - }); - expect(result.jobId).toBe('sched-1'); - expect(result.status).toBe('scheduled'); + describe('addBulkJobs', () => { + it('returns count and jobIds', async () => { + queueService.addBulkJobs.mockResolvedValue([{ id: 'a' }, { id: 'b' }]); + const result = await controller.addBulkJobs({ jobs: [{}, {}] }); + expect(result.count).toBe(2); + expect(result.jobIds).toEqual(['a', 'b']); + }); }); - }); - - describe('scheduleDelayedJob', () => { - it('delegates to scheduler and returns jobId', async () => { - schedulerService.scheduleDelayedJob.mockResolvedValue('delayed-1'); - const result = await controller.scheduleDelayedJob({ name: 'x', data: {}, delayMs: 5000 }); - expect(result.jobId).toBe('delayed-1'); - expect(result.delayMs).toBe(5000); + describe('scheduleJob', () => { + it('throws for a past scheduledTime', async () => { + const past = new Date(Date.now() - 10000).toISOString(); + await expect(controller.scheduleJob({ name: 'x', data: {}, scheduledTime: past })).rejects.toThrow(); + }); + it('schedules a future job', async () => { + schedulerService.scheduleJob.mockResolvedValue('sched-1'); + const future = new Date(Date.now() + 60000).toISOString(); + const result = await controller.scheduleJob({ + name: 'report', + data: {}, + scheduledTime: future, + }); + expect(result.jobId).toBe('sched-1'); + expect(result.status).toBe('scheduled'); + }); }); - }); - - // Queue controls - describe('pauseQueue', () => { - it('calls queueService.pauseQueue and returns confirmation', async () => { - queueService.pauseQueue.mockResolvedValue(undefined); - const result = await controller.pauseQueue(); - expect(result.message).toBe('Queue paused'); - expect(result.timestamp).toBeDefined(); + describe('scheduleDelayedJob', () => { + it('delegates to scheduler and returns jobId', async () => { + schedulerService.scheduleDelayedJob.mockResolvedValue('delayed-1'); + const result = await controller.scheduleDelayedJob({ name: 'x', data: {}, delayMs: 5000 }); + expect(result.jobId).toBe('delayed-1'); + expect(result.delayMs).toBe(5000); + }); }); - }); - - describe('resumeQueue', () => { - it('calls queueService.resumeQueue and returns confirmation', async () => { - queueService.resumeQueue.mockResolvedValue(undefined); - const result = await controller.resumeQueue(); - expect(result.message).toBe('Queue resumed'); + // Queue controls + describe('pauseQueue', () => { + it('calls queueService.pauseQueue and returns confirmation', async () => { + queueService.pauseQueue.mockResolvedValue(undefined); + const result = await controller.pauseQueue(); + expect(result.message).toBe('Queue paused'); + expect(result.timestamp).toBeDefined(); + }); }); - }); - - describe('cleanQueue', () => { - it('delegates grace and status to queueService', async () => { - queueService.cleanQueue.mockResolvedValue(undefined); - await controller.cleanQueue({ grace: 10_000, status: 'failed' }); - expect(queueService.cleanQueue).toHaveBeenCalledWith(10_000, 'failed'); + describe('resumeQueue', () => { + it('calls queueService.resumeQueue and returns confirmation', async () => { + queueService.resumeQueue.mockResolvedValue(undefined); + const result = await controller.resumeQueue(); + expect(result.message).toBe('Queue resumed'); + }); }); - - it('returns cleaned confirmation message', async () => { - queueService.cleanQueue.mockResolvedValue(undefined); - const result = await controller.cleanQueue({ grace: 5000 }); - expect(result.message).toBe('Queue cleaned'); + describe('cleanQueue', () => { + it('delegates grace and status to queueService', async () => { + queueService.cleanQueue.mockResolvedValue(undefined); + await controller.cleanQueue({ grace: 10000, status: 'failed' }); + expect(queueService.cleanQueue).toHaveBeenCalledWith(10000, 'failed'); + }); + it('returns cleaned confirmation message', async () => { + queueService.cleanQueue.mockResolvedValue(undefined); + const result = await controller.cleanQueue({ grace: 5000 }); + expect(result.message).toBe('Queue cleaned'); + }); }); - }); - - describe('getActiveCronJobs', () => { - it('returns wrapped list of cron names', async () => { - schedulerService.getActiveCronJobs.mockReturnValue(['daily-cleanup', 'hourly-metrics']); - const result = await controller.getActiveCronJobs(); - expect(result.jobs).toEqual(['daily-cleanup', 'hourly-metrics']); + describe('getActiveCronJobs', () => { + it('returns wrapped list of cron names', async () => { + schedulerService.getActiveCronJobs.mockReturnValue(['daily-cleanup', 'hourly-metrics']); + const result = await controller.getActiveCronJobs(); + expect(result.jobs).toEqual(['daily-cleanup', 'hourly-metrics']); + }); }); - }); }); diff --git a/src/queues/retry/retry-logic.service.ts b/src/queues/retry/retry-logic.service.ts index 05db0afa..5162ad57 100644 --- a/src/queues/retry/retry-logic.service.ts +++ b/src/queues/retry/retry-logic.service.ts @@ -1,185 +1,150 @@ import { Injectable, Logger } from '@nestjs/common'; import { RetryStrategy } from '../interfaces/queue.interfaces'; - /** * Retry Logic Service * Implements intelligent retry strategies for failed jobs */ @Injectable() export class RetryLogicService { - private readonly logger = new Logger(RetryLogicService.name); - - /** - * Calculate backoff delay for retry attempts - */ - calculateBackoffDelay(attemptNumber: number, strategy: RetryStrategy): number { - if (strategy.backoffType === 'fixed') { - return strategy.initialDelay; + private readonly logger = new Logger(RetryLogicService.name); + /** + * Calculate backoff delay for retry attempts + */ + calculateBackoffDelay(attemptNumber: number, strategy: RetryStrategy): number { + if (strategy.backoffType === 'fixed') { + return strategy.initialDelay; + } + // Exponential backoff: delay = initialDelay * (multiplier ^ attemptNumber) + const multiplier = strategy.multiplier || 2; + let delay = strategy.initialDelay * Math.pow(multiplier, attemptNumber - 1); + // Cap at maxDelay if specified + if (strategy.maxDelay) { + delay = Math.min(delay, strategy.maxDelay); + } + this.logger.log(`Calculated backoff delay for attempt ${attemptNumber}: ${delay}ms`); + return delay; } - - // Exponential backoff: delay = initialDelay * (multiplier ^ attemptNumber) - const multiplier = strategy.multiplier || 2; - let delay = strategy.initialDelay * Math.pow(multiplier, attemptNumber - 1); - - // Cap at maxDelay if specified - if (strategy.maxDelay) { - delay = Math.min(delay, strategy.maxDelay); + /** + * Determine if job should be retried based on error type + */ + shouldRetry(error: Error, attemptNumber: number, maxAttempts: number): boolean { + // Don't retry if max attempts reached + if (attemptNumber >= maxAttempts) { + this.logger.warn(`Max retry attempts (${maxAttempts}) reached, not retrying`); + return false; + } + // Don't retry for certain error types + const nonRetryableErrors = [ + 'ValidationError', + 'AuthenticationError', + 'AuthorizationError', + 'BadRequestError', + ]; + if (nonRetryableErrors.some((type) => error.name.includes(type))) { + this.logger.warn(`Non-retryable error type: ${error.name}, not retrying`); + return false; + } + // Retry for network, timeout, and temporary errors + const retryableErrors = [ + 'NetworkError', + 'TimeoutError', + 'ServiceUnavailableError', + 'TemporaryError', + 'ECONNREFUSED', + 'ETIMEDOUT', + ]; + const isRetryable = retryableErrors.some((type) => error.name.includes(type) || error.message.includes(type)); + if (isRetryable) { + this.logger.log(`Retryable error detected: ${error.name}, will retry (attempt ${attemptNumber}/${maxAttempts})`); + } + return isRetryable; } - - this.logger.log(`Calculated backoff delay for attempt ${attemptNumber}: ${delay}ms`); - - return delay; - } - - /** - * Determine if job should be retried based on error type - */ - shouldRetry(error: Error, attemptNumber: number, maxAttempts: number): boolean { - // Don't retry if max attempts reached - if (attemptNumber >= maxAttempts) { - this.logger.warn(`Max retry attempts (${maxAttempts}) reached, not retrying`); - return false; + /** + * Get default retry strategy based on job type + */ + getDefaultStrategy(jobType: string): RetryStrategy { + const strategies: Record = { + email: { + maxAttempts: 5, + backoffType: 'exponential', + initialDelay: 2000, + maxDelay: 60000, + multiplier: 2, + }, + payment: { + maxAttempts: 3, + backoffType: 'exponential', + initialDelay: 5000, + maxDelay: 30000, + multiplier: 2, + }, + notification: { + maxAttempts: 4, + backoffType: 'exponential', + initialDelay: 1000, + maxDelay: 20000, + multiplier: 2, + }, + backup: { + maxAttempts: 3, + backoffType: 'fixed', + initialDelay: 10000, + }, + report: { + maxAttempts: 2, + backoffType: 'fixed', + initialDelay: 5000, + }, + default: { + maxAttempts: 3, + backoffType: 'exponential', + initialDelay: 3000, + maxDelay: 30000, + multiplier: 2, + }, + }; + return strategies[jobType] || strategies.default; } - - // Don't retry for certain error types - const nonRetryableErrors = [ - 'ValidationError', - 'AuthenticationError', - 'AuthorizationError', - 'BadRequestError', - ]; - - if (nonRetryableErrors.some((type) => error.name.includes(type))) { - this.logger.warn(`Non-retryable error type: ${error.name}, not retrying`); - return false; + /** + * Calculate jitter to prevent thundering herd + */ + addJitter(delay: number, jitterPercent: number = 0.1): number { + const jitter = delay * jitterPercent; + const randomJitter = Math.random() * jitter * 2 - jitter; + return Math.max(0, Math.floor(delay + randomJitter)); } - - // Retry for network, timeout, and temporary errors - const retryableErrors = [ - 'NetworkError', - 'TimeoutError', - 'ServiceUnavailableError', - 'TemporaryError', - 'ECONNREFUSED', - 'ETIMEDOUT', - ]; - - const isRetryable = retryableErrors.some( - (type) => error.name.includes(type) || error.message.includes(type), - ); - - if (isRetryable) { - this.logger.log( - `Retryable error detected: ${error.name}, will retry (attempt ${attemptNumber}/${maxAttempts})`, - ); + /** + * Get retry options for Bull queue + */ + getRetryOptions(strategy: RetryStrategy, attemptNumber: number = 1) { + const baseDelay = this.calculateBackoffDelay(attemptNumber, strategy); + const delayWithJitter = this.addJitter(baseDelay); + return { + attempts: strategy.maxAttempts, + backoff: { + type: strategy.backoffType, + delay: delayWithJitter, + }, + }; + } + /** + * Log retry attempt + */ + logRetryAttempt(jobId: string, jobName: string, attemptNumber: number, error: Error, nextRetryDelay: number): void { + this.logger.warn(`Job ${jobName} (${jobId}) failed on attempt ${attemptNumber}. ` + + `Error: ${error.message}. ` + + `Next retry in ${nextRetryDelay}ms`); + } + /** + * Handle final failure after all retries exhausted + */ + handleFinalFailure(jobId: string, jobName: string, error: Error, attempts: number): void { + this.logger.error(`Job ${jobName} (${jobId}) permanently failed after ${attempts} attempts. ` + + `Final error: ${error.message}`); + // Here you could: + // - Send alert to monitoring system + // - Store in dead letter queue + // - Notify administrators + // - Log to external service } - - return isRetryable; - } - - /** - * Get default retry strategy based on job type - */ - getDefaultStrategy(jobType: string): RetryStrategy { - const strategies: Record = { - email: { - maxAttempts: 5, - backoffType: 'exponential', - initialDelay: 2000, - maxDelay: 60000, - multiplier: 2, - }, - payment: { - maxAttempts: 3, - backoffType: 'exponential', - initialDelay: 5000, - maxDelay: 30000, - multiplier: 2, - }, - notification: { - maxAttempts: 4, - backoffType: 'exponential', - initialDelay: 1000, - maxDelay: 20000, - multiplier: 2, - }, - backup: { - maxAttempts: 3, - backoffType: 'fixed', - initialDelay: 10000, - }, - report: { - maxAttempts: 2, - backoffType: 'fixed', - initialDelay: 5000, - }, - default: { - maxAttempts: 3, - backoffType: 'exponential', - initialDelay: 3000, - maxDelay: 30000, - multiplier: 2, - }, - }; - - return strategies[jobType] || strategies.default; - } - - /** - * Calculate jitter to prevent thundering herd - */ - addJitter(delay: number, jitterPercent: number = 0.1): number { - const jitter = delay * jitterPercent; - const randomJitter = Math.random() * jitter * 2 - jitter; - return Math.max(0, Math.floor(delay + randomJitter)); - } - - /** - * Get retry options for Bull queue - */ - getRetryOptions(strategy: RetryStrategy, attemptNumber: number = 1) { - const baseDelay = this.calculateBackoffDelay(attemptNumber, strategy); - const delayWithJitter = this.addJitter(baseDelay); - - return { - attempts: strategy.maxAttempts, - backoff: { - type: strategy.backoffType, - delay: delayWithJitter, - }, - }; - } - - /** - * Log retry attempt - */ - logRetryAttempt( - jobId: string, - jobName: string, - attemptNumber: number, - error: Error, - nextRetryDelay: number, - ): void { - this.logger.warn( - `Job ${jobName} (${jobId}) failed on attempt ${attemptNumber}. ` + - `Error: ${error.message}. ` + - `Next retry in ${nextRetryDelay}ms`, - ); - } - - /** - * Handle final failure after all retries exhausted - */ - handleFinalFailure(jobId: string, jobName: string, error: Error, attempts: number): void { - this.logger.error( - `Job ${jobName} (${jobId}) permanently failed after ${attempts} attempts. ` + - `Final error: ${error.message}`, - ); - - // Here you could: - // - Send alert to monitoring system - // - Store in dead letter queue - // - Notify administrators - // - Log to external service - } } diff --git a/src/queues/scheduler/job-scheduler.service.ts b/src/queues/scheduler/job-scheduler.service.ts index 84bb185b..365e778d 100644 --- a/src/queues/scheduler/job-scheduler.service.ts +++ b/src/queues/scheduler/job-scheduler.service.ts @@ -5,242 +5,198 @@ import { QUEUE_NAMES } from '../../common/constants/queue.constants'; import { Cron, CronExpression, SchedulerRegistry } from '@nestjs/schedule'; import { CronJob } from 'cron'; import { JobOptions } from '../interfaces/queue.interfaces'; - /** * Job Scheduler Service * Handles scheduled and recurring jobs with cron support */ @Injectable() export class JobSchedulerService { - private readonly logger = new Logger(JobSchedulerService.name); - - constructor( - @InjectQueue(QUEUE_NAMES.DEFAULT) private readonly defaultQueue: Queue, - private readonly schedulerRegistry: SchedulerRegistry, - ) {} - - /** - * Schedule a job to run at a specific time - */ - async scheduleJob( - name: string, - data: T, - scheduledTime: Date, - options?: JobOptions, - ): Promise { - const delay = scheduledTime.getTime() - Date.now(); - - if (delay < 0) { - throw new Error('Scheduled time must be in the future'); + private readonly logger = new Logger(JobSchedulerService.name); + constructor( + @InjectQueue(QUEUE_NAMES.DEFAULT) + private readonly defaultQueue: Queue, private readonly schedulerRegistry: SchedulerRegistry) { } + /** + * Schedule a job to run at a specific time + */ + async scheduleJob(name: string, data: T, scheduledTime: Date, options?: JobOptions): Promise { + const delay = scheduledTime.getTime() - Date.now(); + if (delay < 0) { + throw new Error('Scheduled time must be in the future'); + } + const job = await this.defaultQueue.add(name, data, { + ...options, + delay, + }); + this.logger.log(`Job ${name} scheduled for ${scheduledTime.toISOString()} (ID: ${job.id})`); + return job.id.toString(); } - - const job = await this.defaultQueue.add(name, data, { - ...options, - delay, - }); - - this.logger.log(`Job ${name} scheduled for ${scheduledTime.toISOString()} (ID: ${job.id})`); - - return job.id.toString(); - } - - /** - * Schedule a recurring job with cron expression - */ - scheduleRecurringJob(name: string, cronExpression: string, callback: () => Promise): void { - try { - const job = new CronJob(cronExpression, async () => { - this.logger.log(`Executing recurring job: ${name}`); + /** + * Schedule a recurring job with cron expression + */ + scheduleRecurringJob(name: string, cronExpression: string, callback: () => Promise): void { try { - await callback(); - } catch (error) { - this.logger.error(`Error in recurring job ${name}:`, error); + const job = new CronJob(cronExpression, async () => { + this.logger.log(`Executing recurring job: ${name}`); + try { + await callback(); + } + catch (error) { + this.logger.error(`Error in recurring job ${name}:`, error); + } + }); + this.schedulerRegistry.addCronJob(name, job); + job.start(); + this.logger.log(`Recurring job ${name} scheduled with cron: ${cronExpression}`); + } + catch (error) { + this.logger.error(`Failed to schedule recurring job ${name}:`, error); + throw error; } - }); - - this.schedulerRegistry.addCronJob(name, job); - job.start(); - - this.logger.log(`Recurring job ${name} scheduled with cron: ${cronExpression}`); - } catch (error) { - this.logger.error(`Failed to schedule recurring job ${name}:`, error); - throw error; } - } - - /** - * Remove a scheduled recurring job - */ - removeRecurringJob(name: string): void { - try { - const job = this.schedulerRegistry.getCronJob(name); - job.stop(); - this.schedulerRegistry.deleteCronJob(name); - this.logger.log(`Recurring job ${name} removed`); - } catch (error) { - this.logger.error(`Failed to remove recurring job ${name}:`, error); + /** + * Remove a scheduled recurring job + */ + removeRecurringJob(name: string): void { + try { + const job = this.schedulerRegistry.getCronJob(name); + job.stop(); + this.schedulerRegistry.deleteCronJob(name); + this.logger.log(`Recurring job ${name} removed`); + } + catch (error) { + this.logger.error(`Failed to remove recurring job ${name}:`, error); + } } - } - - /** - * Schedule a job to run after a delay - */ - async scheduleDelayedJob( - name: string, - data: T, - delayMs: number, - options?: JobOptions, - ): Promise { - const job = await this.defaultQueue.add(name, data, { - ...options, - delay: delayMs, - }); - - this.logger.log(`Job ${name} scheduled with ${delayMs}ms delay (ID: ${job.id})`); - - return job.id.toString(); - } - - /** - * Schedule a batch of jobs - */ - async scheduleBatchJobs( - jobs: Array<{ - name: string; - data: T; - scheduledTime: Date; - options?: JobOptions; - }>, - ): Promise { - const jobIds: string[] = []; - - for (const job of jobs) { - const id = await this.scheduleJob(job.name, job.data, job.scheduledTime, job.options); - jobIds.push(id); + /** + * Schedule a job to run after a delay + */ + async scheduleDelayedJob(name: string, data: T, delayMs: number, options?: JobOptions): Promise { + const job = await this.defaultQueue.add(name, data, { + ...options, + delay: delayMs, + }); + this.logger.log(`Job ${name} scheduled with ${delayMs}ms delay (ID: ${job.id})`); + return job.id.toString(); } - - this.logger.log(`Scheduled ${jobIds.length} jobs in batch`); - return jobIds; - } - - /** - * Get all scheduled jobs - */ - async getScheduledJobs(): Promise { - const delayed = await this.defaultQueue.getDelayed(); - return delayed.map((job) => ({ - id: job.id, - name: job.name, - data: job.data, - scheduledFor: new Date(job.timestamp + (job.opts.delay || 0)), - priority: job.opts.priority, - })); - } - - /** - * Cancel a scheduled job - */ - async cancelScheduledJob(jobId: string): Promise { - const job = await this.defaultQueue.getJob(jobId); - if (job) { - await job.remove(); - this.logger.log(`Scheduled job ${jobId} cancelled`); + /** + * Schedule a batch of jobs + */ + async scheduleBatchJobs(jobs: Array<{ + name: string; + data: T; + scheduledTime: Date; + options?: JobOptions; + }>): Promise { + const jobIds: string[] = []; + for (const job of jobs) { + const id = await this.scheduleJob(job.name, job.data, job.scheduledTime, job.options); + jobIds.push(id); + } + this.logger.log(`Scheduled ${jobIds.length} jobs in batch`); + return jobIds; } - } - - /** - * Reschedule a job - */ - async rescheduleJob(jobId: string, newScheduledTime: Date): Promise { - const job = await this.defaultQueue.getJob(jobId); - if (!job) { - throw new Error(`Job ${jobId} not found`); + /** + * Get all scheduled jobs + */ + async getScheduledJobs(): Promise { + const delayed = await this.defaultQueue.getDelayed(); + return delayed.map((job) => ({ + id: job.id, + name: job.name, + data: job.data, + scheduledFor: new Date(job.timestamp + (job.opts.delay || 0)), + priority: job.opts.priority, + })); } - - // Remove old job - await job.remove(); - - // Create new job with same data - const newJobId = await this.scheduleJob(job.name, job.data, newScheduledTime, job.opts as any); - - this.logger.log( - `Job ${jobId} rescheduled to ${newScheduledTime.toISOString()} (new ID: ${newJobId})`, - ); - - return newJobId; - } - - /** - * Example: Daily cleanup job - */ - @Cron(CronExpression.EVERY_DAY_AT_MIDNIGHT) - async dailyCleanup(): Promise { - this.logger.log('Running daily cleanup job'); - try { - // Clean completed jobs older than 24 hours - await this.defaultQueue.clean(24 * 60 * 60 * 1000, 'completed'); - // Clean failed jobs older than 7 days - await this.defaultQueue.clean(7 * 24 * 60 * 60 * 1000, 'failed'); - this.logger.log('Daily cleanup completed'); - } catch (error) { - this.logger.error('Error during daily cleanup:', error); + /** + * Cancel a scheduled job + */ + async cancelScheduledJob(jobId: string): Promise { + const job = await this.defaultQueue.getJob(jobId); + if (job) { + await job.remove(); + this.logger.log(`Scheduled job ${jobId} cancelled`); + } } - } - - /** - * Example: Hourly metrics collection - */ - @Cron(CronExpression.EVERY_HOUR) - async hourlyMetrics(): Promise { - this.logger.log('Collecting hourly metrics'); - try { - const counts = { - waiting: await this.defaultQueue.getWaitingCount(), - active: await this.defaultQueue.getActiveCount(), - completed: await this.defaultQueue.getCompletedCount(), - failed: await this.defaultQueue.getFailedCount(), - }; - this.logger.log('Hourly metrics:', counts); - } catch (error) { - this.logger.error('Error collecting hourly metrics:', error); + /** + * Reschedule a job + */ + async rescheduleJob(jobId: string, newScheduledTime: Date): Promise { + const job = await this.defaultQueue.getJob(jobId); + if (!job) { + throw new Error(`Job ${jobId} not found`); + } + // Remove old job + await job.remove(); + // Create new job with same data + const newJobId = await this.scheduleJob(job.name, job.data, newScheduledTime, job.opts as unknown); + this.logger.log(`Job ${jobId} rescheduled to ${newScheduledTime.toISOString()} (new ID: ${newJobId})`); + return newJobId; } - } - - /** - * Schedule job with retry on specific days/times - */ - async scheduleWithBusinessHours( - name: string, - data: T, - options?: JobOptions, - ): Promise { - const now = new Date(); - const scheduledTime = new Date(now); - - // If outside business hours (9 AM - 5 PM), schedule for next business day at 9 AM - const hour = now.getHours(); - if (hour < 9 || hour >= 17) { - scheduledTime.setDate(scheduledTime.getDate() + 1); - scheduledTime.setHours(9, 0, 0, 0); + /** + * Example: Daily cleanup job + */ + @Cron(CronExpression.EVERY_DAY_AT_MIDNIGHT) + async dailyCleanup(): Promise { + this.logger.log('Running daily cleanup job'); + try { + // Clean completed jobs older than 24 hours + await this.defaultQueue.clean(24 * 60 * 60 * 1000, 'completed'); + // Clean failed jobs older than 7 days + await this.defaultQueue.clean(7 * 24 * 60 * 60 * 1000, 'failed'); + this.logger.log('Daily cleanup completed'); + } + catch (error) { + this.logger.error('Error during daily cleanup:', error); + } + } + /** + * Example: Hourly metrics collection + */ + @Cron(CronExpression.EVERY_HOUR) + async hourlyMetrics(): Promise { + this.logger.log('Collecting hourly metrics'); + try { + const counts = { + waiting: await this.defaultQueue.getWaitingCount(), + active: await this.defaultQueue.getActiveCount(), + completed: await this.defaultQueue.getCompletedCount(), + failed: await this.defaultQueue.getFailedCount(), + }; + this.logger.log('Hourly metrics:', counts); + } + catch (error) { + this.logger.error('Error collecting hourly metrics:', error); + } + } + /** + * Schedule job with retry on specific days/times + */ + async scheduleWithBusinessHours(name: string, data: T, options?: JobOptions): Promise { + const now = new Date(); + const scheduledTime = new Date(now); + // If outside business hours (9 AM - 5 PM), schedule for next business day at 9 AM + const hour = now.getHours(); + if (hour < 9 || hour >= 17) { + scheduledTime.setDate(scheduledTime.getDate() + 1); + scheduledTime.setHours(9, 0, 0, 0); + } + // Skip weekends + const day = scheduledTime.getDay(); + if (day === 0) { + // Sunday -> Monday + scheduledTime.setDate(scheduledTime.getDate() + 1); + } + else if (day === 6) { + // Saturday -> Monday + scheduledTime.setDate(scheduledTime.getDate() + 2); + } + return this.scheduleJob(name, data, scheduledTime, options); } - - // Skip weekends - const day = scheduledTime.getDay(); - if (day === 0) { - // Sunday -> Monday - scheduledTime.setDate(scheduledTime.getDate() + 1); - } else if (day === 6) { - // Saturday -> Monday - scheduledTime.setDate(scheduledTime.getDate() + 2); + /** + * Get all active cron jobs + */ + getActiveCronJobs(): string[] { + return Array.from(this.schedulerRegistry.getCronJobs().keys()); } - - return this.scheduleJob(name, data, scheduledTime, options); - } - - /** - * Get all active cron jobs - */ - getActiveCronJobs(): string[] { - return Array.from(this.schedulerRegistry.getCronJobs().keys()); - } } diff --git a/src/rate-limiting/dto/create-rate-limiting.dto.ts b/src/rate-limiting/dto/create-rate-limiting.dto.ts index fb2a97f7..5953eb20 100644 --- a/src/rate-limiting/dto/create-rate-limiting.dto.ts +++ b/src/rate-limiting/dto/create-rate-limiting.dto.ts @@ -1,97 +1,79 @@ -import { - IsString, - IsNotEmpty, - IsOptional, - IsNumber, - IsEnum, - Min, - Max, - IsObject, -} from 'class-validator'; +import { IsString, IsNotEmpty, IsOptional, IsNumber, IsEnum, Min, Max, IsObject, } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; - export enum RateLimitType { - IP = 'ip', - USER = 'user', - ENDPOINT = 'endpoint', - GLOBAL = 'global', + IP = 'ip', + USER = 'user', + ENDPOINT = 'endpoint', + GLOBAL = 'global' } - export class CreateRateLimitingDto { - @ApiProperty({ - description: 'Name of the rate limit rule', - example: 'api-login-limit', - }) - @IsString({ message: 'Name must be a string' }) - @IsNotEmpty({ message: 'Name is required' }) - name: string; - - @ApiProperty({ - description: 'Type of rate limiting', - enum: RateLimitType, - example: RateLimitType.USER, - }) - @IsEnum(RateLimitType, { message: 'Type must be a valid rate limit type' }) - type: RateLimitType; - - @ApiProperty({ - description: 'Maximum number of requests allowed', - example: 100, - minimum: 1, - maximum: 1000000, - }) - @IsNumber({}, { message: 'Limit must be a number' }) - @Min(1, { message: 'Limit must be at least 1' }) - @Max(1000000, { message: 'Limit cannot exceed 1,000,000' }) - limit: number; - - @ApiProperty({ - description: 'Time window in seconds', - example: 3600, - minimum: 1, - maximum: 86400, - }) - @IsNumber({}, { message: 'Window must be a number' }) - @Min(1, { message: 'Window must be at least 1 second' }) - @Max(86400, { message: 'Window cannot exceed 24 hours' }) - windowSeconds: number; - - @ApiPropertyOptional({ - description: 'Specific endpoint to limit', - example: '/api/auth/login', - }) - @IsOptional() - @IsString({ message: 'Endpoint must be a string' }) - endpoint?: string; - - @ApiPropertyOptional({ - description: 'Priority of this rule (higher = more important)', - example: 1, - minimum: 1, - maximum: 100, - }) - @IsOptional() - @IsNumber({}, { message: 'Priority must be a number' }) - @Min(1, { message: 'Priority must be at least 1' }) - @Max(100, { message: 'Priority cannot exceed 100' }) - priority?: number; - - @ApiPropertyOptional({ - description: 'Whether this rule is enabled', - default: true, - }) - @IsOptional() - @IsEnum([true, false], { message: 'Enabled must be a boolean' }) - enabled?: boolean; - - @ApiPropertyOptional({ - description: 'Additional metadata for the rule', - example: { - description: 'Limit login attempts per user', - tags: ['auth', 'security'], - }, - }) - @IsOptional() - @IsObject({ message: 'Metadata must be an object' }) - metadata?: Record; + @ApiProperty({ + description: 'Name of the rate limit rule', + example: 'api-login-limit', + }) + @IsString({ message: 'Name must be a string' }) + @IsNotEmpty({ message: 'Name is required' }) + name: string; + @ApiProperty({ + description: 'Type of rate limiting', + enum: RateLimitType, + example: RateLimitType.USER, + }) + @IsEnum(RateLimitType, { message: 'Type must be a valid rate limit type' }) + type: RateLimitType; + @ApiProperty({ + description: 'Maximum number of requests allowed', + example: 100, + minimum: 1, + maximum: 1000000, + }) + @IsNumber({}, { message: 'Limit must be a number' }) + @Min(1, { message: 'Limit must be at least 1' }) + @Max(1000000, { message: 'Limit cannot exceed 1,000,000' }) + limit: number; + @ApiProperty({ + description: 'Time window in seconds', + example: 3600, + minimum: 1, + maximum: 86400, + }) + @IsNumber({}, { message: 'Window must be a number' }) + @Min(1, { message: 'Window must be at least 1 second' }) + @Max(86400, { message: 'Window cannot exceed 24 hours' }) + windowSeconds: number; + @ApiPropertyOptional({ + description: 'Specific endpoint to limit', + example: '/api/auth/login', + }) + @IsOptional() + @IsString({ message: 'Endpoint must be a string' }) + endpoint?: string; + @ApiPropertyOptional({ + description: 'Priority of this rule (higher = more important)', + example: 1, + minimum: 1, + maximum: 100, + }) + @IsOptional() + @IsNumber({}, { message: 'Priority must be a number' }) + @Min(1, { message: 'Priority must be at least 1' }) + @Max(100, { message: 'Priority cannot exceed 100' }) + priority?: number; + @ApiPropertyOptional({ + description: 'Whether this rule is enabled', + default: true, + }) + @IsOptional() + @IsEnum([true, false], { message: 'Enabled must be a boolean' }) + enabled?: boolean; + @ApiPropertyOptional({ + description: 'Additional metadata for the rule', + example: { + description: 'Limit login attempts per user', + tags: ['auth', 'security'], + }, + }) + @IsOptional() + @IsObject({ message: 'Metadata must be an object' }) + metadata?: Record; } diff --git a/src/rate-limiting/dto/update-rate-limiting.dto.ts b/src/rate-limiting/dto/update-rate-limiting.dto.ts index e9d61f08..c30b4bb1 100644 --- a/src/rate-limiting/dto/update-rate-limiting.dto.ts +++ b/src/rate-limiting/dto/update-rate-limiting.dto.ts @@ -1,4 +1,4 @@ import { PartialType } from '@nestjs/swagger'; import { CreateRateLimitingDto } from './create-rate-limiting.dto'; - -export class UpdateRateLimitingDto extends PartialType(CreateRateLimitingDto) {} +export class UpdateRateLimitingDto extends PartialType(CreateRateLimitingDto) { +} diff --git a/src/rate-limiting/entities/rate-limiting.entity.ts b/src/rate-limiting/entities/rate-limiting.entity.ts index b5bd6b83..022d0d3b 100644 --- a/src/rate-limiting/entities/rate-limiting.entity.ts +++ b/src/rate-limiting/entities/rate-limiting.entity.ts @@ -1 +1,2 @@ -export class RateLimiting {} +export class RateLimiting { +} diff --git a/src/rate-limiting/rate-limiting.controller.spec.ts b/src/rate-limiting/rate-limiting.controller.spec.ts index 1040ae69..d77b3616 100644 --- a/src/rate-limiting/rate-limiting.controller.spec.ts +++ b/src/rate-limiting/rate-limiting.controller.spec.ts @@ -2,28 +2,24 @@ import { Test, TestingModule } from '@nestjs/testing'; import { RateLimitingController } from './rate-limiting.controller'; import { RateLimitingService } from './rate-limiting.service'; import { ThrottlingService } from './services/throttling.service'; - describe('RateLimitingController', () => { - let controller: RateLimitingController; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - controllers: [RateLimitingController], - providers: [ - RateLimitingService, - { - provide: ThrottlingService, - useValue: { - checkLimit: jest.fn(), - }, - }, - ], - }).compile(); - - controller = module.get(RateLimitingController); - }); - - it('should be defined', () => { - expect(controller).toBeDefined(); - }); + let controller: RateLimitingController; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + controllers: [RateLimitingController], + providers: [ + RateLimitingService, + { + provide: ThrottlingService, + useValue: { + checkLimit: jest.fn(), + }, + }, + ], + }).compile(); + controller = module.get(RateLimitingController); + }); + it('should be defined', () => { + expect(controller).toBeDefined(); + }); }); diff --git a/src/rate-limiting/rate-limiting.controller.ts b/src/rate-limiting/rate-limiting.controller.ts index fcee8a74..ae304b51 100644 --- a/src/rate-limiting/rate-limiting.controller.ts +++ b/src/rate-limiting/rate-limiting.controller.ts @@ -4,34 +4,38 @@ import { RateLimitGuard } from './services/limit-guard/guard'; import { CreateRateLimitingDto } from './dto/create-rate-limiting.dto'; import { UpdateRateLimitingDto } from './dto/update-rate-limiting.dto'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; - @Controller('rate-limiting') export class RateLimitingController { - constructor(private readonly rateLimitingService: RateLimitingService) {} - - @Post() - create(@Body() createRateLimitingDto: CreateRateLimitingDto) { - return this.rateLimitingService.create(createRateLimitingDto); - } - - @UseGuards(JwtAuthGuard, RateLimitGuard) - @Get() - findAll() { - return this.rateLimitingService.findAll(); - } - - @Get(':id') - findOne(@Param('id') id: string) { - return this.rateLimitingService.findOne(+id); - } - - @Patch(':id') - update(@Param('id') id: string, @Body() updateRateLimitingDto: UpdateRateLimitingDto) { - return this.rateLimitingService.update(+id, updateRateLimitingDto); - } - - @Delete(':id') - remove(@Param('id') id: string) { - return this.rateLimitingService.remove(+id); - } + constructor(private readonly rateLimitingService: RateLimitingService) { } + @Post() + create( + @Body() + createRateLimitingDto: CreateRateLimitingDto) { + return this.rateLimitingService.create(createRateLimitingDto); + } + @UseGuards(JwtAuthGuard, RateLimitGuard) + @Get() + findAll() { + return this.rateLimitingService.findAll(); + } + @Get(':id') + findOne( + @Param('id') + id: string) { + return this.rateLimitingService.findOne(+id); + } + @Patch(':id') + update( + @Param('id') + id: string, + @Body() + updateRateLimitingDto: UpdateRateLimitingDto) { + return this.rateLimitingService.update(+id, updateRateLimitingDto); + } + @Delete(':id') + remove( + @Param('id') + id: string) { + return this.rateLimitingService.remove(+id); + } } diff --git a/src/rate-limiting/rate-limiting.service.ts b/src/rate-limiting/rate-limiting.service.ts index 3e63873f..f10fe0a6 100644 --- a/src/rate-limiting/rate-limiting.service.ts +++ b/src/rate-limiting/rate-limiting.service.ts @@ -3,27 +3,25 @@ import { ThrottlingService } from './services/throttling.service'; import { UserTier } from './services/quota.service'; import { CreateRateLimitingDto } from './dto/create-rate-limiting.dto'; import { UpdateRateLimitingDto } from './dto/update-rate-limiting.dto'; - @Injectable() export class RateLimitingService { - create(_createRateLimitingDto: CreateRateLimitingDto) { - throw new Error('Method not implemented.'); - } - findAll() { - throw new Error('Method not implemented.'); - } - findOne(_arg0: number) { - throw new Error('Method not implemented.'); - } - update(_arg0: number, _updateRateLimitingDto: UpdateRateLimitingDto) { - throw new Error('Method not implemented.'); - } - remove(_arg0: number) { - throw new Error('Method not implemented.'); - } - constructor(private readonly throttlingService: ThrottlingService) {} - - async protect(userId: string, tier: UserTier, endpoint: string) { - await this.throttlingService.handleRequest(userId, tier, endpoint); - } + create(_createRateLimitingDto: CreateRateLimitingDto) { + throw new Error('Method not implemented.'); + } + findAll() { + throw new Error('Method not implemented.'); + } + findOne(_arg0: number) { + throw new Error('Method not implemented.'); + } + update(_arg0: number, _updateRateLimitingDto: UpdateRateLimitingDto) { + throw new Error('Method not implemented.'); + } + remove(_arg0: number) { + throw new Error('Method not implemented.'); + } + constructor(private readonly throttlingService: ThrottlingService) { } + async protect(userId: string, tier: UserTier, endpoint: string) { + await this.throttlingService.handleRequest(userId, tier, endpoint); + } } diff --git a/src/rate-limiting/services/adaptive-rate-limiting.service.ts b/src/rate-limiting/services/adaptive-rate-limiting.service.ts index f896655e..3513f305 100644 --- a/src/rate-limiting/services/adaptive-rate-limiting.service.ts +++ b/src/rate-limiting/services/adaptive-rate-limiting.service.ts @@ -1,21 +1,19 @@ import { Injectable } from '@nestjs/common'; import * as os from 'os'; - @Injectable() export class AdaptiveRateLimitingService { - getSystemLoadFactor(): number { - const load = os.loadavg()[0]; // 1-minute average - const cpuCount = os.cpus().length; - - const loadPercentage = load / cpuCount; - - if (loadPercentage > 0.9) return 0.5; // reduce limits by 50% - if (loadPercentage > 0.7) return 0.7; - return 1; - } - - adjustLimit(baseLimit: number): number { - const factor = this.getSystemLoadFactor(); - return Math.floor(baseLimit * factor); - } + getSystemLoadFactor(): number { + const load = os.loadavg()[0]; // 1-minute average + const cpuCount = os.cpus().length; + const loadPercentage = load / cpuCount; + if (loadPercentage > 0.9) + return 0.5; // reduce limits by 50% + if (loadPercentage > 0.7) + return 0.7; + return 1; + } + adjustLimit(baseLimit: number): number { + const factor = this.getSystemLoadFactor(); + return Math.floor(baseLimit * factor); + } } diff --git a/src/rate-limiting/services/distrubutes.service.ts b/src/rate-limiting/services/distrubutes.service.ts index c6d13431..d6f05050 100644 --- a/src/rate-limiting/services/distrubutes.service.ts +++ b/src/rate-limiting/services/distrubutes.service.ts @@ -1,33 +1,26 @@ import { Injectable, ForbiddenException } from '@nestjs/common'; import Redis from 'ioredis'; - @Injectable() export class DistributedLimiterService { - private redis: Redis; - - constructor() { - this.redis = new Redis(process.env.REDIS_URL); - this.redis.on('error', () => { - // Prevent unhandled error events during Redis outages. - }); - } - - async slidingWindowCheck(key: string, limit: number, windowInSeconds: number): Promise { - const now = Date.now(); - const windowStart = now - windowInSeconds * 1000; - - const pipeline = this.redis.pipeline(); - - pipeline.zremrangebyscore(key, 0, windowStart); - pipeline.zadd(key, now, `${now}`); - pipeline.zcard(key); - pipeline.expire(key, windowInSeconds); - - const results = await pipeline.exec(); - const requestCount = results?.[2]?.[1] as number; - - if (requestCount > limit) { - throw new ForbiddenException('Rate limit exceeded'); + private redis: Redis; + constructor() { + this.redis = new Redis(process.env.REDIS_URL); + this.redis.on('error', () => { + // Prevent unhandled error events during Redis outages. + }); + } + async slidingWindowCheck(key: string, limit: number, windowInSeconds: number): Promise { + const now = Date.now(); + const windowStart = now - windowInSeconds * 1000; + const pipeline = this.redis.pipeline(); + pipeline.zremrangebyscore(key, 0, windowStart); + pipeline.zadd(key, now, `${now}`); + pipeline.zcard(key); + pipeline.expire(key, windowInSeconds); + const results = await pipeline.exec(); + const requestCount = results?.[2]?.[1] as number; + if (requestCount > limit) { + throw new ForbiddenException('Rate limit exceeded'); + } } - } } diff --git a/src/rate-limiting/services/limit-guard/guard.ts b/src/rate-limiting/services/limit-guard/guard.ts index 7bea66f6..db2bc89c 100644 --- a/src/rate-limiting/services/limit-guard/guard.ts +++ b/src/rate-limiting/services/limit-guard/guard.ts @@ -1,18 +1,13 @@ import { CanActivate, ExecutionContext, Injectable } from '@nestjs/common'; import { RateLimitingService } from '../../rate-limiting.service'; - @Injectable() export class RateLimitGuard implements CanActivate { - constructor(private rateLimiting: RateLimitingService) {} - - async canActivate(context: ExecutionContext): Promise { - const req = context.switchToHttp().getRequest(); - - const user = req.user; - const endpoint = req.route.path; - - await this.rateLimiting.protect(user.id, user.tier, endpoint); - - return true; - } + constructor(private rateLimiting: RateLimitingService) { } + async canActivate(context: ExecutionContext): Promise { + const req = context.switchToHttp().getRequest(); + const user = req.user; + const endpoint = req.route.path; + await this.rateLimiting.protect(user.id, user.tier, endpoint); + return true; + } } diff --git a/src/rate-limiting/services/quota.service.ts b/src/rate-limiting/services/quota.service.ts index 0a5778e8..bae4b2ca 100644 --- a/src/rate-limiting/services/quota.service.ts +++ b/src/rate-limiting/services/quota.service.ts @@ -1,23 +1,21 @@ import { Injectable } from '@nestjs/common'; - export enum UserTier { - FREE = 'FREE', - PRO = 'PRO', - PREMIUM = 'PREMIUM', + FREE = 'FREE', + PRO = 'PRO', + PREMIUM = 'PREMIUM' } - @Injectable() export class QuotaManagementService { - getQuotaForTier(tier: UserTier) { - switch (tier) { - case UserTier.FREE: - return { limit: 100, window: 60 }; - case UserTier.PRO: - return { limit: 500, window: 60 }; - case UserTier.PREMIUM: - return { limit: Infinity, window: 60 }; - default: - return { limit: 50, window: 60 }; + getQuotaForTier(tier: UserTier) { + switch (tier) { + case UserTier.FREE: + return { limit: 100, window: 60 }; + case UserTier.PRO: + return { limit: 500, window: 60 }; + case UserTier.PREMIUM: + return { limit: Infinity, window: 60 }; + default: + return { limit: 50, window: 60 }; + } } - } } diff --git a/src/rate-limiting/services/rate-limiting.module.ts b/src/rate-limiting/services/rate-limiting.module.ts index 058e9330..9f62d2a2 100644 --- a/src/rate-limiting/services/rate-limiting.module.ts +++ b/src/rate-limiting/services/rate-limiting.module.ts @@ -4,15 +4,15 @@ import { ThrottlingService } from './throttling.service'; import { QuotaManagementService } from './quota.service'; import { AdaptiveRateLimitingService } from './adaptive-rate-limiting.service'; import { DistributedLimiterService } from './distrubutes.service'; - @Module({ - providers: [ - RateLimitingService, - ThrottlingService, - QuotaManagementService, - AdaptiveRateLimitingService, - DistributedLimiterService, - ], - exports: [RateLimitingService], + providers: [ + RateLimitingService, + ThrottlingService, + QuotaManagementService, + AdaptiveRateLimitingService, + DistributedLimiterService, + ], + exports: [RateLimitingService], }) -export class RateLimitingModule {} +export class RateLimitingModule { +} diff --git a/src/rate-limiting/services/throttling.service.ts b/src/rate-limiting/services/throttling.service.ts index 4aad34ec..74c4062f 100644 --- a/src/rate-limiting/services/throttling.service.ts +++ b/src/rate-limiting/services/throttling.service.ts @@ -2,25 +2,16 @@ import { Injectable } from '@nestjs/common'; import { DistributedLimiterService } from './distrubutes.service'; import { QuotaManagementService, UserTier } from './quota.service'; import { AdaptiveRateLimitingService } from './adaptive-rate-limiting.service'; - @Injectable() export class ThrottlingService { - constructor( - private readonly distributedLimiter: DistributedLimiterService, - private readonly quotaService: QuotaManagementService, - private readonly adaptiveService: AdaptiveRateLimitingService, - ) {} - - async handleRequest(userId: string, tier: UserTier, endpoint: string) { - if (tier === UserTier.PREMIUM) { - return; // bypass + constructor(private readonly distributedLimiter: DistributedLimiterService, private readonly quotaService: QuotaManagementService, private readonly adaptiveService: AdaptiveRateLimitingService) { } + async handleRequest(userId: string, tier: UserTier, endpoint: string) { + if (tier === UserTier.PREMIUM) { + return; // bypass + } + const { limit, window } = this.quotaService.getQuotaForTier(tier); + const adjustedLimit = this.adaptiveService.adjustLimit(limit); + const key = `rate:${userId}:${endpoint}`; + await this.distributedLimiter.slidingWindowCheck(key, adjustedLimit, window); } - - const { limit, window } = this.quotaService.getQuotaForTier(tier); - const adjustedLimit = this.adaptiveService.adjustLimit(limit); - - const key = `rate:${userId}:${endpoint}`; - - await this.distributedLimiter.slidingWindowCheck(key, adjustedLimit, window); - } } diff --git a/src/search/autocomplete/autocomplete.service.ts b/src/search/autocomplete/autocomplete.service.ts index 675a5cde..1c8ccfec 100644 --- a/src/search/autocomplete/autocomplete.service.ts +++ b/src/search/autocomplete/autocomplete.service.ts @@ -1,34 +1,30 @@ import { Injectable } from '@nestjs/common'; import { ElasticsearchService } from '@nestjs/elasticsearch'; import { COURSES_INDEX } from '../search.service'; - @Injectable() export class AutoCompleteService { - constructor(private readonly elasticsearchService: ElasticsearchService) {} - - async getSuggestions(query: string): Promise { - const sanitizedQuery = (query ?? '').trim().slice(0, 100); - if (!sanitizedQuery) { - return []; + constructor(private readonly elasticsearchService: ElasticsearchService) { } + async getSuggestions(query: string): Promise { + const sanitizedQuery = (query ?? '').trim().slice(0, 100); + if (!sanitizedQuery) { + return []; + } + const result = await this.elasticsearchService.search({ + index: COURSES_INDEX, + _source: false, + timeout: '1000ms', + suggest: { + title_suggest: { + text: sanitizedQuery, + completion: { + field: 'title.suggest', + skip_duplicates: true, + size: 10, + }, + }, + }, + }); + const options = result.suggest?.title_suggest?.[0]?.options ?? []; + return Array.isArray(options) ? options.map((option: unknown) => option.text as string) : []; } - - const result = await this.elasticsearchService.search({ - index: COURSES_INDEX, - _source: false, - timeout: '1000ms', - suggest: { - title_suggest: { - text: sanitizedQuery, - completion: { - field: 'title.suggest', - skip_duplicates: true, - size: 10, - }, - }, - }, - }); - - const options = result.suggest?.title_suggest?.[0]?.options ?? []; - return Array.isArray(options) ? options.map((option: any) => option.text as string) : []; - } } diff --git a/src/search/filters/search-filters.service.ts b/src/search/filters/search-filters.service.ts index 968c75d6..5799d395 100644 --- a/src/search/filters/search-filters.service.ts +++ b/src/search/filters/search-filters.service.ts @@ -1,56 +1,52 @@ import { Injectable } from '@nestjs/common'; import { ElasticsearchService } from '@nestjs/elasticsearch'; import { COURSES_INDEX } from '../search.service'; - @Injectable() export class SearchFiltersService { - constructor(private readonly elasticsearchService: ElasticsearchService) {} - - async getFilters(): Promise { - const result = await this.elasticsearchService.search({ - index: COURSES_INDEX, - size: 0, - _source: false, - timeout: '1500ms', - aggs: { - categories: { terms: { field: 'category', size: 50 } }, - levels: { terms: { field: 'level', size: 10 } }, - languages: { terms: { field: 'language', size: 30 } }, - price_stats: { stats: { field: 'price' } }, - price_ranges: { - range: { - field: 'price', - ranges: [ - { key: 'free', to: 1 }, - { key: 'under_50', from: 1, to: 50 }, - { key: '50_to_100', from: 50, to: 100 }, - { key: '100_to_200', from: 100, to: 200 }, - { key: 'over_200', from: 200 }, - ], - }, - }, - rating_ranges: { - range: { - field: 'rating', - ranges: [ - { key: '4_and_up', from: 4 }, - { key: '3_and_up', from: 3 }, - { key: '2_and_up', from: 2 }, - ], - }, - }, - }, - }); - - const aggs = result.aggregations as any; - - return { - categories: aggs?.categories?.buckets ?? [], - levels: aggs?.levels?.buckets ?? [], - languages: aggs?.languages?.buckets ?? [], - priceStats: aggs?.price_stats ?? {}, - priceRanges: aggs?.price_ranges?.buckets ?? [], - ratingRanges: aggs?.rating_ranges?.buckets ?? [], - }; - } + constructor(private readonly elasticsearchService: ElasticsearchService) { } + async getFilters(): Promise { + const result = await this.elasticsearchService.search({ + index: COURSES_INDEX, + size: 0, + _source: false, + timeout: '1500ms', + aggs: { + categories: { terms: { field: 'category', size: 50 } }, + levels: { terms: { field: 'level', size: 10 } }, + languages: { terms: { field: 'language', size: 30 } }, + price_stats: { stats: { field: 'price' } }, + price_ranges: { + range: { + field: 'price', + ranges: [ + { key: 'free', to: 1 }, + { key: 'under_50', from: 1, to: 50 }, + { key: '50_to_100', from: 50, to: 100 }, + { key: '100_to_200', from: 100, to: 200 }, + { key: 'over_200', from: 200 }, + ], + }, + }, + rating_ranges: { + range: { + field: 'rating', + ranges: [ + { key: '4_and_up', from: 4 }, + { key: '3_and_up', from: 3 }, + { key: '2_and_up', from: 2 }, + ], + }, + }, + }, + }); + const aggs = result.aggregations as unknown; + return { + categories: aggs?.categories?.buckets ?? [], + levels: aggs?.levels?.buckets ?? [], + languages: aggs?.languages?.buckets ?? [], + priceStats: aggs?.price_stats ?? {}, + priceRanges: aggs?.price_ranges?.buckets ?? [], + ratingRanges: aggs?.rating_ranges?.buckets ?? [], + }; + } } diff --git a/src/search/indexing/indexing.service.ts b/src/search/indexing/indexing.service.ts index 9216e3a2..68831d3b 100644 --- a/src/search/indexing/indexing.service.ts +++ b/src/search/indexing/indexing.service.ts @@ -1,260 +1,232 @@ import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; import { ElasticsearchService } from '@nestjs/elasticsearch'; import { COURSES_INDEX, SEARCH_ANALYTICS_INDEX } from '../search.service'; - @Injectable() export class IndexingService implements OnModuleInit { - private readonly logger = new Logger(IndexingService.name); - private readonly reindexOnBoot = process.env.SEARCH_REINDEX_ON_BOOT === 'true'; - - constructor(private readonly elasticsearchService: ElasticsearchService) {} - - async onModuleInit(): Promise { - await this.ensureIndices(); - } - - // ── Generic document operations ───────────────────────────────────────────── - - async indexDocument(index: string, id: string, document: any): Promise { - return this.elasticsearchService.index({ index, id, document }); - } - - async updateDocument(index: string, id: string, document: any): Promise { - return this.elasticsearchService.update({ index, id, doc: document }); - } - - async deleteDocument(index: string, id: string): Promise { - return this.elasticsearchService.delete({ index, id }); - } - - async createIndex(index: string, mapping: any): Promise { - return this.elasticsearchService.indices.create({ index, mappings: mapping }); - } - - async bulkIndex(documents: any[]): Promise { - const operations = documents.flatMap((doc) => [ - { index: { _index: COURSES_INDEX, _id: String(doc.id) } }, - doc, - ]); - return this.elasticsearchService.bulk({ operations }); - } - - // ── Course-specific sync operations ───────────────────────────────────────── - - async syncCourse(course: Record): Promise { - const { id, ...fields } = course; - await this.elasticsearchService.index({ - index: COURSES_INDEX, - id: String(id), - document: { ...fields, updatedAt: new Date().toISOString() }, - }); - } - - async updateCourse(id: string, fields: Record): Promise { - await this.elasticsearchService.update({ - index: COURSES_INDEX, - id, - doc: { ...fields, updatedAt: new Date().toISOString() }, - }); - } - - async removeCourse(id: string): Promise { - await this.elasticsearchService.delete({ index: COURSES_INDEX, id }); - } - - async reindexAll(courses: Array>): Promise { - if (courses.length === 0) return; - - const operations = courses.flatMap((course) => { - const { id, ...fields } = course; - return [ - { index: { _index: COURSES_INDEX, _id: String(id) } }, - { ...fields, updatedAt: new Date().toISOString() }, - ]; - }); - - const result = await this.elasticsearchService.bulk({ operations }); - if (result.errors) { - const failed = result.items.filter((item: any) => item.index?.error); - this.logger.error(`Bulk reindex had ${failed.length} failures`); + private readonly logger = new Logger(IndexingService.name); + private readonly reindexOnBoot = process.env.SEARCH_REINDEX_ON_BOOT === 'true'; + constructor(private readonly elasticsearchService: ElasticsearchService) { } + async onModuleInit(): Promise { + await this.ensureIndices(); } - return result; - } - - // ── Index bootstrap ────────────────────────────────────────────────────────── - - private async ensureIndices(): Promise { - await Promise.all([ - this.createCoursesIndex(this.reindexOnBoot), - this.createSearchAnalyticsIndex(), - ]); - } - - async createCoursesIndex(forceReindex = false): Promise { - const exists = await this.elasticsearchService.indices.exists({ index: COURSES_INDEX }); - - if (exists) { - await this.ensureExistingCoursesIndexSettings(); - if (forceReindex) { - await this.reindexCoursesIndexWithCurrentMapping(); - } - return; + // ── Generic document operations ───────────────────────────────────────────── + async indexDocument(index: string, id: string, document: unknown): Promise { + return this.elasticsearchService.index({ index, id, document }); } - - this.logger.log(`Creating index: ${COURSES_INDEX}`); - return this.elasticsearchService.indices.create({ - index: COURSES_INDEX, - ...this.getCoursesIndexDefinition(), - }); - } - - private async ensureExistingCoursesIndexSettings(): Promise { - try { - await this.elasticsearchService.indices.putSettings({ - index: COURSES_INDEX, - settings: { - refresh_interval: '30s', - }, - }); - } catch (error) { - this.logger.warn(`Failed to update settings for ${COURSES_INDEX}: ${String(error)}`); + async updateDocument(index: string, id: string, document: unknown): Promise { + return this.elasticsearchService.update({ index, id, doc: document }); } - } - - private async reindexCoursesIndexWithCurrentMapping(): Promise { - const tempIndex = `${COURSES_INDEX}_tmp_${Date.now()}`; - this.logger.log( - `SEARCH_REINDEX_ON_BOOT enabled, reindexing ${COURSES_INDEX} using temporary index ${tempIndex}`, - ); - - try { - await this.elasticsearchService.indices.create({ - index: tempIndex, - ...this.getCoursesIndexDefinition(), - }); - - const sourceCount = await this.elasticsearchService.count({ index: COURSES_INDEX }); - if (sourceCount.count > 0) { - await this.elasticsearchService.reindex({ - wait_for_completion: true, - refresh: true, - source: { index: COURSES_INDEX }, - dest: { index: tempIndex }, + async deleteDocument(index: string, id: string): Promise { + return this.elasticsearchService.delete({ index, id }); + } + async createIndex(index: string, mapping: unknown): Promise { + return this.elasticsearchService.indices.create({ index, mappings: mapping }); + } + async bulkIndex(documents: unknown[]): Promise { + const operations = documents.flatMap((doc) => [ + { index: { _index: COURSES_INDEX, _id: String(doc.id) } }, + doc, + ]); + return this.elasticsearchService.bulk({ operations }); + } + // ── Course-specific sync operations ───────────────────────────────────────── + async syncCourse(course: Record): Promise { + const { id, ...fields } = course; + await this.elasticsearchService.index({ + index: COURSES_INDEX, + id: String(id), + document: { ...fields, updatedAt: new Date().toISOString() }, }); - } - - await this.elasticsearchService.indices.delete({ index: COURSES_INDEX }); - await this.elasticsearchService.indices.create({ - index: COURSES_INDEX, - ...this.getCoursesIndexDefinition(), - }); - - const tempCount = await this.elasticsearchService.count({ index: tempIndex }); - if (tempCount.count > 0) { - await this.elasticsearchService.reindex({ - wait_for_completion: true, - refresh: true, - source: { index: tempIndex }, - dest: { index: COURSES_INDEX }, + } + async updateCourse(id: string, fields: Record): Promise { + await this.elasticsearchService.update({ + index: COURSES_INDEX, + id, + doc: { ...fields, updatedAt: new Date().toISOString() }, }); - } - - this.logger.log(`Reindex completed successfully for ${COURSES_INDEX}`); - } catch (error) { - this.logger.error(`Failed to reindex ${COURSES_INDEX}: ${String(error)}`); - throw error; - } finally { - const tempExists = await this.elasticsearchService.indices.exists({ index: tempIndex }); - if (tempExists) { - await this.elasticsearchService.indices.delete({ index: tempIndex }); - } } - } - - private getCoursesIndexDefinition() { - return { - settings: { - number_of_shards: 1, - number_of_replicas: 1, - refresh_interval: '30s', - analysis: { - analyzer: { - english_custom: { - type: 'custom', - tokenizer: 'standard', - filter: ['lowercase', 'english_stop', 'english_stemmer'], + async removeCourse(id: string): Promise { + await this.elasticsearchService.delete({ index: COURSES_INDEX, id }); + } + async reindexAll(courses: Array>): Promise { + if (courses.length === 0) + return; + const operations = courses.flatMap((course) => { + const { id, ...fields } = course; + return [ + { index: { _index: COURSES_INDEX, _id: String(id) } }, + { ...fields, updatedAt: new Date().toISOString() }, + ]; + }); + const result = await this.elasticsearchService.bulk({ operations }); + if (result.errors) { + const failed = result.items.filter((item: unknown) => item.index?.error); + this.logger.error(`Bulk reindex had ${failed.length} failures`); + } + return result; + } + // ── Index bootstrap ────────────────────────────────────────────────────────── + private async ensureIndices(): Promise { + await Promise.all([ + this.createCoursesIndex(this.reindexOnBoot), + this.createSearchAnalyticsIndex(), + ]); + } + async createCoursesIndex(forceReindex = false): Promise { + const exists = await this.elasticsearchService.indices.exists({ index: COURSES_INDEX }); + if (exists) { + await this.ensureExistingCoursesIndexSettings(); + if (forceReindex) { + await this.reindexCoursesIndexWithCurrentMapping(); + } + return; + } + this.logger.log(`Creating index: ${COURSES_INDEX}`); + return this.elasticsearchService.indices.create({ + index: COURSES_INDEX, + ...this.getCoursesIndexDefinition(), + }); + } + private async ensureExistingCoursesIndexSettings(): Promise { + try { + await this.elasticsearchService.indices.putSettings({ + index: COURSES_INDEX, + settings: { + refresh_interval: '30s', + }, + }); + } + catch (error) { + this.logger.warn(`Failed to update settings for ${COURSES_INDEX}: ${String(error)}`); + } + } + private async reindexCoursesIndexWithCurrentMapping(): Promise { + const tempIndex = `${COURSES_INDEX}_tmp_${Date.now()}`; + this.logger.log(`SEARCH_REINDEX_ON_BOOT enabled, reindexing ${COURSES_INDEX} using temporary index ${tempIndex}`); + try { + await this.elasticsearchService.indices.create({ + index: tempIndex, + ...this.getCoursesIndexDefinition(), + }); + const sourceCount = await this.elasticsearchService.count({ index: COURSES_INDEX }); + if (sourceCount.count > 0) { + await this.elasticsearchService.reindex({ + wait_for_completion: true, + refresh: true, + source: { index: COURSES_INDEX }, + dest: { index: tempIndex }, + }); + } + await this.elasticsearchService.indices.delete({ index: COURSES_INDEX }); + await this.elasticsearchService.indices.create({ + index: COURSES_INDEX, + ...this.getCoursesIndexDefinition(), + }); + const tempCount = await this.elasticsearchService.count({ index: tempIndex }); + if (tempCount.count > 0) { + await this.elasticsearchService.reindex({ + wait_for_completion: true, + refresh: true, + source: { index: tempIndex }, + dest: { index: COURSES_INDEX }, + }); + } + this.logger.log(`Reindex completed successfully for ${COURSES_INDEX}`); + } + catch (error) { + this.logger.error(`Failed to reindex ${COURSES_INDEX}: ${String(error)}`); + throw error; + } + finally { + const tempExists = await this.elasticsearchService.indices.exists({ index: tempIndex }); + if (tempExists) { + await this.elasticsearchService.indices.delete({ index: tempIndex }); + } + } + } + private getCoursesIndexDefinition() { + return { + settings: { + number_of_shards: 1, + number_of_replicas: 1, + refresh_interval: '30s', + analysis: { + analyzer: { + english_custom: { + type: 'custom', + tokenizer: 'standard', + filter: ['lowercase', 'english_stop', 'english_stemmer'], + }, + }, + normalizer: { + lowercase_normalizer: { + type: 'custom', + filter: ['lowercase'], + }, + }, + filter: { + english_stop: { type: 'stop', stopwords: '_english_' }, + english_stemmer: { type: 'stemmer', language: 'english' }, + }, + }, + }, + mappings: { + properties: { + id: { type: 'keyword' }, + title: { + type: 'text', + analyzer: 'english_custom', + fields: { + keyword: { type: 'keyword' }, + suggest: { type: 'completion' }, + search: { type: 'search_as_you_type' }, + }, + }, + description: { type: 'text', analyzer: 'english_custom' }, + content: { type: 'text', analyzer: 'english_custom' }, + tags: { type: 'keyword', normalizer: 'lowercase_normalizer' }, + category: { type: 'keyword', normalizer: 'lowercase_normalizer' }, + level: { type: 'keyword', normalizer: 'lowercase_normalizer' }, + language: { type: 'keyword', normalizer: 'lowercase_normalizer' }, + price: { type: 'float' }, + rating: { type: 'float' }, + views: { type: 'integer' }, + enrollments: { type: 'integer' }, + duration: { type: 'integer' }, + instructorId: { type: 'keyword' }, + instructorName: { + type: 'text', + fields: { keyword: { type: 'keyword' } }, + }, + status: { type: 'keyword' }, + createdAt: { type: 'date' }, + updatedAt: { type: 'date' }, + }, }, - }, - normalizer: { - lowercase_normalizer: { - type: 'custom', - filter: ['lowercase'], + }; + } + async createSearchAnalyticsIndex(): Promise { + const exists = await this.elasticsearchService.indices.exists({ + index: SEARCH_ANALYTICS_INDEX, + }); + if (exists) + return; + this.logger.log(`Creating index: ${SEARCH_ANALYTICS_INDEX}`); + return this.elasticsearchService.indices.create({ + index: SEARCH_ANALYTICS_INDEX, + settings: { + number_of_shards: 1, + number_of_replicas: 1, }, - }, - filter: { - english_stop: { type: 'stop', stopwords: '_english_' }, - english_stemmer: { type: 'stemmer', language: 'english' }, - }, - }, - }, - mappings: { - properties: { - id: { type: 'keyword' }, - title: { - type: 'text', - analyzer: 'english_custom', - fields: { - keyword: { type: 'keyword' }, - suggest: { type: 'completion' }, - search: { type: 'search_as_you_type' }, + mappings: { + properties: { + query: { type: 'keyword' }, + resultsCount: { type: 'integer' }, + filters: { type: 'object', enabled: false }, + sort: { type: 'keyword' }, + timestamp: { type: 'date' }, + }, }, - }, - description: { type: 'text', analyzer: 'english_custom' }, - content: { type: 'text', analyzer: 'english_custom' }, - tags: { type: 'keyword', normalizer: 'lowercase_normalizer' }, - category: { type: 'keyword', normalizer: 'lowercase_normalizer' }, - level: { type: 'keyword', normalizer: 'lowercase_normalizer' }, - language: { type: 'keyword', normalizer: 'lowercase_normalizer' }, - price: { type: 'float' }, - rating: { type: 'float' }, - views: { type: 'integer' }, - enrollments: { type: 'integer' }, - duration: { type: 'integer' }, - instructorId: { type: 'keyword' }, - instructorName: { - type: 'text', - fields: { keyword: { type: 'keyword' } }, - }, - status: { type: 'keyword' }, - createdAt: { type: 'date' }, - updatedAt: { type: 'date' }, - }, - }, - }; - } - - async createSearchAnalyticsIndex(): Promise { - const exists = await this.elasticsearchService.indices.exists({ - index: SEARCH_ANALYTICS_INDEX, - }); - if (exists) return; - - this.logger.log(`Creating index: ${SEARCH_ANALYTICS_INDEX}`); - return this.elasticsearchService.indices.create({ - index: SEARCH_ANALYTICS_INDEX, - settings: { - number_of_shards: 1, - number_of_replicas: 1, - }, - mappings: { - properties: { - query: { type: 'keyword' }, - resultsCount: { type: 'integer' }, - filters: { type: 'object', enabled: false }, - sort: { type: 'keyword' }, - timestamp: { type: 'date' }, - }, - }, - }); - } + }); + } } diff --git a/src/search/search.controller.spec.ts b/src/search/search.controller.spec.ts index 327961d0..f6151c0e 100644 --- a/src/search/search.controller.spec.ts +++ b/src/search/search.controller.spec.ts @@ -1,56 +1,36 @@ import { BadRequestException } from '@nestjs/common'; import { SearchController } from './search.controller'; import { SearchService } from './search.service'; - describe('SearchController', () => { - let controller: SearchController; - let searchService: jest.Mocked; - - beforeEach(() => { - searchService = { - performSearch: jest.fn(), - getAutoComplete: jest.fn(), - getAvailableFilters: jest.fn(), - getSearchAnalytics: jest.fn(), - } as unknown as jest.Mocked; - - controller = new SearchController(searchService); - }); - - it('parses filters and pagination before calling search service', async () => { - searchService.performSearch.mockResolvedValueOnce({ - results: [], - total: 0, - page: 2, - limit: 10, - facets: { categories: [], levels: [], priceRanges: [] }, + let controller: SearchController; + let searchService: jest.Mocked; + beforeEach(() => { + searchService = { + performSearch: jest.fn(), + getAutoComplete: jest.fn(), + getAvailableFilters: jest.fn(), + getSearchAnalytics: jest.fn(), + } as unknown as jest.Mocked; + controller = new SearchController(searchService); + }); + it('parses filters and pagination before calling search service', async () => { + searchService.performSearch.mockResolvedValueOnce({ + results: [], + total: 0, + page: 2, + limit: 10, + facets: { categories: [], levels: [], priceRanges: [] }, + }); + await controller.search('nestjs', '{"category":"backend"}', 'relevance', '2', '10'); + expect(searchService.performSearch).toHaveBeenCalledWith('nestjs', { category: 'backend' }, 'relevance', { page: 2, limit: 10 }); + }); + it('throws BadRequestException for invalid JSON filters', async () => { + await expect(controller.search('nestjs', '{bad-json}', 'relevance', '1', '20')).rejects.toThrow(BadRequestException); + }); + it('throws BadRequestException for invalid page', async () => { + await expect(controller.search('nestjs', '{}', 'relevance', '0', '20')).rejects.toThrow('page must be a positive integer'); + }); + it('throws BadRequestException for invalid limit', async () => { + await expect(controller.search('nestjs', '{}', 'relevance', '1', '100')).rejects.toThrow('limit must be an integer between 1 and 50'); }); - - await controller.search('nestjs', '{"category":"backend"}', 'relevance', '2', '10'); - - expect(searchService.performSearch).toHaveBeenCalledWith( - 'nestjs', - { category: 'backend' }, - 'relevance', - { page: 2, limit: 10 }, - ); - }); - - it('throws BadRequestException for invalid JSON filters', async () => { - await expect(controller.search('nestjs', '{bad-json}', 'relevance', '1', '20')).rejects.toThrow( - BadRequestException, - ); - }); - - it('throws BadRequestException for invalid page', async () => { - await expect(controller.search('nestjs', '{}', 'relevance', '0', '20')).rejects.toThrow( - 'page must be a positive integer', - ); - }); - - it('throws BadRequestException for invalid limit', async () => { - await expect(controller.search('nestjs', '{}', 'relevance', '1', '100')).rejects.toThrow( - 'limit must be an integer between 1 and 50', - ); - }); }); diff --git a/src/search/search.controller.ts b/src/search/search.controller.ts index 8b0e0d1b..d959ad2c 100644 --- a/src/search/search.controller.ts +++ b/src/search/search.controller.ts @@ -1,59 +1,59 @@ import { BadRequestException, Controller, Get, Query } from '@nestjs/common'; import { Throttle } from '@nestjs/throttler'; import { SearchService } from './search.service'; - @Throttle({ default: { limit: 30, ttl: 60000 } }) @Controller('search') export class SearchController { - constructor(private readonly searchService: SearchService) {} - - @Get() - async search( - @Query('q') query: string, - @Query('filters') filters?: string, - @Query('sort') sort?: string, - @Query('page') page?: string, - @Query('limit') limit?: string, - ): Promise { - let parsedFilters: Record = {}; - if (filters) { - try { - parsedFilters = JSON.parse(filters); - } catch { - throw new BadRequestException('filters must be valid JSON'); - } + constructor(private readonly searchService: SearchService) { } + @Get() + async search( + @Query('q') + query: string, + @Query('filters') + filters?: string, + @Query('sort') + sort?: string, + @Query('page') + page?: string, + @Query('limit') + limit?: string): Promise { + let parsedFilters: Record = {}; + if (filters) { + try { + parsedFilters = JSON.parse(filters); + } + catch { + throw new BadRequestException('filters must be valid JSON'); + } + } + const parsedPage = page ? Number.parseInt(page, 10) : 1; + const parsedLimit = limit ? Number.parseInt(limit, 10) : 20; + if (!Number.isInteger(parsedPage) || parsedPage < 1) { + throw new BadRequestException('page must be a positive integer'); + } + if (!Number.isInteger(parsedLimit) || parsedLimit < 1 || parsedLimit > 50) { + throw new BadRequestException('limit must be an integer between 1 and 50'); + } + return this.searchService.performSearch(query, parsedFilters, sort, { + page: parsedPage, + limit: parsedLimit, + }); } - - const parsedPage = page ? Number.parseInt(page, 10) : 1; - const parsedLimit = limit ? Number.parseInt(limit, 10) : 20; - - if (!Number.isInteger(parsedPage) || parsedPage < 1) { - throw new BadRequestException('page must be a positive integer'); + @Get('autocomplete') + async autocomplete( + @Query('q') + query: string): Promise { + return this.searchService.getAutoComplete(query); } - - if (!Number.isInteger(parsedLimit) || parsedLimit < 1 || parsedLimit > 50) { - throw new BadRequestException('limit must be an integer between 1 and 50'); + @Get('filters') + async getFilters(): Promise { + return this.searchService.getAvailableFilters(); + } + @Get('analytics') + async getAnalytics( + @Query('days') + days?: string): Promise { + const parsedDays = days ? parseInt(days, 10) : 7; + return this.searchService.getSearchAnalytics(parsedDays); } - - return this.searchService.performSearch(query, parsedFilters, sort, { - page: parsedPage, - limit: parsedLimit, - }); - } - - @Get('autocomplete') - async autocomplete(@Query('q') query: string): Promise { - return this.searchService.getAutoComplete(query); - } - - @Get('filters') - async getFilters(): Promise { - return this.searchService.getAvailableFilters(); - } - - @Get('analytics') - async getAnalytics(@Query('days') days?: string): Promise { - const parsedDays = days ? parseInt(days, 10) : 7; - return this.searchService.getSearchAnalytics(parsedDays); - } } diff --git a/src/search/search.module.ts b/src/search/search.module.ts index aae13640..fc796602 100644 --- a/src/search/search.module.ts +++ b/src/search/search.module.ts @@ -7,18 +7,18 @@ import { IndexingService } from './indexing/indexing.service'; import { AutoCompleteService } from './autocomplete/autocomplete.service'; import { SearchFiltersService } from './filters/search-filters.service'; import { createElasticsearchConfig } from '../config/elasticsearch.config'; - @Module({ - imports: [ - ConfigModule, - ElasticsearchModule.registerAsync({ - imports: [ConfigModule], - inject: [ConfigService], - useFactory: createElasticsearchConfig, - }), - ], - controllers: [SearchController], - providers: [SearchService, IndexingService, AutoCompleteService, SearchFiltersService], - exports: [SearchService, IndexingService, AutoCompleteService, SearchFiltersService], + imports: [ + ConfigModule, + ElasticsearchModule.registerAsync({ + imports: [ConfigModule], + inject: [ConfigService], + useFactory: createElasticsearchConfig, + }), + ], + controllers: [SearchController], + providers: [SearchService, IndexingService, AutoCompleteService, SearchFiltersService], + exports: [SearchService, IndexingService, AutoCompleteService, SearchFiltersService], }) -export class SearchModule {} +export class SearchModule { +} diff --git a/src/search/search.service.spec.ts b/src/search/search.service.spec.ts index 76b9c636..20dddcce 100644 --- a/src/search/search.service.spec.ts +++ b/src/search/search.service.spec.ts @@ -3,126 +3,93 @@ import { ElasticsearchService } from '@nestjs/elasticsearch'; import { AutoCompleteService } from './autocomplete/autocomplete.service'; import { SearchFiltersService } from './filters/search-filters.service'; import { CachingService } from '../caching/caching.service'; - describe('SearchService', () => { - let service: SearchService; - let elasticsearchService: jest.Mocked; - let cachingService: jest.Mocked; - let autoCompleteService: jest.Mocked; - let searchFiltersService: jest.Mocked; - - beforeEach(() => { - elasticsearchService = { - search: jest.fn(), - index: jest.fn(), - } as unknown as jest.Mocked; - elasticsearchService.index.mockResolvedValue({} as any); - - cachingService = { - getOrSet: jest.fn(async (_key: string, factory: () => Promise) => factory()), - } as unknown as jest.Mocked; - - autoCompleteService = { - getSuggestions: jest.fn(), - } as unknown as jest.Mocked; - - searchFiltersService = { - getFilters: jest.fn(), - } as unknown as jest.Mocked; - - service = new SearchService( - elasticsearchService, - autoCompleteService, - searchFiltersService, - cachingService, - ); - }); - - it('sanitizes input, normalizes filters, and applies bounded pagination', async () => { - elasticsearchService.search.mockResolvedValueOnce({ - hits: { - total: { value: 1 }, - hits: [ - { - _id: '1', - _score: 10, - _source: { title: 'NestJS Fundamentals' }, - highlight: { title: ['NestJS'] }, - }, - ], - }, - aggregations: { - categories: { buckets: [] }, - levels: { buckets: [] }, - price_ranges: { buckets: [] }, - }, - } as any); - - const result = await service.performSearch( - ' NestJS ', - { category: ' Backend ', language: ['EN', 'en', ''], price: { gte: 0, invalid: 'x' } }, - 'relevance', - { page: 0, limit: 100 }, - ); - - expect(result.page).toBe(1); - expect(result.limit).toBe(50); - - expect(elasticsearchService.search).toHaveBeenCalledTimes(1); - const esQuery = elasticsearchService.search.mock.calls[0][0] as any; - - expect(esQuery.from).toBe(0); - expect(esQuery.size).toBe(50); - expect(esQuery.timeout).toBe('1500ms'); - expect(esQuery.highlight).toBeDefined(); - expect(esQuery.query.function_score.query.bool.filter).toEqual( - expect.arrayContaining([ - { term: { category: 'backend' } }, - { terms: { language: ['en'] } }, - { range: { price: { gte: 0 } } }, - ]), - ); - }); - - it('supports filter-only search without highlight when query is empty', async () => { - elasticsearchService.search.mockResolvedValueOnce({ - hits: { - total: { value: 0 }, - hits: [], - }, - aggregations: { - categories: { buckets: [] }, - levels: { buckets: [] }, - price_ranges: { buckets: [] }, - }, - } as any); - - await service.performSearch(' ', { level: 'Beginner' }, 'relevance', { - page: 2, - limit: 10, + let service: SearchService; + let elasticsearchService: jest.Mocked; + let cachingService: jest.Mocked; + let autoCompleteService: jest.Mocked; + let searchFiltersService: jest.Mocked; + beforeEach(() => { + elasticsearchService = { + search: jest.fn(), + index: jest.fn(), + } as unknown as jest.Mocked; + elasticsearchService.index.mockResolvedValue({} as unknown); + cachingService = { + getOrSet: jest.fn(async (_key: string, factory: () => Promise) => factory()), + } as unknown as jest.Mocked; + autoCompleteService = { + getSuggestions: jest.fn(), + } as unknown as jest.Mocked; + searchFiltersService = { + getFilters: jest.fn(), + } as unknown as jest.Mocked; + service = new SearchService(elasticsearchService, autoCompleteService, searchFiltersService, cachingService); }); - - const esQuery = elasticsearchService.search.mock.calls[0][0] as any; - expect(esQuery.from).toBe(10); - expect(esQuery.size).toBe(10); - expect(esQuery.highlight).toBeUndefined(); - expect(esQuery.query.function_score.query).toEqual({ - bool: { - filter: [{ term: { level: 'beginner' } }], - }, + it('sanitizes input, normalizes filters, and applies bounded pagination', async () => { + elasticsearchService.search.mockResolvedValueOnce({ + hits: { + total: { value: 1 }, + hits: [ + { + _id: '1', + _score: 10, + _source: { title: 'NestJS Fundamentals' }, + highlight: { title: ['NestJS'] }, + }, + ], + }, + aggregations: { + categories: { buckets: [] }, + levels: { buckets: [] }, + price_ranges: { buckets: [] }, + }, + } as unknown); + const result = await service.performSearch(' NestJS ', { category: ' Backend ', language: ['EN', 'en', ''], price: { gte: 0, invalid: 'x' } }, 'relevance', { page: 0, limit: 100 }); + expect(result.page).toBe(1); + expect(result.limit).toBe(50); + expect(elasticsearchService.search).toHaveBeenCalledTimes(1); + const esQuery = elasticsearchService.search.mock.calls[0][0] as unknown; + expect(esQuery.from).toBe(0); + expect(esQuery.size).toBe(50); + expect(esQuery.timeout).toBe('1500ms'); + expect(esQuery.highlight).toBeDefined(); + expect(esQuery.query.function_score.query.bool.filter).toEqual(expect.arrayContaining([ + { term: { category: 'backend' } }, + { terms: { language: ['en'] } }, + { range: { price: { gte: 0 } } }, + ])); + }); + it('supports filter-only search without highlight when query is empty', async () => { + elasticsearchService.search.mockResolvedValueOnce({ + hits: { + total: { value: 0 }, + hits: [], + }, + aggregations: { + categories: { buckets: [] }, + levels: { buckets: [] }, + price_ranges: { buckets: [] }, + }, + } as unknown); + await service.performSearch(' ', { level: 'Beginner' }, 'relevance', { + page: 2, + limit: 10, + }); + const esQuery = elasticsearchService.search.mock.calls[0][0] as unknown; + expect(esQuery.from).toBe(10); + expect(esQuery.size).toBe(10); + expect(esQuery.highlight).toBeUndefined(); + expect(esQuery.query.function_score.query).toEqual({ + bool: { + filter: [{ term: { level: 'beginner' } }], + }, + }); + }); + it('normalizes autocomplete input before cache and downstream call', async () => { + autoCompleteService.getSuggestions.mockResolvedValueOnce(['nestjs']); + await service.getAutoComplete(' nestjs '); + expect(cachingService.getOrSet).toHaveBeenCalledWith(expect.stringContaining('autocomplete:nestjs'), expect.any(Function), expect.any(Number)); + expect(autoCompleteService.getSuggestions).toHaveBeenCalledWith('nestjs'); }); - }); - - it('normalizes autocomplete input before cache and downstream call', async () => { - autoCompleteService.getSuggestions.mockResolvedValueOnce(['nestjs']); - - await service.getAutoComplete(' nestjs '); - - expect(cachingService.getOrSet).toHaveBeenCalledWith( - expect.stringContaining('autocomplete:nestjs'), - expect.any(Function), - expect.any(Number), - ); - expect(autoCompleteService.getSuggestions).toHaveBeenCalledWith('nestjs'); - }); }); diff --git a/src/search/search.service.ts b/src/search/search.service.ts index 7585e10b..d5bd5d3a 100644 --- a/src/search/search.service.ts +++ b/src/search/search.service.ts @@ -4,437 +4,388 @@ import { AutoCompleteService } from './autocomplete/autocomplete.service'; import { SearchFiltersService } from './filters/search-filters.service'; import { CachingService } from '../caching/caching.service'; import { CACHE_TTL, CACHE_PREFIXES } from '../caching/caching.constants'; - export const COURSES_INDEX = 'courses'; export const SEARCH_ANALYTICS_INDEX = 'search_analytics'; const SEARCH_SOURCE_FIELDS = [ - 'id', - 'title', - 'description', - 'tags', - 'category', - 'level', - 'language', - 'price', - 'rating', - 'views', - 'enrollments', - 'duration', - 'instructorId', - 'instructorName', - 'status', - 'createdAt', - 'updatedAt', + 'id', + 'title', + 'description', + 'tags', + 'category', + 'level', + 'language', + 'price', + 'rating', + 'views', + 'enrollments', + 'duration', + 'instructorId', + 'instructorName', + 'status', + 'createdAt', + 'updatedAt', ]; - type SearchOptions = { - page?: number; - limit?: number; + page?: number; + limit?: number; }; - type SearchFilters = { - category?: string | string[]; - level?: string | string[]; - language?: string | string[]; - instructorId?: string; - price?: { gte?: number; lte?: number; gt?: number; lt?: number }; + category?: string | string[]; + level?: string | string[]; + language?: string | string[]; + instructorId?: string; + price?: { + gte?: number; + lte?: number; + gt?: number; + lt?: number; + }; }; - @Injectable() export class SearchService { - private readonly logger = new Logger(SearchService.name); - - constructor( - private readonly elasticsearchService: ElasticsearchService, - private readonly autoCompleteService: AutoCompleteService, - private readonly searchFiltersService: SearchFiltersService, - private readonly cachingService: CachingService, - ) {} - - async performSearch(query: string, filters: any, sort?: string, options: SearchOptions = {}) { - const sanitizedQuery = (query ?? '').trim().slice(0, 200); - const page = Math.max(1, options.page ?? 1); - const limit = Math.min(50, Math.max(1, options.limit ?? 20)); - const from = (page - 1) * limit; - const normalizedFilters = this.normalizeFilters(filters); - const cacheKey = `${CACHE_PREFIXES.SEARCH}:${this.hashSearchParams( - sanitizedQuery, - normalizedFilters, - sort, - page, - limit, - )}`; - const hasQuery = sanitizedQuery.length > 0; - - return this.cachingService.getOrSet( - cacheKey, - async () => { + private readonly logger = new Logger(SearchService.name); + constructor(private readonly elasticsearchService: ElasticsearchService, private readonly autoCompleteService: AutoCompleteService, private readonly searchFiltersService: SearchFiltersService, private readonly cachingService: CachingService) { } + async performSearch(query: string, filters: unknown, sort?: string, options: SearchOptions = {}) { + const sanitizedQuery = (query ?? '').trim().slice(0, 200); + const page = Math.max(1, options.page ?? 1); + const limit = Math.min(50, Math.max(1, options.limit ?? 20)); + const from = (page - 1) * limit; + const normalizedFilters = this.normalizeFilters(filters); + const cacheKey = `${CACHE_PREFIXES.SEARCH}:${this.hashSearchParams(sanitizedQuery, normalizedFilters, sort, page, limit)}`; + const hasQuery = sanitizedQuery.length > 0; + return this.cachingService.getOrSet(cacheKey, async () => { + const result = await this.elasticsearchService.search({ + index: COURSES_INDEX, + from, + size: limit, + timeout: '1500ms', + track_total_hits: 10000, + _source: SEARCH_SOURCE_FIELDS, + query: { + function_score: { + query: this.buildSearchQuery(sanitizedQuery, normalizedFilters, hasQuery), + functions: [ + { + field_value_factor: { + field: 'views', + factor: 0.1, + modifier: 'log1p' as const, + missing: 1, + }, + }, + { + field_value_factor: { + field: 'rating', + factor: 0.5, + modifier: 'none' as const, + missing: 0, + }, + }, + { + gauss: { + createdAt: { + origin: 'now', + scale: '90d', + offset: '7d', + decay: 0.5, + }, + }, + }, + ], + score_mode: 'sum' as const, + boost_mode: 'multiply' as const, + }, + }, + sort: this.buildSort(sort), + highlight: hasQuery + ? { + fields: { + title: {}, + description: { fragment_size: 150, number_of_fragments: 1 }, + }, + } + : undefined, + aggs: { + categories: { terms: { field: 'category' } }, + levels: { terms: { field: 'level' } }, + price_ranges: { + range: { + field: 'price', + ranges: [{ to: 50 }, { from: 50, to: 100 }, { from: 100, to: 200 }, { from: 200 }], + }, + }, + }, + }); + const total = typeof result.hits.total === 'object' + ? result.hits.total.value + : (result.hits.total ?? 0); + const hits = result.hits.hits; + const aggs = result.aggregations as unknown; + const rankedResults = this.rankResults(hits); + this.logSearch(sanitizedQuery, rankedResults.length, normalizedFilters, sort); + return { + results: rankedResults, + total, + page, + limit, + facets: { + categories: aggs?.categories?.buckets ?? [], + levels: aggs?.levels?.buckets ?? [], + priceRanges: aggs?.price_ranges?.buckets ?? [], + }, + }; + }, CACHE_TTL.SEARCH_RESULTS); + } + async getAutoComplete(query: string) { + const sanitizedQuery = (query ?? '').trim().slice(0, 100); + const cacheKey = `${CACHE_PREFIXES.SEARCH}:autocomplete:${sanitizedQuery}`; + return this.cachingService.getOrSet(cacheKey, () => this.autoCompleteService.getSuggestions(sanitizedQuery), CACHE_TTL.SEARCH_RESULTS); + } + async getAvailableFilters() { + const cacheKey = `${CACHE_PREFIXES.SEARCH}:filters`; + return this.cachingService.getOrSet(cacheKey, () => this.searchFiltersService.getFilters(), CACHE_TTL.STATIC_CONTENT); + } + async getSearchAnalytics(days = 7) { + const from = new Date(); + from.setDate(from.getDate() - days); const result = await this.elasticsearchService.search({ - index: COURSES_INDEX, - from, - size: limit, - timeout: '1500ms', - track_total_hits: 10000, - _source: SEARCH_SOURCE_FIELDS, - query: { - function_score: { - query: this.buildSearchQuery(sanitizedQuery, normalizedFilters, hasQuery), - functions: [ - { - field_value_factor: { - field: 'views', - factor: 0.1, - modifier: 'log1p' as const, - missing: 1, - }, + index: SEARCH_ANALYTICS_INDEX, + size: 0, + query: { + range: { + timestamp: { gte: from.toISOString() }, }, - { - field_value_factor: { - field: 'rating', - factor: 0.5, - modifier: 'none' as const, - missing: 0, - }, + }, + aggs: { + total_searches: { value_count: { field: 'query.keyword' } }, + top_queries: { + terms: { field: 'query.keyword', size: 10 }, + aggs: { + avg_results: { avg: { field: 'resultsCount' } }, + }, }, - { - gauss: { - createdAt: { - origin: 'now', - scale: '90d', - offset: '7d', - decay: 0.5, + zero_result_queries: { + filter: { term: { resultsCount: 0 } }, + aggs: { + queries: { terms: { field: 'query.keyword', size: 10 } }, }, - }, }, - ], - score_mode: 'sum' as const, - boost_mode: 'multiply' as const, - }, - }, - sort: this.buildSort(sort), - highlight: hasQuery - ? { - fields: { - title: {}, - description: { fragment_size: 150, number_of_fragments: 1 }, + searches_over_time: { + date_histogram: { + field: 'timestamp', + calendar_interval: 'day' as const, + }, }, - } - : undefined, - aggs: { - categories: { terms: { field: 'category' } }, - levels: { terms: { field: 'level' } }, - price_ranges: { - range: { - field: 'price', - ranges: [{ to: 50 }, { from: 50, to: 100 }, { from: 100, to: 200 }, { from: 200 }], - }, }, - }, }); - - const total = - typeof result.hits.total === 'object' - ? result.hits.total.value - : (result.hits.total ?? 0); - - const hits = result.hits.hits; - const aggs = result.aggregations as any; - - const rankedResults = this.rankResults(hits); - this.logSearch(sanitizedQuery, rankedResults.length, normalizedFilters, sort); - + const aggs = result.aggregations as unknown; return { - results: rankedResults, - total, - page, - limit, - facets: { - categories: aggs?.categories?.buckets ?? [], - levels: aggs?.levels?.buckets ?? [], - priceRanges: aggs?.price_ranges?.buckets ?? [], - }, + totalSearches: aggs?.total_searches?.value ?? 0, + topQueries: aggs?.top_queries?.buckets ?? [], + zeroResultQueries: aggs?.zero_result_queries?.queries?.buckets ?? [], + searchesOverTime: aggs?.searches_over_time?.buckets ?? [], }; - }, - CACHE_TTL.SEARCH_RESULTS, - ); - } - - async getAutoComplete(query: string) { - const sanitizedQuery = (query ?? '').trim().slice(0, 100); - const cacheKey = `${CACHE_PREFIXES.SEARCH}:autocomplete:${sanitizedQuery}`; - - return this.cachingService.getOrSet( - cacheKey, - () => this.autoCompleteService.getSuggestions(sanitizedQuery), - CACHE_TTL.SEARCH_RESULTS, - ); - } - - async getAvailableFilters() { - const cacheKey = `${CACHE_PREFIXES.SEARCH}:filters`; - - return this.cachingService.getOrSet( - cacheKey, - () => this.searchFiltersService.getFilters(), - CACHE_TTL.STATIC_CONTENT, - ); - } - - async getSearchAnalytics(days = 7) { - const from = new Date(); - from.setDate(from.getDate() - days); - - const result = await this.elasticsearchService.search({ - index: SEARCH_ANALYTICS_INDEX, - size: 0, - query: { - range: { - timestamp: { gte: from.toISOString() }, - }, - }, - aggs: { - total_searches: { value_count: { field: 'query.keyword' } }, - top_queries: { - terms: { field: 'query.keyword', size: 10 }, - aggs: { - avg_results: { avg: { field: 'resultsCount' } }, - }, - }, - zero_result_queries: { - filter: { term: { resultsCount: 0 } }, - aggs: { - queries: { terms: { field: 'query.keyword', size: 10 } }, - }, - }, - searches_over_time: { - date_histogram: { - field: 'timestamp', - calendar_interval: 'day' as const, - }, - }, - }, - }); - - const aggs = result.aggregations as any; - return { - totalSearches: aggs?.total_searches?.value ?? 0, - topQueries: aggs?.top_queries?.buckets ?? [], - zeroResultQueries: aggs?.zero_result_queries?.queries?.buckets ?? [], - searchesOverTime: aggs?.searches_over_time?.buckets ?? [], - }; - } - - private buildFilters(filters: any) { - const esFilters: any[] = []; - if (filters.category) { - const category = this.normalizeKeywordValue(filters.category); - if (Array.isArray(category)) { - esFilters.push({ terms: { category } }); - } else if (category) { - esFilters.push({ term: { category } }); - } - } - if (filters.level) { - const level = this.normalizeKeywordValue(filters.level); - if (Array.isArray(level)) { - esFilters.push({ terms: { level } }); - } else if (level) { - esFilters.push({ term: { level } }); - } } - if (filters.price) { - esFilters.push({ range: { price: filters.price } }); + private buildFilters(filters: unknown) { + const esFilters: unknown[] = []; + if (filters.category) { + const category = this.normalizeKeywordValue(filters.category); + if (Array.isArray(category)) { + esFilters.push({ terms: { category } }); + } + else if (category) { + esFilters.push({ term: { category } }); + } + } + if (filters.level) { + const level = this.normalizeKeywordValue(filters.level); + if (Array.isArray(level)) { + esFilters.push({ terms: { level } }); + } + else if (level) { + esFilters.push({ term: { level } }); + } + } + if (filters.price) { + esFilters.push({ range: { price: filters.price } }); + } + if (filters.language) { + const language = this.normalizeKeywordValue(filters.language); + if (Array.isArray(language)) { + esFilters.push({ terms: { language } }); + } + else if (language) { + esFilters.push({ term: { language } }); + } + } + if (filters.instructorId) { + const instructorId = this.normalizeString(filters.instructorId, false); + if (instructorId) { + esFilters.push({ term: { instructorId } }); + } + } + return esFilters; } - if (filters.language) { - const language = this.normalizeKeywordValue(filters.language); - if (Array.isArray(language)) { - esFilters.push({ terms: { language } }); - } else if (language) { - esFilters.push({ term: { language } }); - } - } - if (filters.instructorId) { - const instructorId = this.normalizeString(filters.instructorId, false); - if (instructorId) { - esFilters.push({ term: { instructorId } }); - } - } - return esFilters; - } - - private buildSearchQuery(query: string, filters: any, hasQuery: boolean): Record { - if (!hasQuery) { - return { - bool: { - filter: this.buildFilters(filters), - }, - }; - } - - return { - bool: { - filter: this.buildFilters(filters), - should: [ - { - multi_match: { - query, - fields: ['title^3', 'description^2', 'content', 'tags^2'], - type: 'best_fields' as const, - operator: 'and' as const, - fuzziness: 'AUTO:4,7', - prefix_length: 1, - }, - }, - { - multi_match: { - query, - type: 'bool_prefix' as const, - fields: ['title.search', 'title.search._2gram', 'title.search._3gram'], - boost: 2, + private buildSearchQuery(query: string, filters: unknown, hasQuery: boolean): Record { + if (!hasQuery) { + return { + bool: { + filter: this.buildFilters(filters), + }, + }; + } + return { + bool: { + filter: this.buildFilters(filters), + should: [ + { + multi_match: { + query, + fields: ['title^3', 'description^2', 'content', 'tags^2'], + type: 'best_fields' as const, + operator: 'and' as const, + fuzziness: 'AUTO:4,7', + prefix_length: 1, + }, + }, + { + multi_match: { + query, + type: 'bool_prefix' as const, + fields: ['title.search', 'title.search._2gram', 'title.search._3gram'], + boost: 2, + }, + }, + ], + minimum_should_match: 1, }, - }, - ], - minimum_should_match: 1, - }, - }; - } - - private buildSort(sort?: string) { - if (sort === 'relevance') { - return ['_score']; - } else if (sort === 'popularity') { - return [{ views: { order: 'desc' as const } }]; - } else if (sort === 'rating') { - return [{ rating: { order: 'desc' as const } }]; - } else if (sort === 'newest') { - return [{ createdAt: { order: 'desc' as const } }]; - } else if (sort === 'price_asc') { - return [{ price: { order: 'asc' as const } }]; - } else if (sort === 'price_desc') { - return [{ price: { order: 'desc' as const } }]; - } - return ['_score']; - } - - private rankResults(hits: any[]) { - return hits.map((hit) => ({ - ...hit._source, - id: hit._id, - score: hit._score, - highlights: hit.highlight ?? {}, - })); - } - - private logSearch(query: string, resultsCount: number, filters?: any, sort?: string): void { - // Fire-and-forget: analytics must not slow down or fail search responses - this.elasticsearchService - .index({ - index: SEARCH_ANALYTICS_INDEX, - document: { - query, - resultsCount, - filters: filters ?? {}, - sort: sort ?? 'relevance', - timestamp: new Date().toISOString(), - }, - }) - .catch((err) => { - this.logger.warn(`Search analytics logging failed: ${err.message}`); - }); - } - - private hashSearchParams( - query: string, - filters: any, - sort?: string, - page = 1, - limit = 20, - ): string { - const str = `${query}:${JSON.stringify(filters)}:${sort ?? 'default'}:${page}:${limit}`; - let hash = 0; - for (let i = 0; i < str.length; i++) { - const char = str.charCodeAt(i); - hash = (hash << 5) - hash + char; - hash = hash & hash; - } - return Math.abs(hash).toString(36); - } - - private normalizeFilters(filters: any): SearchFilters { - const safeFilters = filters && typeof filters === 'object' ? filters : {}; - const normalized: SearchFilters = {}; - - const category = this.normalizeKeywordValue(safeFilters.category); - if (category) { - normalized.category = category; - } - - const level = this.normalizeKeywordValue(safeFilters.level); - if (level) { - normalized.level = level; + }; } - - const language = this.normalizeKeywordValue(safeFilters.language); - if (language) { - normalized.language = language; + private buildSort(sort?: string) { + if (sort === 'relevance') { + return ['_score']; + } + else if (sort === 'popularity') { + return [{ views: { order: 'desc' as const } }]; + } + else if (sort === 'rating') { + return [{ rating: { order: 'desc' as const } }]; + } + else if (sort === 'newest') { + return [{ createdAt: { order: 'desc' as const } }]; + } + else if (sort === 'price_asc') { + return [{ price: { order: 'asc' as const } }]; + } + else if (sort === 'price_desc') { + return [{ price: { order: 'desc' as const } }]; + } + return ['_score']; } - - const instructorId = this.normalizeString(safeFilters.instructorId, false); - if (instructorId) { - normalized.instructorId = instructorId; + private rankResults(hits: unknown[]) { + return hits.map((hit) => ({ + ...hit._source, + id: hit._id, + score: hit._score, + highlights: hit.highlight ?? {}, + })); } - - const price = this.normalizePriceRange(safeFilters.price); - if (price) { - normalized.price = price; + private logSearch(query: string, resultsCount: number, filters?: unknown, sort?: string): void { + // Fire-and-forget: analytics must not slow down or fail search responses + this.elasticsearchService + .index({ + index: SEARCH_ANALYTICS_INDEX, + document: { + query, + resultsCount, + filters: filters ?? {}, + sort: sort ?? 'relevance', + timestamp: new Date().toISOString(), + }, + }) + .catch((err) => { + this.logger.warn(`Search analytics logging failed: ${err.message}`); + }); } - - return normalized; - } - - private normalizeKeywordValue(value: unknown): string | string[] | null { - if (Array.isArray(value)) { - const normalized = value - .map((item) => this.normalizeString(item, true)) - .filter((item): item is string => !!item); - if (normalized.length === 0) { - return null; - } - return Array.from(new Set(normalized)).sort(); + private hashSearchParams(query: string, filters: unknown, sort?: string, page = 1, limit = 20): string { + const str = `${query}:${JSON.stringify(filters)}:${sort ?? 'default'}:${page}:${limit}`; + let hash = 0; + for (let i = 0; i < str.length; i++) { + const char = str.charCodeAt(i); + hash = (hash << 5) - hash + char; + hash = hash & hash; + } + return Math.abs(hash).toString(36); } - - return this.normalizeString(value, true); - } - - private normalizeString(value: unknown, lowerCase: boolean): string | null { - if (typeof value !== 'string') { - return null; + private normalizeFilters(filters: unknown): SearchFilters { + const safeFilters = filters && typeof filters === 'object' ? filters : {}; + const normalized: SearchFilters = {}; + const category = this.normalizeKeywordValue(safeFilters.category); + if (category) { + normalized.category = category; + } + const level = this.normalizeKeywordValue(safeFilters.level); + if (level) { + normalized.level = level; + } + const language = this.normalizeKeywordValue(safeFilters.language); + if (language) { + normalized.language = language; + } + const instructorId = this.normalizeString(safeFilters.instructorId, false); + if (instructorId) { + normalized.instructorId = instructorId; + } + const price = this.normalizePriceRange(safeFilters.price); + if (price) { + normalized.price = price; + } + return normalized; } - - const normalized = value.trim(); - if (!normalized) { - return null; + private normalizeKeywordValue(value: unknown): string | string[] | null { + if (Array.isArray(value)) { + const normalized = value + .map((item) => this.normalizeString(item, true)) + .filter((item): item is string => !!item); + if (normalized.length === 0) { + return null; + } + return Array.from(new Set(normalized)).sort(); + } + return this.normalizeString(value, true); } - - return lowerCase ? normalized.toLowerCase() : normalized; - } - - private normalizePriceRange( - value: unknown, - ): { gte?: number; lte?: number; gt?: number; lt?: number } | null { - if (!value || typeof value !== 'object' || Array.isArray(value)) { - return null; + private normalizeString(value: unknown, lowerCase: boolean): string | null { + if (typeof value !== 'string') { + return null; + } + const normalized = value.trim(); + if (!normalized) { + return null; + } + return lowerCase ? normalized.toLowerCase() : normalized; } - - const range = value as Record; - const normalized: { gte?: number; lte?: number; gt?: number; lt?: number } = {}; - for (const key of ['gte', 'lte', 'gt', 'lt'] as const) { - const currentValue = range[key]; - if (typeof currentValue === 'number' && Number.isFinite(currentValue)) { - normalized[key] = currentValue; - } + private normalizePriceRange(value: unknown): { + gte?: number; + lte?: number; + gt?: number; + lt?: number; + } | null { + if (!value || typeof value !== 'object' || Array.isArray(value)) { + return null; + } + const range = value as Record; + const normalized: { + gte?: number; + lte?: number; + gt?: number; + lt?: number; + } = {}; + for (const key of ['gte', 'lte', 'gt', 'lt'] as const) { + const currentValue = range[key]; + if (typeof currentValue === 'number' && Number.isFinite(currentValue)) { + normalized[key] = currentValue; + } + } + return Object.keys(normalized).length > 0 ? normalized : null; } - - return Object.keys(normalized).length > 0 ? normalized : null; - } } diff --git a/src/security/audit/audit-logging.service.ts b/src/security/audit/audit-logging.service.ts index 30d22583..50dbcc5b 100644 --- a/src/security/audit/audit-logging.service.ts +++ b/src/security/audit/audit-logging.service.ts @@ -1,28 +1,21 @@ import { Injectable, Logger } from '@nestjs/common'; - @Injectable() export class AuditLoggingService { - private readonly logger = new Logger(AuditLoggingService.name); - - log(event: string, data: Record): void { - this.logger.log( - JSON.stringify({ - event, - data, - timestamp: new Date().toISOString(), - }), - ); - } - - logLogin(userId: string): void { - this.log('USER_LOGIN', { userId }); - } - - logDataAccess(userId: string, resource: string): void { - this.log('DATA_ACCESS', { userId, resource }); - } - - logDeletion(userId: string, resource: string): void { - this.log('DATA_DELETION', { userId, resource }); - } + private readonly logger = new Logger(AuditLoggingService.name); + log(event: string, data: Record): void { + this.logger.log(JSON.stringify({ + event, + data, + timestamp: new Date().toISOString(), + })); + } + logLogin(userId: string): void { + this.log('USER_LOGIN', { userId }); + } + logDataAccess(userId: string, resource: string): void { + this.log('DATA_ACCESS', { userId, resource }); + } + logDeletion(userId: string, resource: string): void { + this.log('DATA_DELETION', { userId, resource }); + } } diff --git a/src/security/compliance/compliance.service.ts b/src/security/compliance/compliance.service.ts index 7e88fe6d..d6a4af2a 100644 --- a/src/security/compliance/compliance.service.ts +++ b/src/security/compliance/compliance.service.ts @@ -1,27 +1,24 @@ import { Injectable } from '@nestjs/common'; - @Injectable() export class ComplianceService { - async exportUserData(userId: string): Promise { - // Fetch from all relevant tables - return { - userId, - profile: {}, - files: [], - activityLogs: [], - }; - } - - async deleteUserData(_userId: string): Promise { - // Soft delete or anonymize - return { success: true }; - } - - async anonymizeData(data: any): Promise { - return { - ...data, - email: 'anonymized@domain.com', - phone: null, - }; - } + async exportUserData(userId: string): Promise { + // Fetch from all relevant tables + return { + userId, + profile: {}, + files: [], + activityLogs: [], + }; + } + async deleteUserData(_userId: string): Promise { + // Soft delete or anonymize + return { success: true }; + } + async anonymizeData(data: unknown): Promise { + return { + ...data, + email: 'anonymized@domain.com', + phone: null, + }; + } } diff --git a/src/security/encryption/encryption.service.ts b/src/security/encryption/encryption.service.ts index 6c1abfe8..a4c5b852 100644 --- a/src/security/encryption/encryption.service.ts +++ b/src/security/encryption/encryption.service.ts @@ -1,48 +1,37 @@ import { Injectable } from '@nestjs/common'; import * as crypto from 'crypto'; - @Injectable() export class EncryptionService { - private readonly algorithm = 'aes-256-gcm'; - private readonly key = crypto.createHash('sha256').update(this.getEncryptionSecret()).digest(); - - private getEncryptionSecret(): string { - const secret = process.env.ENCRYPTION_SECRET; - - if (!secret) { - throw new Error('ENCRYPTION_SECRET is required to initialize EncryptionService'); + private readonly algorithm = 'aes-256-gcm'; + private readonly key = crypto.createHash('sha256').update(this.getEncryptionSecret()).digest(); + private getEncryptionSecret(): string { + const secret = process.env.ENCRYPTION_SECRET; + if (!secret) { + throw new Error('ENCRYPTION_SECRET is required to initialize EncryptionService'); + } + return secret; + } + encrypt(text: string) { + const iv = crypto.randomBytes(16); + const cipher = crypto.createCipheriv(this.algorithm, this.key, iv); + const encrypted = Buffer.concat([cipher.update(text, 'utf8'), cipher.final()]); + return { + iv: iv.toString('hex'), + content: encrypted.toString('hex'), + tag: cipher.getAuthTag().toString('hex'), + }; + } + decrypt(payload: { + iv: string; + content: string; + tag: string; + }) { + const decipher = crypto.createDecipheriv(this.algorithm, this.key, Buffer.from(payload.iv, 'hex')); + decipher.setAuthTag(Buffer.from(payload.tag, 'hex')); + const decrypted = Buffer.concat([ + decipher.update(Buffer.from(payload.content, 'hex')), + decipher.final(), + ]); + return decrypted.toString('utf8'); } - - return secret; - } - - encrypt(text: string) { - const iv = crypto.randomBytes(16); - const cipher = crypto.createCipheriv(this.algorithm, this.key, iv); - - const encrypted = Buffer.concat([cipher.update(text, 'utf8'), cipher.final()]); - - return { - iv: iv.toString('hex'), - content: encrypted.toString('hex'), - tag: cipher.getAuthTag().toString('hex'), - }; - } - - decrypt(payload: { iv: string; content: string; tag: string }) { - const decipher = crypto.createDecipheriv( - this.algorithm, - this.key, - Buffer.from(payload.iv, 'hex'), - ); - - decipher.setAuthTag(Buffer.from(payload.tag, 'hex')); - - const decrypted = Buffer.concat([ - decipher.update(Buffer.from(payload.content, 'hex')), - decipher.final(), - ]); - - return decrypted.toString('utf8'); - } } diff --git a/src/security/security.module.ts b/src/security/security.module.ts index 3d70dff7..aad52da8 100644 --- a/src/security/security.module.ts +++ b/src/security/security.module.ts @@ -5,16 +5,16 @@ import { EncryptionService } from './encryption/encryption.service'; import { ThreatDetectionService } from './threats/threat-detection.service'; import { ComplianceService } from './compliance/compliance.service'; import { AuditLoggingService } from './audit/audit-logging.service'; - @Module({ - imports: [ScheduleModule.forRoot()], - providers: [ - SecurityService, - EncryptionService, - ThreatDetectionService, - ComplianceService, - AuditLoggingService, - ], - exports: [SecurityService, EncryptionService], + imports: [ScheduleModule.forRoot()], + providers: [ + SecurityService, + EncryptionService, + ThreatDetectionService, + ComplianceService, + AuditLoggingService, + ], + exports: [SecurityService, EncryptionService], }) -export class SecurityModule {} +export class SecurityModule { +} diff --git a/src/security/security.service.ts b/src/security/security.service.ts index 7ed9ea5c..ddbf0328 100644 --- a/src/security/security.service.ts +++ b/src/security/security.service.ts @@ -1,12 +1,11 @@ import { Injectable } from '@nestjs/common'; import { Cron } from '@nestjs/schedule'; - @Injectable() export class SecurityService { - @Cron('0 2 * * *') - async enforceRetentionPolicy(): Promise { - const retentionDays = Number(process.env.DATA_RETENTION_DAYS); - const cutoffDate = new Date(); - cutoffDate.setDate(cutoffDate.getDate() - retentionDays); - } + @Cron('0 2 * * *') + async enforceRetentionPolicy(): Promise { + const retentionDays = Number(process.env.DATA_RETENTION_DAYS); + const cutoffDate = new Date(); + cutoffDate.setDate(cutoffDate.getDate() - retentionDays); + } } diff --git a/src/security/threats/threat-detection.service.ts b/src/security/threats/threat-detection.service.ts index 76c14e5f..235dabe3 100644 --- a/src/security/threats/threat-detection.service.ts +++ b/src/security/threats/threat-detection.service.ts @@ -1,23 +1,18 @@ import { Injectable, ForbiddenException } from '@nestjs/common'; - @Injectable() export class ThreatDetectionService { - private failedAttempts = new Map(); - - analyzeRequest(ip: string): void { - const attempts = this.failedAttempts.get(ip) || 0; - - if (attempts > 10) { - throw new ForbiddenException('Suspicious activity detected'); + private failedAttempts = new Map(); + analyzeRequest(ip: string): void { + const attempts = this.failedAttempts.get(ip) || 0; + if (attempts > 10) { + throw new ForbiddenException('Suspicious activity detected'); + } + } + recordFailure(ip: string): void { + const attempts = this.failedAttempts.get(ip) || 0; + this.failedAttempts.set(ip, attempts + 1); + } + reset(ip: string): void { + this.failedAttempts.delete(ip); } - } - - recordFailure(ip: string): void { - const attempts = this.failedAttempts.get(ip) || 0; - this.failedAttempts.set(ip, attempts + 1); - } - - reset(ip: string): void { - this.failedAttempts.delete(ip); - } } diff --git a/src/session/session.module.ts b/src/session/session.module.ts index 130b61d4..0bb51f97 100644 --- a/src/session/session.module.ts +++ b/src/session/session.module.ts @@ -3,32 +3,30 @@ import { ConfigModule, ConfigService } from '@nestjs/config'; import Redis from 'ioredis'; import { SESSION_REDIS_CLIENT } from './session.constants'; import { SessionService } from './session.service'; - @Global() @Module({ - imports: [ConfigModule], - providers: [ - { - provide: SESSION_REDIS_CLIENT, - inject: [ConfigService], - useFactory: (configService: ConfigService) => { - const client = new Redis({ - host: configService.get('REDIS_HOST') || 'localhost', - port: parseInt(configService.get('REDIS_PORT') || '6379', 10), - lazyConnect: false, - maxRetriesPerRequest: null, - enableReadyCheck: true, - }); - - client.on('error', () => { - // Prevent unhandled error events when Redis is temporarily unavailable. - }); - - return client; - }, - }, - SessionService, - ], - exports: [SESSION_REDIS_CLIENT, SessionService], + imports: [ConfigModule], + providers: [ + { + provide: SESSION_REDIS_CLIENT, + inject: [ConfigService], + useFactory: (configService: ConfigService) => { + const client = new Redis({ + host: configService.get('REDIS_HOST') || 'localhost', + port: parseInt(configService.get('REDIS_PORT') || '6379', 10), + lazyConnect: false, + maxRetriesPerRequest: null, + enableReadyCheck: true, + }); + client.on('error', () => { + // Prevent unhandled error events when Redis is temporarily unavailable. + }); + return client; + }, + }, + SessionService, + ], + exports: [SESSION_REDIS_CLIENT, SessionService], }) -export class SessionModule {} +export class SessionModule { +} diff --git a/src/session/session.service.ts b/src/session/session.service.ts index bc97118b..cefa7fcd 100644 --- a/src/session/session.service.ts +++ b/src/session/session.service.ts @@ -3,218 +3,174 @@ import { ConfigService } from '@nestjs/config'; import Redis from 'ioredis'; import { randomUUID } from 'crypto'; import { SESSION_REDIS_CLIENT } from './session.constants'; - interface SessionRecord { - sid: string; - userId: string; - metadata: Record; - version: number; - createdAt: number; - updatedAt: number; + sid: string; + userId: string; + metadata: Record; + version: number; + createdAt: number; + updatedAt: number; } - @Injectable() export class SessionService implements OnModuleDestroy { - private readonly logger = new Logger(SessionService.name); - private readonly sessionPrefix: string; - private readonly legacySessionPrefix: string; - private readonly sessionTtlSeconds: number; - private readonly lockTtlMs: number; - private readonly lockRetries: number; - private readonly lockRetryDelayMs: number; - - constructor( - @Inject(SESSION_REDIS_CLIENT) private readonly redis: Redis, - private readonly configService: ConfigService, - ) { - this.sessionPrefix = this.configService.get('AUTH_SESSION_PREFIX') || 'auth:sess:'; - this.legacySessionPrefix = - this.configService.get('AUTH_SESSION_LEGACY_PREFIX') || 'session:'; - this.sessionTtlSeconds = parseInt( - this.configService.get('AUTH_SESSION_TTL_SECONDS') || '604800', - 10, - ); - this.lockTtlMs = parseInt(this.configService.get('SESSION_LOCK_TTL_MS') || '5000', 10); - this.lockRetries = parseInt( - this.configService.get('SESSION_LOCK_MAX_RETRIES') || '5', - 10, - ); - this.lockRetryDelayMs = parseInt( - this.configService.get('SESSION_LOCK_RETRY_DELAY_MS') || '120', - 10, - ); - } - - async onModuleDestroy(): Promise { - if (this.redis.status !== 'end') { - await this.redis.quit(); + private readonly logger = new Logger(SessionService.name); + private readonly sessionPrefix: string; + private readonly legacySessionPrefix: string; + private readonly sessionTtlSeconds: number; + private readonly lockTtlMs: number; + private readonly lockRetries: number; + private readonly lockRetryDelayMs: number; + constructor( + @Inject(SESSION_REDIS_CLIENT) + private readonly redis: Redis, private readonly configService: ConfigService) { + this.sessionPrefix = this.configService.get('AUTH_SESSION_PREFIX') || 'auth:sess:'; + this.legacySessionPrefix = + this.configService.get('AUTH_SESSION_LEGACY_PREFIX') || 'session:'; + this.sessionTtlSeconds = parseInt(this.configService.get('AUTH_SESSION_TTL_SECONDS') || '604800', 10); + this.lockTtlMs = parseInt(this.configService.get('SESSION_LOCK_TTL_MS') || '5000', 10); + this.lockRetries = parseInt(this.configService.get('SESSION_LOCK_MAX_RETRIES') || '5', 10); + this.lockRetryDelayMs = parseInt(this.configService.get('SESSION_LOCK_RETRY_DELAY_MS') || '120', 10); } - } - - async createSession(userId: string, metadata: Record = {}): Promise { - const sid = randomUUID(); - const now = Date.now(); - const session: SessionRecord = { - sid, - userId, - metadata, - version: 1, - createdAt: now, - updatedAt: now, - }; - - await this.redis.set( - this.sessionKey(sid), - JSON.stringify(session), - 'EX', - this.sessionTtlSeconds, - ); - return sid; - } - - async getSession(sid: string): Promise { - const data = await this.redis.get(this.sessionKey(sid)); - if (!data) { - return this.migrateLegacySessionIfNeeded(sid); + async onModuleDestroy(): Promise { + if (this.redis.status !== 'end') { + await this.redis.quit(); + } } - - try { - return JSON.parse(data) as SessionRecord; - } catch { - this.logger.warn(`Invalid session payload for sid=${sid}`); - return null; + async createSession(userId: string, metadata: Record = {}): Promise { + const sid = randomUUID(); + const now = Date.now(); + const session: SessionRecord = { + sid, + userId, + metadata, + version: 1, + createdAt: now, + updatedAt: now, + }; + await this.redis.set(this.sessionKey(sid), JSON.stringify(session), 'EX', this.sessionTtlSeconds); + return sid; } - } - - async touchSession(sid: string, metadataPatch: Record = {}): Promise { - const session = await this.getSession(sid); - if (!session) { - return; + async getSession(sid: string): Promise { + const data = await this.redis.get(this.sessionKey(sid)); + if (!data) { + return this.migrateLegacySessionIfNeeded(sid); + } + try { + return JSON.parse(data) as SessionRecord; + } + catch { + this.logger.warn(`Invalid session payload for sid=${sid}`); + return null; + } } - - const nextSession: SessionRecord = { - ...session, - metadata: { - ...session.metadata, - ...metadataPatch, - }, - updatedAt: Date.now(), - version: session.version + 1, - }; - - await this.redis - .multi() - .set(this.sessionKey(sid), JSON.stringify(nextSession)) - .expire(this.sessionKey(sid), this.sessionTtlSeconds) - .exec(); - } - - async removeSession(sid: string): Promise { - await this.redis.del(this.sessionKey(sid)); - } - - async migrateSession(oldSid: string, newSid = randomUUID()): Promise { - const existing = await this.getSession(oldSid); - if (!existing) { - return newSid; + async touchSession(sid: string, metadataPatch: Record = {}): Promise { + const session = await this.getSession(sid); + if (!session) { + return; + } + const nextSession: SessionRecord = { + ...session, + metadata: { + ...session.metadata, + ...metadataPatch, + }, + updatedAt: Date.now(), + version: session.version + 1, + }; + await this.redis + .multi() + .set(this.sessionKey(sid), JSON.stringify(nextSession)) + .expire(this.sessionKey(sid), this.sessionTtlSeconds) + .exec(); } - - const migrated: SessionRecord = { - ...existing, - sid: newSid, - updatedAt: Date.now(), - version: existing.version + 1, - }; - - await this.redis - .multi() - .set(this.sessionKey(newSid), JSON.stringify(migrated), 'EX', this.sessionTtlSeconds) - .del(this.sessionKey(oldSid)) - .exec(); - - return newSid; - } - - async withLock(lockName: string, handler: () => Promise): Promise { - const lockKey = `lock:${lockName}`; - const lockToken = randomUUID(); - let locked = false; - - for (let attempt = 0; attempt <= this.lockRetries; attempt += 1) { - const response = await this.redis.set(lockKey, lockToken, 'PX', this.lockTtlMs, 'NX'); - if (response === 'OK') { - locked = true; - break; - } - - if (attempt < this.lockRetries) { - await this.delay(this.lockRetryDelayMs); - } + async removeSession(sid: string): Promise { + await this.redis.del(this.sessionKey(sid)); } - - if (!locked) { - throw new Error(`Could not acquire lock: ${lockName}`); + async migrateSession(oldSid: string, newSid = randomUUID()): Promise { + const existing = await this.getSession(oldSid); + if (!existing) { + return newSid; + } + const migrated: SessionRecord = { + ...existing, + sid: newSid, + updatedAt: Date.now(), + version: existing.version + 1, + }; + await this.redis + .multi() + .set(this.sessionKey(newSid), JSON.stringify(migrated), 'EX', this.sessionTtlSeconds) + .del(this.sessionKey(oldSid)) + .exec(); + return newSid; } - - try { - return await handler(); - } finally { - await this.releaseLock(lockKey, lockToken); + async withLock(lockName: string, handler: () => Promise): Promise { + const lockKey = `lock:${lockName}`; + const lockToken = randomUUID(); + let locked = false; + for (let attempt = 0; attempt <= this.lockRetries; attempt += 1) { + const response = await this.redis.set(lockKey, lockToken, 'PX', this.lockTtlMs, 'NX'); + if (response === 'OK') { + locked = true; + break; + } + if (attempt < this.lockRetries) { + await this.delay(this.lockRetryDelayMs); + } + } + if (!locked) { + throw new Error(`Could not acquire lock: ${lockName}`); + } + try { + return await handler(); + } + finally { + await this.releaseLock(lockKey, lockToken); + } } - } - - private async migrateLegacySessionIfNeeded(sid: string): Promise { - const legacyKey = `${this.legacySessionPrefix}${sid}`; - const currentKey = this.sessionKey(sid); - - if (legacyKey === currentKey) { - return null; + private async migrateLegacySessionIfNeeded(sid: string): Promise { + const legacyKey = `${this.legacySessionPrefix}${sid}`; + const currentKey = this.sessionKey(sid); + if (legacyKey === currentKey) { + return null; + } + const legacyData = await this.redis.get(legacyKey); + if (!legacyData) { + return null; + } + const now = Date.now(); + const migrated: SessionRecord = { + sid, + userId: 'unknown', + metadata: { + source: 'legacy', + payload: legacyData, + }, + version: 1, + createdAt: now, + updatedAt: now, + }; + await this.redis + .multi() + .set(currentKey, JSON.stringify(migrated), 'EX', this.sessionTtlSeconds) + .del(legacyKey) + .exec(); + this.logger.log(`Migrated legacy session sid=${sid} to prefix=${this.sessionPrefix}`); + return migrated; } - - const legacyData = await this.redis.get(legacyKey); - if (!legacyData) { - return null; - } - - const now = Date.now(); - const migrated: SessionRecord = { - sid, - userId: 'unknown', - metadata: { - source: 'legacy', - payload: legacyData, - }, - version: 1, - createdAt: now, - updatedAt: now, - }; - - await this.redis - .multi() - .set(currentKey, JSON.stringify(migrated), 'EX', this.sessionTtlSeconds) - .del(legacyKey) - .exec(); - - this.logger.log(`Migrated legacy session sid=${sid} to prefix=${this.sessionPrefix}`); - return migrated; - } - - private async releaseLock(lockKey: string, lockToken: string): Promise { - const releaseScript = ` + private async releaseLock(lockKey: string, lockToken: string): Promise { + const releaseScript = ` if redis.call('GET', KEYS[1]) == ARGV[1] then return redis.call('DEL', KEYS[1]) end return 0 `; - - await this.redis.eval(releaseScript, 1, lockKey, lockToken); - } - - private sessionKey(sid: string): string { - return `${this.sessionPrefix}${sid}`; - } - - private async delay(ms: number): Promise { - await new Promise((resolve) => setTimeout(resolve, ms)); - } + await this.redis.eval(releaseScript, 1, lockKey, lockToken); + } + private sessionKey(sid: string): string { + return `${this.sessionPrefix}${sid}`; + } + private async delay(ms: number): Promise { + await new Promise((resolve) => setTimeout(resolve, ms)); + } } diff --git a/src/sync/cache/cache-invalidation.service.spec.ts b/src/sync/cache/cache-invalidation.service.spec.ts index 4a9d135b..4c784c15 100644 --- a/src/sync/cache/cache-invalidation.service.spec.ts +++ b/src/sync/cache/cache-invalidation.service.spec.ts @@ -2,63 +2,56 @@ import { Test, TestingModule } from '@nestjs/testing'; import { CacheInvalidationService } from './cache-invalidation.service'; import { CACHE_MANAGER } from '@nestjs/cache-manager'; import { EventEmitter2 } from '@nestjs/event-emitter'; - describe('CacheInvalidationService', () => { - let service: CacheInvalidationService; - let cacheManager: any; - let eventEmitter: EventEmitter2; - - beforeEach(async () => { - const mockCache = { - del: jest.fn().mockResolvedValue({}), - reset: jest.fn().mockResolvedValue({}), - clear: jest.fn().mockResolvedValue({}), - store: { - keys: jest.fn().mockResolvedValue(['key1', 'key2']), - }, - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - CacheInvalidationService, - { - provide: CACHE_MANAGER, - useValue: mockCache, - }, - { - provide: EventEmitter2, - useValue: { - emit: jest.fn(), - }, - }, - ], - }).compile(); - - service = module.get(CacheInvalidationService); - cacheManager = module.get(CACHE_MANAGER); - eventEmitter = module.get(EventEmitter2); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('invalidateKey', () => { - it('should call cacheManager.del and emit event', async () => { - await service.invalidateKey('test-key'); - expect(cacheManager.del).toHaveBeenCalledWith('test-key'); - expect(eventEmitter.emit).toHaveBeenCalledWith('cache.invalidated', { - key: 'test-key', - type: 'single', - }); + let service: CacheInvalidationService; + let cacheManager: unknown; + let eventEmitter: EventEmitter2; + beforeEach(async () => { + const mockCache = { + del: jest.fn().mockResolvedValue({}), + reset: jest.fn().mockResolvedValue({}), + clear: jest.fn().mockResolvedValue({}), + store: { + keys: jest.fn().mockResolvedValue(['key1', 'key2']), + }, + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + CacheInvalidationService, + { + provide: CACHE_MANAGER, + useValue: mockCache, + }, + { + provide: EventEmitter2, + useValue: { + emit: jest.fn(), + }, + }, + ], + }).compile(); + service = module.get(CacheInvalidationService); + cacheManager = module.get(CACHE_MANAGER); + eventEmitter = module.get(EventEmitter2); }); - }); - - describe('purgeAll', () => { - it('should call cacheManager.clear and emit event', async () => { - await service.purgeAll(); - expect(cacheManager.clear).toHaveBeenCalled(); - expect(eventEmitter.emit).toHaveBeenCalledWith('cache.purged', expect.any(Object)); + it('should be defined', () => { + expect(service).toBeDefined(); + }); + describe('invalidateKey', () => { + it('should call cacheManager.del and emit event', async () => { + await service.invalidateKey('test-key'); + expect(cacheManager.del).toHaveBeenCalledWith('test-key'); + expect(eventEmitter.emit).toHaveBeenCalledWith('cache.invalidated', { + key: 'test-key', + type: 'single', + }); + }); + }); + describe('purgeAll', () => { + it('should call cacheManager.clear and emit event', async () => { + await service.purgeAll(); + expect(cacheManager.clear).toHaveBeenCalled(); + expect(eventEmitter.emit).toHaveBeenCalledWith('cache.purged', expect.any(Object)); + }); }); - }); }); diff --git a/src/sync/cache/cache-invalidation.service.ts b/src/sync/cache/cache-invalidation.service.ts index 90310b0f..d85d2814 100644 --- a/src/sync/cache/cache-invalidation.service.ts +++ b/src/sync/cache/cache-invalidation.service.ts @@ -3,71 +3,61 @@ import { CACHE_MANAGER } from '@nestjs/cache-manager'; import { Cache } from 'cache-manager'; import { EventEmitter2 } from '@nestjs/event-emitter'; import { APP_EVENTS } from '../../common/constants/event.constants'; - @Injectable() export class CacheInvalidationService { - private readonly logger = new Logger(CacheInvalidationService.name); - - constructor( - @Inject(CACHE_MANAGER) private cacheManager: Cache, - private eventEmitter: EventEmitter2, - ) {} - - /** - * Invalidates a specific cache key. - */ - async invalidateKey(key: string): Promise { - this.logger.log(`Invalidating cache key: ${key}`); - await this.cacheManager.del(key); - this.eventEmitter.emit(APP_EVENTS.CACHE_INVALIDATED, { key, type: 'single' }); - } - - /** - * Invalidates multiple cache keys based on a pattern. - * Note: Standard cache-manager doesn't support pattern deletion easily with all stores, - * so this is a simplified implementation. - */ - async invalidatePattern(pattern: string): Promise { - this.logger.log(`Invalidating cache pattern: ${pattern}`); - - // In a production environment with Redis, we'd use 'SCAN' and 'DEL' - // For now, we'll emit an event that other specialized listeners might handle - this.eventEmitter.emit(APP_EVENTS.CACHE_INVALIDATED, { pattern, type: 'pattern' }); - - // If the store supports a store-specific method, call it here. - const store: any = (this.cacheManager as any).store; - if (store && typeof store.keys === 'function') { - const keys = await store.keys(pattern); - if (keys && keys.length > 0) { - await Promise.all(keys.map((key: string) => this.cacheManager.del(key))); - } + private readonly logger = new Logger(CacheInvalidationService.name); + constructor( + @Inject(CACHE_MANAGER) + private cacheManager: Cache, private eventEmitter: EventEmitter2) { } + /** + * Invalidates a specific cache key. + */ + async invalidateKey(key: string): Promise { + this.logger.log(`Invalidating cache key: ${key}`); + await this.cacheManager.del(key); + this.eventEmitter.emit(APP_EVENTS.CACHE_INVALIDATED, { key, type: 'single' }); } - } - - /** - * Automatically invalidates cache based on data change events. - */ - async handleDataChange(entity: string, id: string): Promise { - this.logger.log(`Handling data change for ${entity}:${id}`); - - const specificKey = `${entity}:${id}`; - const collectionKey = `${entity}:list:*`; - - await this.invalidateKey(specificKey); - await this.invalidatePattern(collectionKey); - } - - /** - * Purges the entire cache. - */ - async purgeAll(): Promise { - this.logger.warn('Purging entire cache'); - // cache-manager v5+ uses clear() instead of reset() - if (typeof this.cacheManager.clear === 'function') { - await this.cacheManager.clear(); - } else if (typeof (this.cacheManager as any).reset === 'function') { - await (this.cacheManager as any).reset(); + /** + * Invalidates multiple cache keys based on a pattern. + * Note: Standard cache-manager doesn't support pattern deletion easily with all stores, + * so this is a simplified implementation. + */ + async invalidatePattern(pattern: string): Promise { + this.logger.log(`Invalidating cache pattern: ${pattern}`); + // In a production environment with Redis, we'd use 'SCAN' and 'DEL' + // For now, we'll emit an event that other specialized listeners might handle + this.eventEmitter.emit(APP_EVENTS.CACHE_INVALIDATED, { pattern, type: 'pattern' }); + // If the store supports a store-specific method, call it here. + const store: unknown = (this.cacheManager as unknown).store; + if (store && typeof store.keys === 'function') { + const keys = await store.keys(pattern); + if (keys && keys.length > 0) { + await Promise.all(keys.map((key: string) => this.cacheManager.del(key))); + } + } + } + /** + * Automatically invalidates cache based on data change events. + */ + async handleDataChange(entity: string, id: string): Promise { + this.logger.log(`Handling data change for ${entity}:${id}`); + const specificKey = `${entity}:${id}`; + const collectionKey = `${entity}:list:*`; + await this.invalidateKey(specificKey); + await this.invalidatePattern(collectionKey); + } + /** + * Purges the entire cache. + */ + async purgeAll(): Promise { + this.logger.warn('Purging entire cache'); + // cache-manager v5+ uses clear() instead of reset() + if (typeof this.cacheManager.clear === 'function') { + await this.cacheManager.clear(); + } + else if (typeof (this.cacheManager as unknown).reset === 'function') { + await (this.cacheManager as unknown).reset(); + } + this.eventEmitter.emit(APP_EVENTS.CACHE_PURGED, { timestamp: new Date() }); } - this.eventEmitter.emit(APP_EVENTS.CACHE_PURGED, { timestamp: new Date() }); - } } diff --git a/src/sync/conflicts/conflict-resolution.service.spec.ts b/src/sync/conflicts/conflict-resolution.service.spec.ts index e0f4c168..840a70c9 100644 --- a/src/sync/conflicts/conflict-resolution.service.spec.ts +++ b/src/sync/conflicts/conflict-resolution.service.spec.ts @@ -1,71 +1,45 @@ import { Test, TestingModule } from '@nestjs/testing'; -import { - ConflictResolutionService, - ConflictResolutionStrategy, - SyncData, -} from './conflict-resolution.service'; - +import { ConflictResolutionService, ConflictResolutionStrategy, SyncData, } from './conflict-resolution.service'; describe('ConflictResolutionService', () => { - let service: ConflictResolutionService; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [ConflictResolutionService], - }).compile(); - - service = module.get(ConflictResolutionService); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('resolve', () => { - const localData: SyncData = { - id: '1', - version: 2, - lastModified: new Date('2023-01-01T10:00:00Z'), - data: { name: 'Local' }, - }; - - const remoteData: SyncData = { - id: '1', - version: 3, - lastModified: new Date('2023-01-01T11:00:00Z'), - data: { name: 'Remote' }, - }; - - it('should resolve using LAST_WRITE_WINS (remote wins)', () => { - const result = service.resolve( - localData, - remoteData, - ConflictResolutionStrategy.LAST_WRITE_WINS, - ); - expect(result.data.name).toBe('Remote'); + let service: ConflictResolutionService; + beforeEach(async () => { + const module: TestingModule = await Test.createTestingModule({ + providers: [ConflictResolutionService], + }).compile(); + service = module.get(ConflictResolutionService); }); - - it('should resolve using LAST_WRITE_WINS (local wins)', () => { - const olderRemote = { ...remoteData, lastModified: new Date('2023-01-01T09:00:00Z') }; - const result = service.resolve( - localData, - olderRemote, - ConflictResolutionStrategy.LAST_WRITE_WINS, - ); - expect(result.data.name).toBe('Local'); + it('should be defined', () => { + expect(service).toBeDefined(); }); - - it('should resolve using VERSIONING', () => { - const result = service.resolve(localData, remoteData, ConflictResolutionStrategy.VERSIONING); - expect(result.data.name).toBe('Remote'); + describe('resolve', () => { + const localData: SyncData = { + id: '1', + version: 2, + lastModified: new Date('2023-01-01T10:00:00Z'), + data: { name: 'Local' }, + }; + const remoteData: SyncData = { + id: '1', + version: 3, + lastModified: new Date('2023-01-01T11:00:00Z'), + data: { name: 'Remote' }, + }; + it('should resolve using LAST_WRITE_WINS (remote wins)', () => { + const result = service.resolve(localData, remoteData, ConflictResolutionStrategy.LAST_WRITE_WINS); + expect(result.data.name).toBe('Remote'); + }); + it('should resolve using LAST_WRITE_WINS (local wins)', () => { + const olderRemote = { ...remoteData, lastModified: new Date('2023-01-01T09:00:00Z') }; + const result = service.resolve(localData, olderRemote, ConflictResolutionStrategy.LAST_WRITE_WINS); + expect(result.data.name).toBe('Local'); + }); + it('should resolve using VERSIONING', () => { + const result = service.resolve(localData, remoteData, ConflictResolutionStrategy.VERSIONING); + expect(result.data.name).toBe('Remote'); + }); + it('should resolve using MANUAL_MERGE', () => { + const result = service.resolve(localData, remoteData, ConflictResolutionStrategy.MANUAL_MERGE); + expect(result.data._conflict.status).toBe('needs_merge'); + }); }); - - it('should resolve using MANUAL_MERGE', () => { - const result = service.resolve( - localData, - remoteData, - ConflictResolutionStrategy.MANUAL_MERGE, - ); - expect(result.data._conflict.status).toBe('needs_merge'); - }); - }); }); diff --git a/src/sync/conflicts/conflict-resolution.service.ts b/src/sync/conflicts/conflict-resolution.service.ts index 31565dbe..25005369 100644 --- a/src/sync/conflicts/conflict-resolution.service.ts +++ b/src/sync/conflicts/conflict-resolution.service.ts @@ -1,66 +1,54 @@ import { Injectable, Logger } from '@nestjs/common'; - export enum ConflictResolutionStrategy { - LAST_WRITE_WINS = 'LAST_WRITE_WINS', - VERSIONING = 'VERSIONING', - MANUAL_MERGE = 'MANUAL_MERGE', + LAST_WRITE_WINS = 'LAST_WRITE_WINS', + VERSIONING = 'VERSIONING', + MANUAL_MERGE = 'MANUAL_MERGE' } - export interface SyncData { - id: string; - version: number; - lastModified: Date; - data: any; + id: string; + version: number; + lastModified: Date; + data: unknown; } - @Injectable() export class ConflictResolutionService { - private readonly logger = new Logger(ConflictResolutionService.name); - - /** - * Resolves conflicts between two sets of data based on a strategy. - */ - resolve( - localData: SyncData, - remoteData: SyncData, - strategy: ConflictResolutionStrategy = ConflictResolutionStrategy.LAST_WRITE_WINS, - ): SyncData { - this.logger.log(`Resolving conflict for ${localData.id} using ${strategy}`); - - switch (strategy) { - case ConflictResolutionStrategy.LAST_WRITE_WINS: - return this.lastWriteWins(localData, remoteData); - case ConflictResolutionStrategy.VERSIONING: - return this.versioning(localData, remoteData); - case ConflictResolutionStrategy.MANUAL_MERGE: - return this.manualMerge(localData, remoteData); - default: - this.logger.warn(`Unknown strategy ${strategy}, defaulting to LAST_WRITE_WINS`); - return this.lastWriteWins(localData, remoteData); + private readonly logger = new Logger(ConflictResolutionService.name); + /** + * Resolves conflicts between two sets of data based on a strategy. + */ + resolve(localData: SyncData, remoteData: SyncData, strategy: ConflictResolutionStrategy = ConflictResolutionStrategy.LAST_WRITE_WINS): SyncData { + this.logger.log(`Resolving conflict for ${localData.id} using ${strategy}`); + switch (strategy) { + case ConflictResolutionStrategy.LAST_WRITE_WINS: + return this.lastWriteWins(localData, remoteData); + case ConflictResolutionStrategy.VERSIONING: + return this.versioning(localData, remoteData); + case ConflictResolutionStrategy.MANUAL_MERGE: + return this.manualMerge(localData, remoteData); + default: + this.logger.warn(`Unknown strategy ${strategy}, defaulting to LAST_WRITE_WINS`); + return this.lastWriteWins(localData, remoteData); + } + } + private lastWriteWins(local: SyncData, remote: SyncData): SyncData { + return local.lastModified >= remote.lastModified ? local : remote; + } + private versioning(local: SyncData, remote: SyncData): SyncData { + return local.version >= remote.version ? local : remote; + } + private manualMerge(local: SyncData, remote: SyncData): SyncData { + // In a real scenario, this would trigger a notification or + // flag the record for human intervention. + // For now, we'll mark it as "needs_merge" in the data. + return { + ...local, + data: { + ...local.data, + _conflict: { + remote: remote.data, + status: 'needs_merge', + }, + }, + }; } - } - - private lastWriteWins(local: SyncData, remote: SyncData): SyncData { - return local.lastModified >= remote.lastModified ? local : remote; - } - - private versioning(local: SyncData, remote: SyncData): SyncData { - return local.version >= remote.version ? local : remote; - } - - private manualMerge(local: SyncData, remote: SyncData): SyncData { - // In a real scenario, this would trigger a notification or - // flag the record for human intervention. - // For now, we'll mark it as "needs_merge" in the data. - return { - ...local, - data: { - ...local.data, - _conflict: { - remote: remote.data, - status: 'needs_merge', - }, - }, - }; - } } diff --git a/src/sync/consistency/data-consistency.service.spec.ts b/src/sync/consistency/data-consistency.service.spec.ts index 7cd26fbf..38096d17 100644 --- a/src/sync/consistency/data-consistency.service.spec.ts +++ b/src/sync/consistency/data-consistency.service.spec.ts @@ -2,66 +2,52 @@ import { Test, TestingModule } from '@nestjs/testing'; import { DataConsistencyService } from './data-consistency.service'; import { EventEmitter2 } from '@nestjs/event-emitter'; import { getQueueToken } from '@nestjs/bull'; - describe('DataConsistencyService', () => { - let service: DataConsistencyService; - let eventEmitter: EventEmitter2; - let queue: any; - - beforeEach(async () => { - const mockQueue = { - add: jest.fn().mockResolvedValue({}), - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - DataConsistencyService, - { - provide: EventEmitter2, - useValue: { - emit: jest.fn(), - }, - }, - { - provide: getQueueToken('sync-tasks'), - useValue: mockQueue, - }, - ], - }).compile(); - - service = module.get(DataConsistencyService); - eventEmitter = module.get(EventEmitter2); - queue = module.get(getQueueToken('sync-tasks')); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('scheduleConsistencyTask', () => { - it('should add task to queue and emit event', async () => { - await service.scheduleConsistencyTask('1', { foo: 'bar' }); - expect(queue.add).toHaveBeenCalled(); - expect(eventEmitter.emit).toHaveBeenCalledWith( - 'data.consistency.scheduled', - expect.any(Object), - ); + let service: DataConsistencyService; + let eventEmitter: EventEmitter2; + let queue: unknown; + beforeEach(async () => { + const mockQueue = { + add: jest.fn().mockResolvedValue({}), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + DataConsistencyService, + { + provide: EventEmitter2, + useValue: { + emit: jest.fn(), + }, + }, + { + provide: getQueueToken('sync-tasks'), + useValue: mockQueue, + }, + ], + }).compile(); + service = module.get(DataConsistencyService); + eventEmitter = module.get(EventEmitter2); + queue = module.get(getQueueToken('sync-tasks')); }); - }); - - describe('performIntegrityCheck', () => { - it('should return consistent when data matches', async () => { - const result = await service.performIntegrityCheck( - { id: '1', version: 1 }, - { id: '1', version: 1 }, - ); - expect(result.consistent).toBe(true); + it('should be defined', () => { + expect(service).toBeDefined(); }); - - it('should return issues when IDs mismatch', async () => { - const result = await service.performIntegrityCheck({ id: '1' }, { id: '2' }); - expect(result.consistent).toBe(false); - expect(result.issues).toContain('ID mismatch: 1 vs 2'); + describe('scheduleConsistencyTask', () => { + it('should add task to queue and emit event', async () => { + await service.scheduleConsistencyTask('1', { foo: 'bar' }); + expect(queue.add).toHaveBeenCalled(); + expect(eventEmitter.emit).toHaveBeenCalledWith('data.consistency.scheduled', expect.any(Object)); + }); + }); + describe('performIntegrityCheck', () => { + it('should return consistent when data matches', async () => { + const result = await service.performIntegrityCheck({ id: '1', version: 1 }, { id: '1', version: 1 }); + expect(result.consistent).toBe(true); + }); + it('should return issues when IDs mismatch', async () => { + const result = await service.performIntegrityCheck({ id: '1' }, { id: '2' }); + expect(result.consistent).toBe(false); + expect(result.issues).toContain('ID mismatch: 1 vs 2'); + }); }); - }); }); diff --git a/src/sync/consistency/data-consistency.service.ts b/src/sync/consistency/data-consistency.service.ts index c888984b..8c989554 100644 --- a/src/sync/consistency/data-consistency.service.ts +++ b/src/sync/consistency/data-consistency.service.ts @@ -4,99 +4,80 @@ import { InjectQueue } from '@nestjs/bull'; import { Queue } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; import { APP_EVENTS } from '../../common/constants/event.constants'; - export interface IntegrityCheckResult { - consistent: boolean; - issues: string[]; - timestamp: Date; + consistent: boolean; + issues: string[]; + timestamp: Date; } - @Injectable() export class DataConsistencyService { - private readonly logger = new Logger(DataConsistencyService.name); - - constructor( - private eventEmitter: EventEmitter2, - @InjectQueue(QUEUE_NAMES.SYNC_TASKS) private syncQueue: Queue, - ) {} - - /** - * Schedules an eventual consistency task. - */ - async scheduleConsistencyTask(dataId: string, payload: any): Promise { - this.logger.log(`Scheduling consistency task for ${dataId}`); - - // Add to queue for background processing - await this.syncQueue.add( - JOB_NAMES.CONSISTENCY_CHECK, - { - dataId, - payload, - timestamp: new Date(), - }, - { - attempts: 3, - backoff: { - type: 'exponential', - delay: 1000, - }, - }, - ); - - // Emit event for real-time subscribers - this.eventEmitter.emit(APP_EVENTS.DATA_CONSISTENCY_SCHEDULED, { - dataId, - timestamp: new Date(), - }); - } - - /** - * Performs a data integrity check across multiple sources. - */ - async performIntegrityCheck(sourceA: any, sourceB: any): Promise { - this.logger.log('Performing data integrity check'); - const issues: string[] = []; - - // Simple deep equality check or hash comparison could go here - // For demonstration, we'll check if IDs match - if (sourceA.id !== sourceB.id) { - issues.push(`ID mismatch: ${sourceA.id} vs ${sourceB.id}`); + private readonly logger = new Logger(DataConsistencyService.name); + constructor(private eventEmitter: EventEmitter2, + @InjectQueue(QUEUE_NAMES.SYNC_TASKS) + private syncQueue: Queue) { } + /** + * Schedules an eventual consistency task. + */ + async scheduleConsistencyTask(dataId: string, payload: unknown): Promise { + this.logger.log(`Scheduling consistency task for ${dataId}`); + // Add to queue for background processing + await this.syncQueue.add(JOB_NAMES.CONSISTENCY_CHECK, { + dataId, + payload, + timestamp: new Date(), + }, { + attempts: 3, + backoff: { + type: 'exponential', + delay: 1000, + }, + }); + // Emit event for real-time subscribers + this.eventEmitter.emit(APP_EVENTS.DATA_CONSISTENCY_SCHEDULED, { + dataId, + timestamp: new Date(), + }); } - - // Check version consistency if available - if (sourceA.version !== undefined && sourceB.version !== undefined) { - if (Math.abs(sourceA.version - sourceB.version) > 1) { - issues.push(`Version drift too large: ${sourceA.version} vs ${sourceB.version}`); - } + /** + * Performs a data integrity check across multiple sources. + */ + async performIntegrityCheck(sourceA: unknown, sourceB: unknown): Promise { + this.logger.log('Performing data integrity check'); + const issues: string[] = []; + // Simple deep equality check or hash comparison could go here + // For demonstration, we'll check if IDs match + if (sourceA.id !== sourceB.id) { + issues.push(`ID mismatch: ${sourceA.id} vs ${sourceB.id}`); + } + // Check version consistency if available + if (sourceA.version !== undefined && sourceB.version !== undefined) { + if (Math.abs(sourceA.version - sourceB.version) > 1) { + issues.push(`Version drift too large: ${sourceA.version} vs ${sourceB.version}`); + } + } + const consistent = issues.length === 0; + if (!consistent) { + this.logger.warn(`Integrity check failed with issues: ${issues.join(', ')}`); + this.eventEmitter.emit(APP_EVENTS.DATA_INTEGRITY_VIOLATION, { + issues, + timestamp: new Date(), + }); + } + return { + consistent, + issues, + timestamp: new Date(), + }; } - - const consistent = issues.length === 0; - - if (!consistent) { - this.logger.warn(`Integrity check failed with issues: ${issues.join(', ')}`); - this.eventEmitter.emit(APP_EVENTS.DATA_INTEGRITY_VIOLATION, { - issues, - timestamp: new Date(), - }); + /** + * Heals data based on a source of truth. + */ + async heal(staleData: unknown, sourceOfTruth: unknown): Promise { + this.logger.log(`Healing data for ID: ${sourceOfTruth.id}`); + // In a real app, this would update the database or cache + return { + ...sourceOfTruth, + _recoveredAt: new Date(), + }; } - - return { - consistent, - issues, - timestamp: new Date(), - }; - } - - /** - * Heals data based on a source of truth. - */ - async heal(staleData: any, sourceOfTruth: any): Promise { - this.logger.log(`Healing data for ID: ${sourceOfTruth.id}`); - - // In a real app, this would update the database or cache - return { - ...sourceOfTruth, - _recoveredAt: new Date(), - }; - } } diff --git a/src/sync/replication/replication.service.spec.ts b/src/sync/replication/replication.service.spec.ts index cb7e5883..ca1d0243 100644 --- a/src/sync/replication/replication.service.spec.ts +++ b/src/sync/replication/replication.service.spec.ts @@ -2,56 +2,46 @@ import { Test, TestingModule } from '@nestjs/testing'; import { ReplicationService } from './replication.service'; import { EventEmitter2 } from '@nestjs/event-emitter'; import { getQueueToken } from '@nestjs/bull'; - describe('ReplicationService', () => { - let service: ReplicationService; - let eventEmitter: EventEmitter2; - let queue: any; - - beforeEach(async () => { - const mockQueue = { - add: jest.fn().mockResolvedValue({}), - }; - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - ReplicationService, - { - provide: EventEmitter2, - useValue: { - emit: jest.fn(), - }, - }, - { - provide: getQueueToken('sync-tasks'), - useValue: mockQueue, - }, - ], - }).compile(); - - service = module.get(ReplicationService); - eventEmitter = module.get(EventEmitter2); - queue = module.get(getQueueToken('sync-tasks')); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('replicateToRegion', () => { - it('should add replication task to queue', async () => { - await service.replicateToRegion('1', { foo: 'bar' }, 'eu-west-1'); - expect(queue.add).toHaveBeenCalled(); - expect(eventEmitter.emit).toHaveBeenCalledWith( - 'data.replication.started', - expect.any(Object), - ); + let service: ReplicationService; + let eventEmitter: EventEmitter2; + let queue: unknown; + beforeEach(async () => { + const mockQueue = { + add: jest.fn().mockResolvedValue({}), + }; + const module: TestingModule = await Test.createTestingModule({ + providers: [ + ReplicationService, + { + provide: EventEmitter2, + useValue: { + emit: jest.fn(), + }, + }, + { + provide: getQueueToken('sync-tasks'), + useValue: mockQueue, + }, + ], + }).compile(); + service = module.get(ReplicationService); + eventEmitter = module.get(EventEmitter2); + queue = module.get(getQueueToken('sync-tasks')); }); - - it('should skip if target region is same as current', async () => { - // currentRegion defaults to us-east-1 in implementation if not set - await service.replicateToRegion('1', { foo: 'bar' }, 'us-east-1'); - expect(queue.add).not.toHaveBeenCalled(); + it('should be defined', () => { + expect(service).toBeDefined(); + }); + describe('replicateToRegion', () => { + it('should add replication task to queue', async () => { + await service.replicateToRegion('1', { foo: 'bar' }, 'eu-west-1'); + expect(queue.add).toHaveBeenCalled(); + expect(eventEmitter.emit).toHaveBeenCalledWith('data.replication.started', expect.any(Object)); + }); + it('should skip if target region is same as current', async () => { + // currentRegion defaults to us-east-1 in implementation if not set + await service.replicateToRegion('1', { foo: 'bar' }, 'us-east-1'); + expect(queue.add).not.toHaveBeenCalled(); + }); }); - }); }); diff --git a/src/sync/replication/replication.service.ts b/src/sync/replication/replication.service.ts index 57fc8cd0..9f7e7e9d 100644 --- a/src/sync/replication/replication.service.ts +++ b/src/sync/replication/replication.service.ts @@ -3,80 +3,64 @@ import { EventEmitter2 } from '@nestjs/event-emitter'; import { InjectQueue } from '@nestjs/bull'; import { Queue } from 'bull'; import { QUEUE_NAMES, JOB_NAMES } from '../../common/constants/queue.constants'; - export interface ReplicationEvent { - entityId: string; - sourceRegion: string; - targetRegion: string; - data: any; - timestamp: Date; + entityId: string; + sourceRegion: string; + targetRegion: string; + data: unknown; + timestamp: Date; } - @Injectable() export class ReplicationService { - private readonly logger = new Logger(ReplicationService.name); - private readonly currentRegion = process.env.REGION || 'us-east-1'; - - constructor( - private eventEmitter: EventEmitter2, - @InjectQueue(QUEUE_NAMES.SYNC_TASKS) private syncQueue: Queue, - ) {} - - /** - * Replicates data to a target region. - */ - async replicateToRegion(entityId: string, data: any, targetRegion: string): Promise { - if (targetRegion === this.currentRegion) { - this.logger.debug(`Skipping replication to current region: ${this.currentRegion}`); - return; + private readonly logger = new Logger(ReplicationService.name); + private readonly currentRegion = process.env.REGION || 'us-east-1'; + constructor(private eventEmitter: EventEmitter2, + @InjectQueue(QUEUE_NAMES.SYNC_TASKS) + private syncQueue: Queue) { } + /** + * Replicates data to a target region. + */ + async replicateToRegion(entityId: string, data: unknown, targetRegion: string): Promise { + if (targetRegion === this.currentRegion) { + this.logger.debug(`Skipping replication to current region: ${this.currentRegion}`); + return; + } + this.logger.log(`Replicating ${entityId} from ${this.currentRegion} to ${targetRegion}`); + const event: ReplicationEvent = { + entityId, + sourceRegion: this.currentRegion, + targetRegion, + data, + timestamp: new Date(), + }; + // Add to queue for asynchronous replication + await this.syncQueue.add(JOB_NAMES.REPLICATE_DATA, event, { + attempts: 5, + backoff: { + type: 'exponential', + delay: 2000, + }, + }); + this.eventEmitter.emit('data.replication.started', event); + } + /** + * Broadcasts data to all regions except the current one. + */ + async broadcastToAllRegions(entityId: string, data: unknown): Promise { + const allRegions = ['us-east-1', 'eu-west-1', 'ap-southeast-1']; + this.logger.log(`Broadcasting ${entityId} to all regions`); + const replicationPromises = allRegions + .filter((region) => region !== this.currentRegion) + .map((region) => this.replicateToRegion(entityId, data, region)); + await Promise.all(replicationPromises); + } + /** + * Handles incoming replication data from another region. + */ + async handleIncomingReplication(event: ReplicationEvent): Promise { + this.logger.log(`Received replication for ${event.entityId} from ${event.sourceRegion}`); + // In a real app, logic to update the local database would go here. + // This might also trigger conflict resolution if the local version is different. + this.eventEmitter.emit('data.replication.received', event); } - - this.logger.log(`Replicating ${entityId} from ${this.currentRegion} to ${targetRegion}`); - - const event: ReplicationEvent = { - entityId, - sourceRegion: this.currentRegion, - targetRegion, - data, - timestamp: new Date(), - }; - - // Add to queue for asynchronous replication - await this.syncQueue.add(JOB_NAMES.REPLICATE_DATA, event, { - attempts: 5, - backoff: { - type: 'exponential', - delay: 2000, - }, - }); - - this.eventEmitter.emit('data.replication.started', event); - } - - /** - * Broadcasts data to all regions except the current one. - */ - async broadcastToAllRegions(entityId: string, data: any): Promise { - const allRegions = ['us-east-1', 'eu-west-1', 'ap-southeast-1']; - - this.logger.log(`Broadcasting ${entityId} to all regions`); - - const replicationPromises = allRegions - .filter((region) => region !== this.currentRegion) - .map((region) => this.replicateToRegion(entityId, data, region)); - - await Promise.all(replicationPromises); - } - - /** - * Handles incoming replication data from another region. - */ - async handleIncomingReplication(event: ReplicationEvent): Promise { - this.logger.log(`Received replication for ${event.entityId} from ${event.sourceRegion}`); - - // In a real app, logic to update the local database would go here. - // This might also trigger conflict resolution if the local version is different. - - this.eventEmitter.emit('data.replication.received', event); - } } diff --git a/src/sync/sync.module.ts b/src/sync/sync.module.ts index 21f09848..80649473 100644 --- a/src/sync/sync.module.ts +++ b/src/sync/sync.module.ts @@ -6,26 +6,26 @@ import { DataConsistencyService } from './consistency/data-consistency.service'; import { ConflictResolutionService } from './conflicts/conflict-resolution.service'; import { CacheInvalidationService } from './cache/cache-invalidation.service'; import { ReplicationService } from './replication/replication.service'; - @Module({ - imports: [ - BullModule.registerQueue({ - name: QUEUE_NAMES.SYNC_TASKS, - }), - ], - providers: [ - SyncService, - DataConsistencyService, - ConflictResolutionService, - CacheInvalidationService, - ReplicationService, - ], - exports: [ - SyncService, - DataConsistencyService, - ConflictResolutionService, - CacheInvalidationService, - ReplicationService, - ], + imports: [ + BullModule.registerQueue({ + name: QUEUE_NAMES.SYNC_TASKS, + }), + ], + providers: [ + SyncService, + DataConsistencyService, + ConflictResolutionService, + CacheInvalidationService, + ReplicationService, + ], + exports: [ + SyncService, + DataConsistencyService, + ConflictResolutionService, + CacheInvalidationService, + ReplicationService, + ], }) -export class SyncModule {} +export class SyncModule { +} diff --git a/src/sync/sync.service.ts b/src/sync/sync.service.ts index ee84e9cd..c2e7e221 100644 --- a/src/sync/sync.service.ts +++ b/src/sync/sync.service.ts @@ -1,60 +1,40 @@ import { Injectable, Logger } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { APP_EVENTS } from '../common/constants/event.constants'; -import { - ConflictResolutionService, - ConflictResolutionStrategy, - SyncData, -} from './conflicts/conflict-resolution.service'; +import { ConflictResolutionService, ConflictResolutionStrategy, SyncData, } from './conflicts/conflict-resolution.service'; import { DataConsistencyService } from './consistency/data-consistency.service'; import { CacheInvalidationService } from './cache/cache-invalidation.service'; import { ReplicationService } from './replication/replication.service'; - @Injectable() export class SyncService { - private readonly logger = new Logger(SyncService.name); - - constructor( - private conflictResolution: ConflictResolutionService, - private consistencyService: DataConsistencyService, - private cacheInvalidation: CacheInvalidationService, - private replicationService: ReplicationService, - ) {} - - /** - * Synchronizes data between two sources. - */ - async synchronize( - localData: SyncData, - remoteData: SyncData, - strategy: ConflictResolutionStrategy = ConflictResolutionStrategy.LAST_WRITE_WINS, - ): Promise { - this.logger.log(`Starting synchronization for ${localData.id}`); - - // Resolve any conflicts - const resolvedData = this.conflictResolution.resolve(localData, remoteData, strategy); - - // Ensure consistency - await this.consistencyService.scheduleConsistencyTask(resolvedData.id, resolvedData.data); - - // Invalidate cache - await this.cacheInvalidation.handleDataChange('entity', resolvedData.id); - - // Replicate to other regions - await this.replicationService.broadcastToAllRegions(resolvedData.id, resolvedData.data); - - this.logger.log(`Synchronization completed for ${resolvedData.id}`); - return resolvedData; - } - - @OnEvent(APP_EVENTS.DATA_UPDATED) - async handleDataUpdate(payload: { entity: string; id: string; data: any }) { - this.logger.log(`Handling data update event for ${payload.entity}:${payload.id}`); - - // Invalidate cache immediately on update - await this.cacheInvalidation.handleDataChange(payload.entity, payload.id); - - // Broadcast change - await this.replicationService.broadcastToAllRegions(payload.id, payload.data); - } + private readonly logger = new Logger(SyncService.name); + constructor(private conflictResolution: ConflictResolutionService, private consistencyService: DataConsistencyService, private cacheInvalidation: CacheInvalidationService, private replicationService: ReplicationService) { } + /** + * Synchronizes data between two sources. + */ + async synchronize(localData: SyncData, remoteData: SyncData, strategy: ConflictResolutionStrategy = ConflictResolutionStrategy.LAST_WRITE_WINS): Promise { + this.logger.log(`Starting synchronization for ${localData.id}`); + // Resolve any conflicts + const resolvedData = this.conflictResolution.resolve(localData, remoteData, strategy); + // Ensure consistency + await this.consistencyService.scheduleConsistencyTask(resolvedData.id, resolvedData.data); + // Invalidate cache + await this.cacheInvalidation.handleDataChange('entity', resolvedData.id); + // Replicate to other regions + await this.replicationService.broadcastToAllRegions(resolvedData.id, resolvedData.data); + this.logger.log(`Synchronization completed for ${resolvedData.id}`); + return resolvedData; + } + @OnEvent(APP_EVENTS.DATA_UPDATED) + async handleDataUpdate(payload: { + entity: string; + id: string; + data: unknown; + }) { + this.logger.log(`Handling data update event for ${payload.entity}:${payload.id}`); + // Invalidate cache immediately on update + await this.cacheInvalidation.handleDataChange(payload.entity, payload.id); + // Broadcast change + await this.replicationService.broadcastToAllRegions(payload.id, payload.data); + } } diff --git a/src/tenancy/admin/tenant-admin.service.ts b/src/tenancy/admin/tenant-admin.service.ts index 59d7881d..bb8b4ee5 100644 --- a/src/tenancy/admin/tenant-admin.service.ts +++ b/src/tenancy/admin/tenant-admin.service.ts @@ -6,262 +6,226 @@ import { Tenant, TenantStatus, TenantPlan } from '../entities/tenant.entity'; import { TenantConfig } from '../entities/tenant-config.entity'; import { TenantBilling } from '../entities/tenant-billing.entity'; import { TenantCustomization } from '../entities/tenant-customization.entity'; - export interface TenantStatistics { - totalUsers: number; - activeUsers: number; - storageUsed: number; - apiCalls: number; - lastActivityAt?: Date; + totalUsers: number; + activeUsers: number; + storageUsed: number; + apiCalls: number; + lastActivityAt?: Date; } - export interface TenantHealth { - status: string; - issues: string[]; - score: number; + status: string; + issues: string[]; + score: number; } - @Injectable() export class TenantAdminService { - constructor( + constructor( @InjectRepository(Tenant) - private readonly tenantRepository: Repository, + private readonly tenantRepository: Repository, @InjectRepository(TenantConfig) - private readonly configRepository: Repository, + private readonly configRepository: Repository, @InjectRepository(TenantBilling) - private readonly billingRepository: Repository, + private readonly billingRepository: Repository, @InjectRepository(TenantCustomization) - private readonly customizationRepository: Repository, - ) {} - - /** - * Get tenant statistics - */ - async getTenantStatistics(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + private readonly customizationRepository: Repository) { } + /** + * Get tenant statistics + */ + async getTenantStatistics(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + const billing = await this.billingRepository.findOne({ where: { tenantId } }); + return { + totalUsers: tenant.currentUserCount, + activeUsers: billing?.usageMetrics?.activeUsers || 0, + storageUsed: tenant.currentStorageUsage, + apiCalls: billing?.usageMetrics?.apiCalls || 0, + lastActivityAt: tenant.updatedAt, + }; } - - const billing = await this.billingRepository.findOne({ where: { tenantId } }); - - return { - totalUsers: tenant.currentUserCount, - activeUsers: billing?.usageMetrics?.activeUsers || 0, - storageUsed: tenant.currentStorageUsage, - apiCalls: billing?.usageMetrics?.apiCalls || 0, - lastActivityAt: tenant.updatedAt, - }; - } - - /** - * Suspend tenant - */ - async suspendTenant(tenantId: string, reason?: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + /** + * Suspend tenant + */ + async suspendTenant(tenantId: string, reason?: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + tenant.status = TenantStatus.SUSPENDED; + tenant.metadata = { + ...tenant.metadata, + suspensionReason: reason, + suspendedAt: new Date(), + }; + return await this.tenantRepository.save(tenant); } - - tenant.status = TenantStatus.SUSPENDED; - tenant.metadata = { - ...tenant.metadata, - suspensionReason: reason, - suspendedAt: new Date(), - }; - - return await this.tenantRepository.save(tenant); - } - - /** - * Activate tenant - */ - async activateTenant(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + /** + * Activate tenant + */ + async activateTenant(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + tenant.status = TenantStatus.ACTIVE; + tenant.metadata = { + ...tenant.metadata, + suspensionReason: undefined, + suspendedAt: undefined, + activatedAt: new Date(), + }; + return await this.tenantRepository.save(tenant); } - - tenant.status = TenantStatus.ACTIVE; - tenant.metadata = { - ...tenant.metadata, - suspensionReason: undefined, - suspendedAt: undefined, - activatedAt: new Date(), - }; - - return await this.tenantRepository.save(tenant); - } - - /** - * Upgrade tenant plan - */ - async upgradePlan(tenantId: string, newPlan: TenantPlan): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + /** + * Upgrade tenant plan + */ + async upgradePlan(tenantId: string, newPlan: TenantPlan): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + const oldPlan = tenant.plan; + tenant.plan = newPlan; + // Update limits based on plan + const limits = this.getPlanLimits(newPlan); + tenant.userLimit = limits.userLimit; + tenant.storageLimit = limits.storageLimit; + tenant.metadata = { + ...tenant.metadata, + planUpgradeHistory: [ + ...(tenant.metadata?.planUpgradeHistory || []), + { + from: oldPlan, + to: newPlan, + upgradedAt: new Date(), + }, + ], + }; + return await this.tenantRepository.save(tenant); } - - const oldPlan = tenant.plan; - tenant.plan = newPlan; - - // Update limits based on plan - const limits = this.getPlanLimits(newPlan); - tenant.userLimit = limits.userLimit; - tenant.storageLimit = limits.storageLimit; - - tenant.metadata = { - ...tenant.metadata, - planUpgradeHistory: [ - ...(tenant.metadata?.planUpgradeHistory || []), - { - from: oldPlan, - to: newPlan, - upgradedAt: new Date(), - }, - ], - }; - - return await this.tenantRepository.save(tenant); - } - - /** - * Check tenant health - */ - async checkTenantHealth(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + /** + * Check tenant health + */ + async checkTenantHealth(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + const issues: string[] = []; + let score = 100; + // Check if tenant is suspended + if (tenant.status === TenantStatus.SUSPENDED) { + issues.push('Tenant is suspended'); + score -= 50; + } + // Check if approaching user limit + const userUsagePercent = (tenant.currentUserCount / tenant.userLimit) * 100; + if (userUsagePercent > 90) { + issues.push('Approaching user limit'); + score -= 10; + } + // Check if approaching storage limit + const storageUsagePercent = (tenant.currentStorageUsage / tenant.storageLimit) * 100; + if (storageUsagePercent > 90) { + issues.push('Approaching storage limit'); + score -= 10; + } + // Check billing status + const billing = await this.billingRepository.findOne({ where: { tenantId } }); + if (billing && Number(billing.currentBalance) > 0) { + issues.push('Outstanding billing balance'); + score -= 15; + } + // Check if trial expired + if (tenant.status === TenantStatus.TRIAL && + tenant.trialEndsAt && + tenant.trialEndsAt < new Date()) { + issues.push('Trial period expired'); + score -= 20; + } + return { + status: score > 70 ? 'healthy' : score > 40 ? 'warning' : 'critical', + issues, + score, + }; } - - const issues: string[] = []; - let score = 100; - - // Check if tenant is suspended - if (tenant.status === TenantStatus.SUSPENDED) { - issues.push('Tenant is suspended'); - score -= 50; + /** + * Reset tenant data + */ + async resetTenantData(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + // Reset counters + tenant.currentUserCount = 0; + tenant.currentStorageUsage = 0; + await this.tenantRepository.save(tenant); + // Reset billing + const billing = await this.billingRepository.findOne({ where: { tenantId } }); + if (billing) { + billing.usageMetrics = {}; + await this.billingRepository.save(billing); + } } - - // Check if approaching user limit - const userUsagePercent = (tenant.currentUserCount / tenant.userLimit) * 100; - if (userUsagePercent > 90) { - issues.push('Approaching user limit'); - score -= 10; + /** + * Export tenant data + */ + async exportTenantData(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + const config = await this.configRepository.findOne({ where: { tenantId } }); + const billing = await this.billingRepository.findOne({ where: { tenantId } }); + const customization = await this.customizationRepository.findOne({ where: { tenantId } }); + return { + tenant, + config, + billing, + customization, + exportedAt: new Date(), + }; } - - // Check if approaching storage limit - const storageUsagePercent = (tenant.currentStorageUsage / tenant.storageLimit) * 100; - if (storageUsagePercent > 90) { - issues.push('Approaching storage limit'); - score -= 10; + /** + * Get all tenants with pagination + */ + async getAllTenants(page: number = 1, limit: number = 10): Promise<{ + tenants: Tenant[]; + total: number; + }> { + const [tenants, total] = await this.tenantRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + }); + return { tenants, total }; } - - // Check billing status - const billing = await this.billingRepository.findOne({ where: { tenantId } }); - if (billing && Number(billing.currentBalance) > 0) { - issues.push('Outstanding billing balance'); - score -= 15; + /** + * Search tenants + */ + async searchTenants(query: string): Promise { + const safeQuery = sanitizeSqlLike(query); + return await this.tenantRepository + .createQueryBuilder('tenant') + .where("tenant.name ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) + .orWhere("tenant.slug ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) + .orWhere("tenant.domain ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) + .getMany(); } - - // Check if trial expired - if ( - tenant.status === TenantStatus.TRIAL && - tenant.trialEndsAt && - tenant.trialEndsAt < new Date() - ) { - issues.push('Trial period expired'); - score -= 20; + /** + * Get plan limits + */ + private getPlanLimits(plan: TenantPlan): { + userLimit: number; + storageLimit: number; + } { + const limits = { + [TenantPlan.FREE]: { userLimit: 10, storageLimit: 1024 }, // 1GB + [TenantPlan.BASIC]: { userLimit: 50, storageLimit: 10240 }, // 10GB + [TenantPlan.PROFESSIONAL]: { userLimit: 200, storageLimit: 51200 }, // 50GB + [TenantPlan.ENTERPRISE]: { userLimit: -1, storageLimit: -1 }, // Unlimited + }; + return limits[plan] || limits[TenantPlan.FREE]; } - - return { - status: score > 70 ? 'healthy' : score > 40 ? 'warning' : 'critical', - issues, - score, - }; - } - - /** - * Reset tenant data - */ - async resetTenantData(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); - } - - // Reset counters - tenant.currentUserCount = 0; - tenant.currentStorageUsage = 0; - await this.tenantRepository.save(tenant); - - // Reset billing - const billing = await this.billingRepository.findOne({ where: { tenantId } }); - if (billing) { - billing.usageMetrics = {}; - await this.billingRepository.save(billing); - } - } - - /** - * Export tenant data - */ - async exportTenantData(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - const config = await this.configRepository.findOne({ where: { tenantId } }); - const billing = await this.billingRepository.findOne({ where: { tenantId } }); - const customization = await this.customizationRepository.findOne({ where: { tenantId } }); - - return { - tenant, - config, - billing, - customization, - exportedAt: new Date(), - }; - } - - /** - * Get all tenants with pagination - */ - async getAllTenants( - page: number = 1, - limit: number = 10, - ): Promise<{ tenants: Tenant[]; total: number }> { - const [tenants, total] = await this.tenantRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - }); - - return { tenants, total }; - } - - /** - * Search tenants - */ - async searchTenants(query: string): Promise { - const safeQuery = sanitizeSqlLike(query); - - return await this.tenantRepository - .createQueryBuilder('tenant') - .where("tenant.name ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) - .orWhere("tenant.slug ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) - .orWhere("tenant.domain ILIKE :query ESCAPE '\\'", { query: `%${safeQuery}%` }) - .getMany(); - } - - /** - * Get plan limits - */ - private getPlanLimits(plan: TenantPlan): { userLimit: number; storageLimit: number } { - const limits = { - [TenantPlan.FREE]: { userLimit: 10, storageLimit: 1024 }, // 1GB - [TenantPlan.BASIC]: { userLimit: 50, storageLimit: 10240 }, // 10GB - [TenantPlan.PROFESSIONAL]: { userLimit: 200, storageLimit: 51200 }, // 50GB - [TenantPlan.ENTERPRISE]: { userLimit: -1, storageLimit: -1 }, // Unlimited - }; - - return limits[plan] || limits[TenantPlan.FREE]; - } } diff --git a/src/tenancy/billing/tenant-billing.service.ts b/src/tenancy/billing/tenant-billing.service.ts index 6a50a4bc..90a333cb 100644 --- a/src/tenancy/billing/tenant-billing.service.ts +++ b/src/tenancy/billing/tenant-billing.service.ts @@ -3,214 +3,173 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { TenantBilling, BillingCycle } from '../entities/tenant-billing.entity'; import { Tenant } from '../entities/tenant.entity'; - export interface UsageMetrics { - activeUsers?: number; - storageUsed?: number; - apiCalls?: number; - bandwidth?: number; - [key: string]: any; + activeUsers?: number; + storageUsed?: number; + apiCalls?: number; + bandwidth?: number; + [key: string]: unknown; } - export interface BillingRecord { - date: Date; - amount: number; - status: string; - invoiceId?: string; + date: Date; + amount: number; + status: string; + invoiceId?: string; } - @Injectable() export class TenantBillingService { - constructor( + constructor( @InjectRepository(TenantBilling) - private readonly billingRepository: Repository, + private readonly billingRepository: Repository, @InjectRepository(Tenant) - private readonly tenantRepository: Repository, - ) {} - - /** - * Get billing information for a tenant - */ - async getBillingInfo(tenantId: string): Promise { - const billing = await this.billingRepository.findOne({ where: { tenantId } }); - if (!billing) { - throw new NotFoundException(`Billing info not found for tenant ${tenantId}`); + private readonly tenantRepository: Repository) { } + /** + * Get billing information for a tenant + */ + async getBillingInfo(tenantId: string): Promise { + const billing = await this.billingRepository.findOne({ where: { tenantId } }); + if (!billing) { + throw new NotFoundException(`Billing info not found for tenant ${tenantId}`); + } + return billing; } - return billing; - } - - /** - * Create billing record for a tenant - */ - async createBillingRecord( - tenantId: string, - billingCycle: BillingCycle = BillingCycle.MONTHLY, - ): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant ${tenantId} not found`); + /** + * Create billing record for a tenant + */ + async createBillingRecord(tenantId: string, billingCycle: BillingCycle = BillingCycle.MONTHLY): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant ${tenantId} not found`); + } + const billing = this.billingRepository.create({ + tenantId, + billingCycle, + monthlyFee: this.calculateMonthlyFee(tenant.plan), + nextBillingDate: this.calculateNextBillingDate(billingCycle), + }); + return await this.billingRepository.save(billing); } - - const billing = this.billingRepository.create({ - tenantId, - billingCycle, - monthlyFee: this.calculateMonthlyFee(tenant.plan), - nextBillingDate: this.calculateNextBillingDate(billingCycle), - }); - - return await this.billingRepository.save(billing); - } - - /** - * Update usage metrics - */ - async updateUsageMetrics(tenantId: string, metrics: UsageMetrics): Promise { - const billing = await this.getBillingInfo(tenantId); - - billing.usageMetrics = { - ...billing.usageMetrics, - ...metrics, - }; - - return await this.billingRepository.save(billing); - } - - /** - * Record a payment - */ - async recordPayment( - tenantId: string, - amount: number, - invoiceId?: string, - ): Promise { - const billing = await this.getBillingInfo(tenantId); - - const billingRecord: BillingRecord = { - date: new Date(), - amount, - status: 'paid', - invoiceId, - }; - - billing.billingHistory = billing.billingHistory || []; - billing.billingHistory.push(billingRecord); - billing.totalPaid = Number(billing.totalPaid) + amount; - billing.currentBalance = Number(billing.currentBalance) - amount; - billing.lastBillingDate = new Date(); - - return await this.billingRepository.save(billing); - } - - /** - * Generate invoice - */ - async generateInvoice(tenantId: string): Promise { - const billing = await this.getBillingInfo(tenantId); - const amount = Number(billing.monthlyFee); - - const invoice: BillingRecord = { - date: new Date(), - amount, - status: 'pending', - invoiceId: `INV-${tenantId}-${Date.now()}`, - }; - - billing.currentBalance = Number(billing.currentBalance) + amount; - billing.billingHistory = billing.billingHistory || []; - billing.billingHistory.push(invoice); - billing.nextBillingDate = this.calculateNextBillingDate(billing.billingCycle); - - await this.billingRepository.save(billing); - - return invoice; - } - - /** - * Update billing cycle - */ - async updateBillingCycle(tenantId: string, billingCycle: BillingCycle): Promise { - const billing = await this.getBillingInfo(tenantId); - billing.billingCycle = billingCycle; - billing.nextBillingDate = this.calculateNextBillingDate(billingCycle); - return await this.billingRepository.save(billing); - } - - /** - * Get billing history - */ - async getBillingHistory(tenantId: string): Promise { - const billing = await this.getBillingInfo(tenantId); - return billing.billingHistory || []; - } - - /** - * Calculate current usage cost - */ - async calculateUsageCost(tenantId: string): Promise { - const billing = await this.getBillingInfo(tenantId); - const metrics = billing.usageMetrics || {}; - - let cost = 0; - - // Example pricing logic (customize as needed) - cost += (metrics.activeUsers || 0) * 5; // $5 per active user - cost += ((metrics.storageUsed || 0) / 1024) * 0.1; // $0.10 per GB - cost += ((metrics.apiCalls || 0) / 1000) * 0.01; // $0.01 per 1000 API calls - - return cost; - } - - /** - * Check if tenant has outstanding balance - */ - async hasOutstandingBalance(tenantId: string): Promise { - const billing = await this.getBillingInfo(tenantId); - return Number(billing.currentBalance) > 0; - } - - /** - * Update Stripe customer ID - */ - async updateStripeCustomer( - tenantId: string, - customerId: string, - subscriptionId?: string, - ): Promise { - const billing = await this.getBillingInfo(tenantId); - billing.stripeCustomerId = customerId; - if (subscriptionId) { - billing.stripeSubscriptionId = subscriptionId; + /** + * Update usage metrics + */ + async updateUsageMetrics(tenantId: string, metrics: UsageMetrics): Promise { + const billing = await this.getBillingInfo(tenantId); + billing.usageMetrics = { + ...billing.usageMetrics, + ...metrics, + }; + return await this.billingRepository.save(billing); } - return await this.billingRepository.save(billing); - } - - /** - * Calculate monthly fee based on plan - */ - private calculateMonthlyFee(plan: string): number { - const pricing = { - free: 0, - basic: 29, - professional: 99, - enterprise: 299, - }; - return pricing[plan] || 0; - } - - /** - * Calculate next billing date based on cycle - */ - private calculateNextBillingDate(cycle: BillingCycle): Date { - const now = new Date(); - switch (cycle) { - case BillingCycle.MONTHLY: - return new Date(now.setMonth(now.getMonth() + 1)); - case BillingCycle.QUARTERLY: - return new Date(now.setMonth(now.getMonth() + 3)); - case BillingCycle.YEARLY: - return new Date(now.setFullYear(now.getFullYear() + 1)); - default: - return new Date(now.setMonth(now.getMonth() + 1)); + /** + * Record a payment + */ + async recordPayment(tenantId: string, amount: number, invoiceId?: string): Promise { + const billing = await this.getBillingInfo(tenantId); + const billingRecord: BillingRecord = { + date: new Date(), + amount, + status: 'paid', + invoiceId, + }; + billing.billingHistory = billing.billingHistory || []; + billing.billingHistory.push(billingRecord); + billing.totalPaid = Number(billing.totalPaid) + amount; + billing.currentBalance = Number(billing.currentBalance) - amount; + billing.lastBillingDate = new Date(); + return await this.billingRepository.save(billing); + } + /** + * Generate invoice + */ + async generateInvoice(tenantId: string): Promise { + const billing = await this.getBillingInfo(tenantId); + const amount = Number(billing.monthlyFee); + const invoice: BillingRecord = { + date: new Date(), + amount, + status: 'pending', + invoiceId: `INV-${tenantId}-${Date.now()}`, + }; + billing.currentBalance = Number(billing.currentBalance) + amount; + billing.billingHistory = billing.billingHistory || []; + billing.billingHistory.push(invoice); + billing.nextBillingDate = this.calculateNextBillingDate(billing.billingCycle); + await this.billingRepository.save(billing); + return invoice; + } + /** + * Update billing cycle + */ + async updateBillingCycle(tenantId: string, billingCycle: BillingCycle): Promise { + const billing = await this.getBillingInfo(tenantId); + billing.billingCycle = billingCycle; + billing.nextBillingDate = this.calculateNextBillingDate(billingCycle); + return await this.billingRepository.save(billing); + } + /** + * Get billing history + */ + async getBillingHistory(tenantId: string): Promise { + const billing = await this.getBillingInfo(tenantId); + return billing.billingHistory || []; + } + /** + * Calculate current usage cost + */ + async calculateUsageCost(tenantId: string): Promise { + const billing = await this.getBillingInfo(tenantId); + const metrics = billing.usageMetrics || {}; + let cost = 0; + // Example pricing logic (customize as needed) + cost += (metrics.activeUsers || 0) * 5; // $5 per active user + cost += ((metrics.storageUsed || 0) / 1024) * 0.1; // $0.10 per GB + cost += ((metrics.apiCalls || 0) / 1000) * 0.01; // $0.01 per 1000 API calls + return cost; + } + /** + * Check if tenant has outstanding balance + */ + async hasOutstandingBalance(tenantId: string): Promise { + const billing = await this.getBillingInfo(tenantId); + return Number(billing.currentBalance) > 0; + } + /** + * Update Stripe customer ID + */ + async updateStripeCustomer(tenantId: string, customerId: string, subscriptionId?: string): Promise { + const billing = await this.getBillingInfo(tenantId); + billing.stripeCustomerId = customerId; + if (subscriptionId) { + billing.stripeSubscriptionId = subscriptionId; + } + return await this.billingRepository.save(billing); + } + /** + * Calculate monthly fee based on plan + */ + private calculateMonthlyFee(plan: string): number { + const pricing = { + free: 0, + basic: 29, + professional: 99, + enterprise: 299, + }; + return pricing[plan] || 0; + } + /** + * Calculate next billing date based on cycle + */ + private calculateNextBillingDate(cycle: BillingCycle): Date { + const now = new Date(); + switch (cycle) { + case BillingCycle.MONTHLY: + return new Date(now.setMonth(now.getMonth() + 1)); + case BillingCycle.QUARTERLY: + return new Date(now.setMonth(now.getMonth() + 3)); + case BillingCycle.YEARLY: + return new Date(now.setFullYear(now.getFullYear() + 1)); + default: + return new Date(now.setMonth(now.getMonth() + 1)); + } } - } } diff --git a/src/tenancy/customization/customization.service.ts b/src/tenancy/customization/customization.service.ts index 02534003..49d31957 100644 --- a/src/tenancy/customization/customization.service.ts +++ b/src/tenancy/customization/customization.service.ts @@ -3,197 +3,166 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { TenantCustomization } from '../entities/tenant-customization.entity'; import { UpdateTenantCustomizationDto } from '../dto/tenant.dto'; - @Injectable() export class CustomizationService { - constructor( + constructor( @InjectRepository(TenantCustomization) - private readonly customizationRepository: Repository, - ) {} - - /** - * Get customization for a tenant - */ - async getCustomization(tenantId: string): Promise { - const customization = await this.customizationRepository.findOne({ where: { tenantId } }); - if (!customization) { - throw new NotFoundException(`Customization not found for tenant ${tenantId}`); + private readonly customizationRepository: Repository) { } + /** + * Get customization for a tenant + */ + async getCustomization(tenantId: string): Promise { + const customization = await this.customizationRepository.findOne({ where: { tenantId } }); + if (!customization) { + throw new NotFoundException(`Customization not found for tenant ${tenantId}`); + } + return customization; + } + /** + * Create default customization for a tenant + */ + async createDefaultCustomization(tenantId: string): Promise { + const customization = this.customizationRepository.create({ + tenantId, + theme: { + mode: 'light', + colors: {}, + fonts: {}, + spacing: {}, + }, + }); + return await this.customizationRepository.save(customization); + } + /** + * Update tenant customization + */ + async updateCustomization(tenantId: string, updateDto: UpdateTenantCustomizationDto): Promise { + const customization = await this.getCustomization(tenantId); + Object.assign(customization, updateDto); + return await this.customizationRepository.save(customization); + } + /** + * Update logo + */ + async updateLogo(tenantId: string, logoUrl: string): Promise { + const customization = await this.getCustomization(tenantId); + customization.logoUrl = logoUrl; + return await this.customizationRepository.save(customization); + } + /** + * Update theme colors + */ + async updateColors(tenantId: string, colors: { + primary?: string; + secondary?: string; + accent?: string; + }): Promise { + const customization = await this.getCustomization(tenantId); + if (colors.primary) + customization.primaryColor = colors.primary; + if (colors.secondary) + customization.secondaryColor = colors.secondary; + if (colors.accent) + customization.accentColor = colors.accent; + return await this.customizationRepository.save(customization); + } + /** + * Update theme configuration + */ + async updateTheme(tenantId: string, theme: Record): Promise { + const customization = await this.getCustomization(tenantId); + customization.theme = { + ...customization.theme, + ...theme, + }; + return await this.customizationRepository.save(customization); + } + /** + * Update email templates + */ + async updateEmailTemplates(tenantId: string, templates: Record): Promise { + const customization = await this.getCustomization(tenantId); + customization.emailTemplates = { + ...customization.emailTemplates, + ...templates, + }; + return await this.customizationRepository.save(customization); + } + /** + * Set custom domain + */ + async setCustomDomain(tenantId: string, domain: string): Promise { + const customization = await this.getCustomization(tenantId); + customization.customDomain = domain; + customization.customDomainVerified = false; + return await this.customizationRepository.save(customization); + } + /** + * Verify custom domain + */ + async verifyCustomDomain(tenantId: string): Promise { + const customization = await this.getCustomization(tenantId); + // TODO: Implement actual domain verification logic + customization.customDomainVerified = true; + return await this.customizationRepository.save(customization); + } + /** + * Update social links + */ + async updateSocialLinks(tenantId: string, socialLinks: Record): Promise { + const customization = await this.getCustomization(tenantId); + customization.socialLinks = { + ...customization.socialLinks, + ...socialLinks, + }; + return await this.customizationRepository.save(customization); + } + /** + * Update landing page configuration + */ + async updateLandingPage(tenantId: string, config: Record): Promise { + const customization = await this.getCustomization(tenantId); + customization.landingPageConfig = { + ...customization.landingPageConfig, + ...config, + }; + return await this.customizationRepository.save(customization); + } + /** + * Add custom CSS + */ + async addCustomCss(tenantId: string, css: string): Promise { + const customization = await this.getCustomization(tenantId); + customization.customCss = css; + return await this.customizationRepository.save(customization); + } + /** + * Add custom JavaScript + */ + async addCustomJs(tenantId: string, js: string): Promise { + const customization = await this.getCustomization(tenantId); + customization.customJs = js; + return await this.customizationRepository.save(customization); + } + /** + * Reset customization to defaults + */ + async resetToDefaults(tenantId: string): Promise { + const customization = await this.getCustomization(tenantId); + customization.logoUrl = null; + customization.faviconUrl = null; + customization.primaryColor = null; + customization.secondaryColor = null; + customization.accentColor = null; + customization.fontFamily = null; + customization.customCss = null; + customization.customJs = null; + customization.theme = { + mode: 'light', + colors: {}, + fonts: {}, + spacing: {}, + }; + return await this.customizationRepository.save(customization); } - return customization; - } - - /** - * Create default customization for a tenant - */ - async createDefaultCustomization(tenantId: string): Promise { - const customization = this.customizationRepository.create({ - tenantId, - theme: { - mode: 'light', - colors: {}, - fonts: {}, - spacing: {}, - }, - }); - - return await this.customizationRepository.save(customization); - } - - /** - * Update tenant customization - */ - async updateCustomization( - tenantId: string, - updateDto: UpdateTenantCustomizationDto, - ): Promise { - const customization = await this.getCustomization(tenantId); - - Object.assign(customization, updateDto); - - return await this.customizationRepository.save(customization); - } - - /** - * Update logo - */ - async updateLogo(tenantId: string, logoUrl: string): Promise { - const customization = await this.getCustomization(tenantId); - customization.logoUrl = logoUrl; - return await this.customizationRepository.save(customization); - } - - /** - * Update theme colors - */ - async updateColors( - tenantId: string, - colors: { primary?: string; secondary?: string; accent?: string }, - ): Promise { - const customization = await this.getCustomization(tenantId); - - if (colors.primary) customization.primaryColor = colors.primary; - if (colors.secondary) customization.secondaryColor = colors.secondary; - if (colors.accent) customization.accentColor = colors.accent; - - return await this.customizationRepository.save(customization); - } - - /** - * Update theme configuration - */ - async updateTheme(tenantId: string, theme: Record): Promise { - const customization = await this.getCustomization(tenantId); - customization.theme = { - ...customization.theme, - ...theme, - }; - return await this.customizationRepository.save(customization); - } - - /** - * Update email templates - */ - async updateEmailTemplates( - tenantId: string, - templates: Record, - ): Promise { - const customization = await this.getCustomization(tenantId); - customization.emailTemplates = { - ...customization.emailTemplates, - ...templates, - }; - return await this.customizationRepository.save(customization); - } - - /** - * Set custom domain - */ - async setCustomDomain(tenantId: string, domain: string): Promise { - const customization = await this.getCustomization(tenantId); - customization.customDomain = domain; - customization.customDomainVerified = false; - return await this.customizationRepository.save(customization); - } - - /** - * Verify custom domain - */ - async verifyCustomDomain(tenantId: string): Promise { - const customization = await this.getCustomization(tenantId); - // TODO: Implement actual domain verification logic - customization.customDomainVerified = true; - return await this.customizationRepository.save(customization); - } - - /** - * Update social links - */ - async updateSocialLinks( - tenantId: string, - socialLinks: Record, - ): Promise { - const customization = await this.getCustomization(tenantId); - customization.socialLinks = { - ...customization.socialLinks, - ...socialLinks, - }; - return await this.customizationRepository.save(customization); - } - - /** - * Update landing page configuration - */ - async updateLandingPage( - tenantId: string, - config: Record, - ): Promise { - const customization = await this.getCustomization(tenantId); - customization.landingPageConfig = { - ...customization.landingPageConfig, - ...config, - }; - return await this.customizationRepository.save(customization); - } - - /** - * Add custom CSS - */ - async addCustomCss(tenantId: string, css: string): Promise { - const customization = await this.getCustomization(tenantId); - customization.customCss = css; - return await this.customizationRepository.save(customization); - } - - /** - * Add custom JavaScript - */ - async addCustomJs(tenantId: string, js: string): Promise { - const customization = await this.getCustomization(tenantId); - customization.customJs = js; - return await this.customizationRepository.save(customization); - } - - /** - * Reset customization to defaults - */ - async resetToDefaults(tenantId: string): Promise { - const customization = await this.getCustomization(tenantId); - - customization.logoUrl = null; - customization.faviconUrl = null; - customization.primaryColor = null; - customization.secondaryColor = null; - customization.accentColor = null; - customization.fontFamily = null; - customization.customCss = null; - customization.customJs = null; - customization.theme = { - mode: 'light', - colors: {}, - fonts: {}, - spacing: {}, - }; - - return await this.customizationRepository.save(customization); - } } diff --git a/src/tenancy/decorators/current-tenant.decorator.ts b/src/tenancy/decorators/current-tenant.decorator.ts index 5e1f5378..28e92544 100644 --- a/src/tenancy/decorators/current-tenant.decorator.ts +++ b/src/tenancy/decorators/current-tenant.decorator.ts @@ -1,6 +1,5 @@ import { createParamDecorator, ExecutionContext } from '@nestjs/common'; - export const CurrentTenant = createParamDecorator((data: unknown, ctx: ExecutionContext) => { - const request = ctx.switchToHttp().getRequest(); - return request.tenant; + const request = ctx.switchToHttp().getRequest(); + return request.tenant; }); diff --git a/src/tenancy/decorators/requires-tenant.decorator.ts b/src/tenancy/decorators/requires-tenant.decorator.ts index 41ea42f9..4ca8d5ea 100644 --- a/src/tenancy/decorators/requires-tenant.decorator.ts +++ b/src/tenancy/decorators/requires-tenant.decorator.ts @@ -1,4 +1,3 @@ import { SetMetadata } from '@nestjs/common'; - export const TENANT_KEY = 'tenant'; export const RequiresTenant = () => SetMetadata(TENANT_KEY, true); diff --git a/src/tenancy/dto/tenant.dto.ts b/src/tenancy/dto/tenant.dto.ts index 91a62edf..53ee0778 100644 --- a/src/tenancy/dto/tenant.dto.ts +++ b/src/tenancy/dto/tenant.dto.ts @@ -1,258 +1,205 @@ -import { - IsString, - IsOptional, - IsEmail, - IsEnum, - IsInt, - IsObject, - Min, - IsNotEmpty, - IsNumber, - IsUrl, -} from 'class-validator'; +import { IsString, IsOptional, IsEmail, IsEnum, IsInt, IsObject, Min, IsNotEmpty, IsNumber, IsUrl, } from 'class-validator'; import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; import { TenantStatus, TenantPlan } from '../entities/tenant.entity'; - export class CreateTenantDto { - @ApiProperty({ example: 'acme-corp' }) - @IsString() - @IsNotEmpty() - slug: string; - - @ApiProperty({ example: 'Acme Corporation' }) - @IsString() - @IsNotEmpty() - name: string; - - @ApiPropertyOptional({ example: 'Leading provider of innovative solutions' }) - @IsOptional() - @IsString() - description?: string; - - @ApiPropertyOptional({ example: 'acme.com' }) - @IsOptional() - @IsString() - domain?: string; - - @ApiPropertyOptional({ enum: TenantPlan, default: TenantPlan.FREE }) - @IsOptional() - @IsEnum(TenantPlan) - plan?: TenantPlan; - - @ApiPropertyOptional({ example: 'owner@acme.com' }) - @IsOptional() - @IsEmail() - @IsString() - ownerEmail?: string; - - @ApiPropertyOptional({ example: 'contact@acme.com' }) - @IsOptional() - @IsEmail() - @IsString() - contactEmail?: string; - - @ApiPropertyOptional({ example: '+1234567890' }) - @IsOptional() - @IsString() - contactPhone?: string; - - @ApiPropertyOptional({ example: 100 }) - @IsOptional() - @IsInt() - @Min(0) - @IsNumber() - userLimit?: number; - - @ApiPropertyOptional({ example: 10000 }) - @IsOptional() - @IsInt() - @Min(0) - @IsNumber() - storageLimit?: number; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - metadata?: Record; + @ApiProperty({ example: 'acme-corp' }) + @IsString() + @IsNotEmpty() + slug: string; + @ApiProperty({ example: 'Acme Corporation' }) + @IsString() + @IsNotEmpty() + name: string; + @ApiPropertyOptional({ example: 'Leading provider of innovative solutions' }) + @IsOptional() + @IsString() + description?: string; + @ApiPropertyOptional({ example: 'acme.com' }) + @IsOptional() + @IsString() + domain?: string; + @ApiPropertyOptional({ enum: TenantPlan, default: TenantPlan.FREE }) + @IsOptional() + @IsEnum(TenantPlan) + plan?: TenantPlan; + @ApiPropertyOptional({ example: 'owner@acme.com' }) + @IsOptional() + @IsEmail() + @IsString() + ownerEmail?: string; + @ApiPropertyOptional({ example: 'contact@acme.com' }) + @IsOptional() + @IsEmail() + @IsString() + contactEmail?: string; + @ApiPropertyOptional({ example: '+1234567890' }) + @IsOptional() + @IsString() + contactPhone?: string; + @ApiPropertyOptional({ example: 100 }) + @IsOptional() + @IsInt() + @Min(0) + @IsNumber() + userLimit?: number; + @ApiPropertyOptional({ example: 10000 }) + @IsOptional() + @IsInt() + @Min(0) + @IsNumber() + storageLimit?: number; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + metadata?: Record; } - export class UpdateTenantDto { - @ApiPropertyOptional({ example: 'Acme Corporation' }) - @IsOptional() - @IsString() - name?: string; - - @ApiPropertyOptional({ example: 'Leading provider of innovative solutions' }) - @IsOptional() - @IsString() - description?: string; - - @ApiPropertyOptional({ example: 'acme.com' }) - @IsOptional() - @IsString() - domain?: string; - - @ApiPropertyOptional({ enum: TenantStatus }) - @IsOptional() - @IsEnum(TenantStatus) - status?: TenantStatus; - - @ApiPropertyOptional({ enum: TenantPlan }) - @IsOptional() - @IsEnum(TenantPlan) - plan?: TenantPlan; - - @ApiPropertyOptional({ example: 'contact@acme.com' }) - @IsOptional() - @IsEmail() - @IsString() - contactEmail?: string; - - @ApiPropertyOptional({ example: '+1234567890' }) - @IsOptional() - @IsString() - contactPhone?: string; - - @ApiPropertyOptional({ example: 100 }) - @IsOptional() - @IsInt() - @Min(0) - @IsNumber() - userLimit?: number; - - @ApiPropertyOptional({ example: 10000 }) - @IsOptional() - @IsInt() - @Min(0) - @IsNumber() - storageLimit?: number; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - metadata?: Record; + @ApiPropertyOptional({ example: 'Acme Corporation' }) + @IsOptional() + @IsString() + name?: string; + @ApiPropertyOptional({ example: 'Leading provider of innovative solutions' }) + @IsOptional() + @IsString() + description?: string; + @ApiPropertyOptional({ example: 'acme.com' }) + @IsOptional() + @IsString() + domain?: string; + @ApiPropertyOptional({ enum: TenantStatus }) + @IsOptional() + @IsEnum(TenantStatus) + status?: TenantStatus; + @ApiPropertyOptional({ enum: TenantPlan }) + @IsOptional() + @IsEnum(TenantPlan) + plan?: TenantPlan; + @ApiPropertyOptional({ example: 'contact@acme.com' }) + @IsOptional() + @IsEmail() + @IsString() + contactEmail?: string; + @ApiPropertyOptional({ example: '+1234567890' }) + @IsOptional() + @IsString() + contactPhone?: string; + @ApiPropertyOptional({ example: 100 }) + @IsOptional() + @IsInt() + @Min(0) + @IsNumber() + userLimit?: number; + @ApiPropertyOptional({ example: 10000 }) + @IsOptional() + @IsInt() + @Min(0) + @IsNumber() + storageLimit?: number; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + metadata?: Record; } - export class UpdateTenantConfigDto { - @ApiPropertyOptional({ example: 'en' }) - @IsOptional() - @IsString() - defaultLanguage?: string; - - @ApiPropertyOptional({ example: 'UTC' }) - @IsOptional() - @IsString() - timezone?: string; - - @ApiPropertyOptional({ example: 'USD' }) - @IsOptional() - @IsString() - currency?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - features?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - notifications?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - security?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - integrations?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - customSettings?: Record; + @ApiPropertyOptional({ example: 'en' }) + @IsOptional() + @IsString() + defaultLanguage?: string; + @ApiPropertyOptional({ example: 'UTC' }) + @IsOptional() + @IsString() + timezone?: string; + @ApiPropertyOptional({ example: 'USD' }) + @IsOptional() + @IsString() + currency?: string; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + features?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + notifications?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + security?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + integrations?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + customSettings?: Record; } - export class UpdateTenantCustomizationDto { - @ApiPropertyOptional({ example: 'https://example.com/logo.png' }) - @IsOptional() - @IsString() - @IsUrl() - logoUrl?: string; - - @ApiPropertyOptional({ example: 'https://example.com/favicon.ico' }) - @IsOptional() - @IsString() - @IsUrl() - faviconUrl?: string; - - @ApiPropertyOptional({ example: '#007bff' }) - @IsOptional() - @IsString() - primaryColor?: string; - - @ApiPropertyOptional({ example: '#6c757d' }) - @IsOptional() - @IsString() - secondaryColor?: string; - - @ApiPropertyOptional({ example: '#28a745' }) - @IsOptional() - @IsString() - accentColor?: string; - - @ApiPropertyOptional({ example: 'Roboto, sans-serif' }) - @IsOptional() - @IsString() - fontFamily?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - theme?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsString() - customCss?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsString() - customJs?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsEmail() - @IsString() - emailTemplates?: Record; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsString() - landingPageConfig?: Record; - - @ApiPropertyOptional({ example: 'custom.acme.com' }) - @IsOptional() - @IsString() - customDomain?: string; - - @ApiPropertyOptional() - @IsOptional() - @IsObject() - @IsUrl() - @IsString() - socialLinks?: Record; + @ApiPropertyOptional({ example: 'https://example.com/logo.png' }) + @IsOptional() + @IsString() + @IsUrl() + logoUrl?: string; + @ApiPropertyOptional({ example: 'https://example.com/favicon.ico' }) + @IsOptional() + @IsString() + @IsUrl() + faviconUrl?: string; + @ApiPropertyOptional({ example: '#007bff' }) + @IsOptional() + @IsString() + primaryColor?: string; + @ApiPropertyOptional({ example: '#6c757d' }) + @IsOptional() + @IsString() + secondaryColor?: string; + @ApiPropertyOptional({ example: '#28a745' }) + @IsOptional() + @IsString() + accentColor?: string; + @ApiPropertyOptional({ example: 'Roboto, sans-serif' }) + @IsOptional() + @IsString() + fontFamily?: string; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + theme?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsString() + customCss?: string; + @ApiPropertyOptional() + @IsOptional() + @IsString() + customJs?: string; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsEmail() + @IsString() + emailTemplates?: Record; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsString() + landingPageConfig?: Record; + @ApiPropertyOptional({ example: 'custom.acme.com' }) + @IsOptional() + @IsString() + customDomain?: string; + @ApiPropertyOptional() + @IsOptional() + @IsObject() + @IsUrl() + @IsString() + socialLinks?: Record; } diff --git a/src/tenancy/entities/tenant-billing.entity.ts b/src/tenancy/entities/tenant-billing.entity.ts index db40a3a3..cd936bef 100644 --- a/src/tenancy/entities/tenant-billing.entity.ts +++ b/src/tenancy/entities/tenant-billing.entity.ts @@ -1,92 +1,63 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { Tenant } from './tenant.entity'; - export enum BillingCycle { - MONTHLY = 'monthly', - QUARTERLY = 'quarterly', - YEARLY = 'yearly', + MONTHLY = 'monthly', + QUARTERLY = 'quarterly', + YEARLY = 'yearly' } - @Entity('tenant_billing') export class TenantBilling { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - tenantId: string; - - @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'tenantId' }) - tenant: Tenant; - - @Column({ - type: 'enum', - enum: BillingCycle, - default: BillingCycle.MONTHLY, - }) - billingCycle: BillingCycle; - - @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) - monthlyFee: number; - - @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) - currentBalance: number; - - @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) - totalPaid: number; - - @Column({ type: 'timestamp', nullable: true }) - lastBillingDate?: Date; - - @Column({ type: 'timestamp', nullable: true }) - nextBillingDate?: Date; - - @Column({ nullable: true }) - stripeCustomerId?: string; - - @Column({ nullable: true }) - stripeSubscriptionId?: string; - - @Column({ type: 'jsonb', nullable: true }) - usageMetrics?: { - activeUsers?: number; - storageUsed?: number; - apiCalls?: number; - bandwidth?: number; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - billingHistory?: Array<{ - date: Date; - amount: number; - status: string; - invoiceId?: string; - }>; - - @Column({ default: true }) - autoRenew: boolean; - - @Column({ nullable: true }) - paymentMethod?: string; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + tenantId: string; + @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'tenantId' }) + tenant: Tenant; + @Column({ + type: 'enum', + enum: BillingCycle, + default: BillingCycle.MONTHLY, + }) + billingCycle: BillingCycle; + @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) + monthlyFee: number; + @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) + currentBalance: number; + @Column({ type: 'decimal', precision: 10, scale: 2, default: 0 }) + totalPaid: number; + @Column({ type: 'timestamp', nullable: true }) + lastBillingDate?: Date; + @Column({ type: 'timestamp', nullable: true }) + nextBillingDate?: Date; + @Column({ nullable: true }) + stripeCustomerId?: string; + @Column({ nullable: true }) + stripeSubscriptionId?: string; + @Column({ type: 'jsonb', nullable: true }) + usageMetrics?: { + activeUsers?: number; + storageUsed?: number; + apiCalls?: number; + bandwidth?: number; + [key: string]: unknown; + }; + @Column({ type: 'jsonb', nullable: true }) + billingHistory?: Array<{ + date: Date; + amount: number; + status: string; + invoiceId?: string; + }>; + @Column({ default: true }) + autoRenew: boolean; + @Column({ nullable: true }) + paymentMethod?: string; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/tenancy/entities/tenant-config.entity.ts b/src/tenancy/entities/tenant-config.entity.ts index bbb7767b..4aea3a85 100644 --- a/src/tenancy/entities/tenant-config.entity.ts +++ b/src/tenancy/entities/tenant-config.entity.ts @@ -1,86 +1,70 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { Tenant } from './tenant.entity'; - @Entity('tenant_configs') export class TenantConfig { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - tenantId: string; - - @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'tenantId' }) - tenant: Tenant; - - @Column({ default: 'en' }) - defaultLanguage: string; - - @Column({ default: 'UTC' }) - timezone: string; - - @Column({ default: 'USD' }) - currency: string; - - @Column({ type: 'jsonb', nullable: true }) - features?: { - analytics?: boolean; - messaging?: boolean; - courses?: boolean; - assessments?: boolean; - recommendations?: boolean; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - notifications?: { - email?: boolean; - push?: boolean; - sms?: boolean; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - security?: { - mfaRequired?: boolean; - passwordPolicy?: { - minLength?: number; - requireNumbers?: boolean; - requireSpecialChars?: boolean; - requireUppercase?: boolean; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + tenantId: string; + @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'tenantId' }) + tenant: Tenant; + @Column({ default: 'en' }) + defaultLanguage: string; + @Column({ default: 'UTC' }) + timezone: string; + @Column({ default: 'USD' }) + currency: string; + @Column({ type: 'jsonb', nullable: true }) + features?: { + analytics?: boolean; + messaging?: boolean; + courses?: boolean; + assessments?: boolean; + recommendations?: boolean; + [key: string]: unknown; }; - sessionTimeout?: number; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - integrations?: { - stripe?: { enabled: boolean; publicKey?: string }; - aws?: { enabled: boolean; region?: string }; - openai?: { enabled: boolean }; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - customSettings?: Record; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @Column({ type: 'jsonb', nullable: true }) + notifications?: { + email?: boolean; + push?: boolean; + sms?: boolean; + [key: string]: unknown; + }; + @Column({ type: 'jsonb', nullable: true }) + security?: { + mfaRequired?: boolean; + passwordPolicy?: { + minLength?: number; + requireNumbers?: boolean; + requireSpecialChars?: boolean; + requireUppercase?: boolean; + }; + sessionTimeout?: number; + [key: string]: unknown; + }; + @Column({ type: 'jsonb', nullable: true }) + integrations?: { + stripe?: { + enabled: boolean; + publicKey?: string; + }; + aws?: { + enabled: boolean; + region?: string; + }; + openai?: { + enabled: boolean; + }; + [key: string]: unknown; + }; + @Column({ type: 'jsonb', nullable: true }) + customSettings?: Record; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/tenancy/entities/tenant-customization.entity.ts b/src/tenancy/entities/tenant-customization.entity.ts index 0a710d84..ecff360b 100644 --- a/src/tenancy/entities/tenant-customization.entity.ts +++ b/src/tenancy/entities/tenant-customization.entity.ts @@ -1,106 +1,76 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - ManyToOne, - JoinColumn, - Index, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, ManyToOne, JoinColumn, Index, } from 'typeorm'; import { Tenant } from './tenant.entity'; - @Entity('tenant_customizations') export class TenantCustomization { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column() - @Index() - tenantId: string; - - @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) - @JoinColumn({ name: 'tenantId' }) - tenant: Tenant; - - @Column({ nullable: true }) - logoUrl?: string; - - @Column({ nullable: true }) - faviconUrl?: string; - - @Column({ nullable: true }) - primaryColor?: string; - - @Column({ nullable: true }) - secondaryColor?: string; - - @Column({ nullable: true }) - accentColor?: string; - - @Column({ nullable: true }) - fontFamily?: string; - - @Column({ type: 'jsonb', nullable: true }) - theme?: { - mode?: 'light' | 'dark' | 'auto'; - colors?: Record; - fonts?: Record; - spacing?: Record; - [key: string]: any; - }; - - @Column({ type: 'text', nullable: true }) - customCss?: string; - - @Column({ type: 'text', nullable: true }) - customJs?: string; - - @Column({ type: 'jsonb', nullable: true }) - emailTemplates?: { - welcome?: string; - passwordReset?: string; - notification?: string; - [key: string]: any; - }; - - @Column({ type: 'jsonb', nullable: true }) - landingPageConfig?: { - hero?: { - title?: string; - subtitle?: string; - backgroundImage?: string; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column() + @Index() + tenantId: string; + @ManyToOne(() => Tenant, { onDelete: 'CASCADE' }) + @JoinColumn({ name: 'tenantId' }) + tenant: Tenant; + @Column({ nullable: true }) + logoUrl?: string; + @Column({ nullable: true }) + faviconUrl?: string; + @Column({ nullable: true }) + primaryColor?: string; + @Column({ nullable: true }) + secondaryColor?: string; + @Column({ nullable: true }) + accentColor?: string; + @Column({ nullable: true }) + fontFamily?: string; + @Column({ type: 'jsonb', nullable: true }) + theme?: { + mode?: 'light' | 'dark' | 'auto'; + colors?: Record; + fonts?: Record; + spacing?: Record; + [key: string]: unknown; }; - features?: Array<{ - title: string; - description: string; - icon?: string; - }>; - [key: string]: any; - }; - - @Column({ nullable: true }) - customDomain?: string; - - @Column({ default: false }) - customDomainVerified: boolean; - - @Column({ type: 'jsonb', nullable: true }) - socialLinks?: { - facebook?: string; - twitter?: string; - linkedin?: string; - instagram?: string; - [key: string]: string; - }; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @Column({ type: 'text', nullable: true }) + customCss?: string; + @Column({ type: 'text', nullable: true }) + customJs?: string; + @Column({ type: 'jsonb', nullable: true }) + emailTemplates?: { + welcome?: string; + passwordReset?: string; + notification?: string; + [key: string]: unknown; + }; + @Column({ type: 'jsonb', nullable: true }) + landingPageConfig?: { + hero?: { + title?: string; + subtitle?: string; + backgroundImage?: string; + }; + features?: Array<{ + title: string; + description: string; + icon?: string; + }>; + [key: string]: unknown; + }; + @Column({ nullable: true }) + customDomain?: string; + @Column({ default: false }) + customDomainVerified: boolean; + @Column({ type: 'jsonb', nullable: true }) + socialLinks?: { + facebook?: string; + twitter?: string; + linkedin?: string; + instagram?: string; + [key: string]: string; + }; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/tenancy/entities/tenant.entity.ts b/src/tenancy/entities/tenant.entity.ts index a1a6647a..58eff6ef 100644 --- a/src/tenancy/entities/tenant.entity.ts +++ b/src/tenancy/entities/tenant.entity.ts @@ -1,101 +1,69 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - Index, -} from 'typeorm'; - +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, Index, } from 'typeorm'; export enum TenantStatus { - ACTIVE = 'active', - SUSPENDED = 'suspended', - INACTIVE = 'inactive', - TRIAL = 'trial', + ACTIVE = 'active', + SUSPENDED = 'suspended', + INACTIVE = 'inactive', + TRIAL = 'trial' } - export enum TenantPlan { - FREE = 'free', - BASIC = 'basic', - PROFESSIONAL = 'professional', - ENTERPRISE = 'enterprise', + FREE = 'free', + BASIC = 'basic', + PROFESSIONAL = 'professional', + ENTERPRISE = 'enterprise' } - @Entity('tenants') export class Tenant { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ unique: true }) - @Index() - slug: string; - - @Column() - name: string; - - @Column({ nullable: true }) - description?: string; - - @Column({ nullable: true }) - domain?: string; - - @Column({ - type: 'enum', - enum: TenantStatus, - default: TenantStatus.TRIAL, - }) - status: TenantStatus; - - @Column({ - type: 'enum', - enum: TenantPlan, - default: TenantPlan.FREE, - }) - plan: TenantPlan; - - @Column({ nullable: true }) - ownerId?: string; - - @Column({ nullable: true }) - ownerEmail?: string; - - @Column({ nullable: true }) - contactEmail?: string; - - @Column({ nullable: true }) - contactPhone?: string; - - @Column({ type: 'jsonb', nullable: true }) - metadata?: Record; - - @Column({ type: 'int', default: 0 }) - userLimit: number; - - @Column({ type: 'int', default: 0 }) - storageLimit: number; // in MB - - @Column({ type: 'int', default: 0 }) - currentUserCount: number; - - @Column({ type: 'int', default: 0 }) - currentStorageUsage: number; // in MB - - @Column({ type: 'timestamp', nullable: true }) - trialEndsAt?: Date; - - @Column({ type: 'timestamp', nullable: true }) - subscriptionStartsAt?: Date; - - @Column({ type: 'timestamp', nullable: true }) - subscriptionEndsAt?: Date; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ unique: true }) + @Index() + slug: string; + @Column() + name: string; + @Column({ nullable: true }) + description?: string; + @Column({ nullable: true }) + domain?: string; + @Column({ + type: 'enum', + enum: TenantStatus, + default: TenantStatus.TRIAL, + }) + status: TenantStatus; + @Column({ + type: 'enum', + enum: TenantPlan, + default: TenantPlan.FREE, + }) + plan: TenantPlan; + @Column({ nullable: true }) + ownerId?: string; + @Column({ nullable: true }) + ownerEmail?: string; + @Column({ nullable: true }) + contactEmail?: string; + @Column({ nullable: true }) + contactPhone?: string; + @Column({ type: 'jsonb', nullable: true }) + metadata?: Record; + @Column({ type: 'int', default: 0 }) + userLimit: number; + @Column({ type: 'int', default: 0 }) + storageLimit: number; // in MB + @Column({ type: 'int', default: 0 }) + currentUserCount: number; + @Column({ type: 'int', default: 0 }) + currentStorageUsage: number; // in MB + @Column({ type: 'timestamp', nullable: true }) + trialEndsAt?: Date; + @Column({ type: 'timestamp', nullable: true }) + subscriptionStartsAt?: Date; + @Column({ type: 'timestamp', nullable: true }) + subscriptionEndsAt?: Date; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/tenancy/guards/tenant.guard.ts b/src/tenancy/guards/tenant.guard.ts index aadefcb3..d6bf1970 100644 --- a/src/tenancy/guards/tenant.guard.ts +++ b/src/tenancy/guards/tenant.guard.ts @@ -2,51 +2,41 @@ import { Injectable, CanActivate, ExecutionContext, UnauthorizedException } from import { Reflector } from '@nestjs/core'; import { TENANT_KEY } from '../decorators/requires-tenant.decorator'; import { IsolationService } from '../isolation/isolation.service'; - @Injectable() export class TenantGuard implements CanActivate { - constructor( - private reflector: Reflector, - private isolationService: IsolationService, - ) {} - - async canActivate(context: ExecutionContext): Promise { - const requiresTenant = this.reflector.getAllAndOverride(TENANT_KEY, [ - context.getHandler(), - context.getClass(), - ]); - - if (!requiresTenant) { - return true; + constructor(private reflector: Reflector, private isolationService: IsolationService) { } + async canActivate(context: ExecutionContext): Promise { + const requiresTenant = this.reflector.getAllAndOverride(TENANT_KEY, [ + context.getHandler(), + context.getClass(), + ]); + if (!requiresTenant) { + return true; + } + const request = context.switchToHttp().getRequest(); + // Try to get tenant from various sources + const tenantId = request.headers['x-tenant-id'] || request.query.tenantId || request.user?.tenantId; + const tenantSlug = request.headers['x-tenant-slug'] || request.query.tenantSlug; + const tenantDomain = request.headers['x-tenant-domain'] || request.hostname; + try { + if (tenantId) { + await this.isolationService.setTenant(tenantId); + } + else if (tenantSlug) { + await this.isolationService.setTenantBySlug(tenantSlug); + } + else if (tenantDomain) { + await this.isolationService.setTenantByDomain(tenantDomain); + } + else { + throw new UnauthorizedException('Tenant context is required but not provided'); + } + // Store tenant in request for later use + request.tenant = this.isolationService.getTenant(); + return true; + } + catch (error) { + throw new UnauthorizedException(`Failed to set tenant context: ${error.message}`); + } } - - const request = context.switchToHttp().getRequest(); - - // Try to get tenant from various sources - const tenantId = - request.headers['x-tenant-id'] || request.query.tenantId || request.user?.tenantId; - - const tenantSlug = request.headers['x-tenant-slug'] || request.query.tenantSlug; - - const tenantDomain = request.headers['x-tenant-domain'] || request.hostname; - - try { - if (tenantId) { - await this.isolationService.setTenant(tenantId); - } else if (tenantSlug) { - await this.isolationService.setTenantBySlug(tenantSlug); - } else if (tenantDomain) { - await this.isolationService.setTenantByDomain(tenantDomain); - } else { - throw new UnauthorizedException('Tenant context is required but not provided'); - } - - // Store tenant in request for later use - request.tenant = this.isolationService.getTenant(); - - return true; - } catch (error) { - throw new UnauthorizedException(`Failed to set tenant context: ${error.message}`); - } - } } diff --git a/src/tenancy/isolation/isolation.service.ts b/src/tenancy/isolation/isolation.service.ts index 8c6a5211..4a03f088 100644 --- a/src/tenancy/isolation/isolation.service.ts +++ b/src/tenancy/isolation/isolation.service.ts @@ -2,145 +2,130 @@ import { Injectable, Scope, NotFoundException } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { Tenant } from '../entities/tenant.entity'; - /** * IsolationService manages tenant context and data isolation * This service is request-scoped to maintain tenant context per request */ @Injectable({ scope: Scope.REQUEST }) export class IsolationService { - private currentTenantId: string | null = null; - private currentTenant: Tenant | null = null; - - constructor( + private currentTenantId: string | null = null; + private currentTenant: Tenant | null = null; + constructor( @InjectRepository(Tenant) - private readonly tenantRepository: Repository, - ) {} - - /** - * Set the current tenant context - */ - async setTenant(tenantId: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); - if (!tenant) { - throw new NotFoundException(`Tenant with ID ${tenantId} not found`); + private readonly tenantRepository: Repository) { } + /** + * Set the current tenant context + */ + async setTenant(tenantId: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id: tenantId } }); + if (!tenant) { + throw new NotFoundException(`Tenant with ID ${tenantId} not found`); + } + this.currentTenantId = tenantId; + this.currentTenant = tenant; } - this.currentTenantId = tenantId; - this.currentTenant = tenant; - } - - /** - * Set tenant by slug - */ - async setTenantBySlug(slug: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { slug } }); - if (!tenant) { - throw new NotFoundException(`Tenant with slug ${slug} not found`); + /** + * Set tenant by slug + */ + async setTenantBySlug(slug: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { slug } }); + if (!tenant) { + throw new NotFoundException(`Tenant with slug ${slug} not found`); + } + this.currentTenantId = tenant.id; + this.currentTenant = tenant; } - this.currentTenantId = tenant.id; - this.currentTenant = tenant; - } - - /** - * Set tenant by domain - */ - async setTenantByDomain(domain: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { domain } }); - if (!tenant) { - throw new NotFoundException(`Tenant with domain ${domain} not found`); + /** + * Set tenant by domain + */ + async setTenantByDomain(domain: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { domain } }); + if (!tenant) { + throw new NotFoundException(`Tenant with domain ${domain} not found`); + } + this.currentTenantId = tenant.id; + this.currentTenant = tenant; } - this.currentTenantId = tenant.id; - this.currentTenant = tenant; - } - - /** - * Get the current tenant ID - */ - getTenantId(): string | null { - return this.currentTenantId; - } - - /** - * Get the current tenant - */ - getTenant(): Tenant | null { - return this.currentTenant; - } - - /** - * Check if tenant context is set - */ - hasTenantContext(): boolean { - return this.currentTenantId !== null; - } - - /** - * Clear tenant context - */ - clearTenant(): void { - this.currentTenantId = null; - this.currentTenant = null; - } - - /** - * Ensure tenant context is set, throw error if not - */ - ensureTenantContext(): void { - if (!this.hasTenantContext()) { - throw new Error('Tenant context is not set'); + /** + * Get the current tenant ID + */ + getTenantId(): string | null { + return this.currentTenantId; } - } - - /** - * Add tenant filter to query builder - */ - applyTenantFilter(queryBuilder: any, entityAlias: string): any { - if (!this.currentTenantId) { - throw new Error('Cannot apply tenant filter without tenant context'); + /** + * Get the current tenant + */ + getTenant(): Tenant | null { + return this.currentTenant; } - return queryBuilder.andWhere(`${entityAlias}.tenantId = :tenantId`, { - tenantId: this.currentTenantId, - }); - } - - /** - * Check if tenant is active - */ - isActiveTenant(): boolean { - return this.currentTenant?.status === 'active'; - } - - /** - * Check if tenant is in trial - */ - isTrialTenant(): boolean { - return this.currentTenant?.status === 'trial'; - } - - /** - * Check if tenant has reached user limit - */ - hasReachedUserLimit(): boolean { - if (!this.currentTenant) return false; - return this.currentTenant.currentUserCount >= this.currentTenant.userLimit; - } - - /** - * Check if tenant has reached storage limit - */ - hasReachedStorageLimit(): boolean { - if (!this.currentTenant) return false; - return this.currentTenant.currentStorageUsage >= this.currentTenant.storageLimit; - } - - /** - * Get tenant feature flags - */ - async getTenantFeatures(): Promise> { - if (!this.currentTenant) { - return {}; + /** + * Check if tenant context is set + */ + hasTenantContext(): boolean { + return this.currentTenantId !== null; + } + /** + * Clear tenant context + */ + clearTenant(): void { + this.currentTenantId = null; + this.currentTenant = null; + } + /** + * Ensure tenant context is set, throw error if not + */ + ensureTenantContext(): void { + if (!this.hasTenantContext()) { + throw new Error('Tenant context is not set'); + } + } + /** + * Add tenant filter to query builder + */ + applyTenantFilter(queryBuilder: unknown, entityAlias: string): unknown { + if (!this.currentTenantId) { + throw new Error('Cannot apply tenant filter without tenant context'); + } + return queryBuilder.andWhere(`${entityAlias}.tenantId = :tenantId`, { + tenantId: this.currentTenantId, + }); + } + /** + * Check if tenant is active + */ + isActiveTenant(): boolean { + return this.currentTenant?.status === 'active'; + } + /** + * Check if tenant is in trial + */ + isTrialTenant(): boolean { + return this.currentTenant?.status === 'trial'; + } + /** + * Check if tenant has reached user limit + */ + hasReachedUserLimit(): boolean { + if (!this.currentTenant) + return false; + return this.currentTenant.currentUserCount >= this.currentTenant.userLimit; + } + /** + * Check if tenant has reached storage limit + */ + hasReachedStorageLimit(): boolean { + if (!this.currentTenant) + return false; + return this.currentTenant.currentStorageUsage >= this.currentTenant.storageLimit; + } + /** + * Get tenant feature flags + */ + async getTenantFeatures(): Promise> { + if (!this.currentTenant) { + return {}; + } + // This would typically fetch from TenantConfig + return this.currentTenant.metadata?.features || {}; } - // This would typically fetch from TenantConfig - return this.currentTenant.metadata?.features || {}; - } } diff --git a/src/tenancy/tenancy.controller.ts b/src/tenancy/tenancy.controller.ts index eba6e926..cadac2f2 100644 --- a/src/tenancy/tenancy.controller.ts +++ b/src/tenancy/tenancy.controller.ts @@ -1,187 +1,196 @@ -import { - Controller, - Get, - Post, - Body, - Patch, - Param, - Delete, - Query, - UseGuards, -} from '@nestjs/common'; +import { Controller, Get, Post, Body, Patch, Param, Delete, Query, UseGuards, } from '@nestjs/common'; import { ApiBearerAuth, ApiTags, ApiOperation, ApiQuery } from '@nestjs/swagger'; import { TenancyService } from './tenancy.service'; import { TenantAdminService } from './admin/tenant-admin.service'; import { TenantBillingService } from './billing/tenant-billing.service'; import { CustomizationService } from './customization/customization.service'; -import { - CreateTenantDto, - UpdateTenantDto, - UpdateTenantConfigDto, - UpdateTenantCustomizationDto, -} from './dto/tenant.dto'; +import { CreateTenantDto, UpdateTenantDto, UpdateTenantConfigDto, UpdateTenantCustomizationDto, } from './dto/tenant.dto'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; import { RolesGuard } from '../auth/guards/roles.guard'; import { Roles } from '../auth/decorators/roles.decorator'; import { UserRole } from '../users/entities/user.entity'; import { TenantPlan } from './entities/tenant.entity'; - @ApiTags('tenancy') @Controller('tenants') @UseGuards(JwtAuthGuard, RolesGuard) @ApiBearerAuth() export class TenancyController { - constructor( - private readonly tenancyService: TenancyService, - private readonly adminService: TenantAdminService, - private readonly billingService: TenantBillingService, - private readonly customizationService: CustomizationService, - ) {} - - @Post() - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Create a new tenant (Admin only)' }) - create(@Body() createTenantDto: CreateTenantDto) { - return this.tenancyService.create(createTenantDto); - } - - @Get() - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Get all tenants (Admin only)' }) - @ApiQuery({ name: 'page', required: false, type: Number }) - @ApiQuery({ name: 'limit', required: false, type: Number }) - findAll(@Query('page') page?: number, @Query('limit') limit?: number) { - return this.tenancyService.findAll(page, limit); - } - - @Get('search') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Search tenants (Admin only)' }) - @ApiQuery({ name: 'q', required: true }) - search(@Query('q') query: string) { - return this.adminService.searchTenants(query); - } - - @Get(':id') - @ApiOperation({ summary: 'Get tenant by ID' }) - findOne(@Param('id') id: string) { - return this.tenancyService.findOne(id); - } - - @Get(':id/full') - @ApiOperation({ summary: 'Get tenant with all related data' }) - getTenantWithRelations(@Param('id') id: string) { - return this.tenancyService.getTenantWithRelations(id); - } - - @Get(':id/statistics') - @ApiOperation({ summary: 'Get tenant statistics' }) - getStatistics(@Param('id') id: string) { - return this.adminService.getTenantStatistics(id); - } - - @Get(':id/health') - @ApiOperation({ summary: 'Check tenant health' }) - checkHealth(@Param('id') id: string) { - return this.adminService.checkTenantHealth(id); - } - - @Patch(':id') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Update tenant (Admin only)' }) - update(@Param('id') id: string, @Body() updateTenantDto: UpdateTenantDto) { - return this.tenancyService.update(id, updateTenantDto); - } - - @Delete(':id') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Delete tenant (Admin only)' }) - remove(@Param('id') id: string) { - return this.tenancyService.remove(id); - } - - // Configuration endpoints - @Get(':id/config') - @ApiOperation({ summary: 'Get tenant configuration' }) - getConfig(@Param('id') id: string) { - return this.tenancyService.getConfig(id); - } - - @Patch(':id/config') - @ApiOperation({ summary: 'Update tenant configuration' }) - updateConfig(@Param('id') id: string, @Body() updateConfigDto: UpdateTenantConfigDto) { - return this.tenancyService.updateConfig(id, updateConfigDto); - } - - // Billing endpoints - @Get(':id/billing') - @ApiOperation({ summary: 'Get tenant billing information' }) - getBilling(@Param('id') id: string) { - return this.billingService.getBillingInfo(id); - } - - @Get(':id/billing/history') - @ApiOperation({ summary: 'Get tenant billing history' }) - getBillingHistory(@Param('id') id: string) { - return this.billingService.getBillingHistory(id); - } - - @Post(':id/billing/invoice') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Generate invoice for tenant (Admin only)' }) - generateInvoice(@Param('id') id: string) { - return this.billingService.generateInvoice(id); - } - - // Customization endpoints - @Get(':id/customization') - @ApiOperation({ summary: 'Get tenant customization' }) - getCustomization(@Param('id') id: string) { - return this.customizationService.getCustomization(id); - } - - @Patch(':id/customization') - @ApiOperation({ summary: 'Update tenant customization' }) - updateCustomization( - @Param('id') id: string, - @Body() updateCustomizationDto: UpdateTenantCustomizationDto, - ) { - return this.customizationService.updateCustomization(id, updateCustomizationDto); - } - - // Admin operations - @Post(':id/suspend') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Suspend tenant (Admin only)' }) - suspend(@Param('id') id: string, @Body('reason') reason?: string) { - return this.adminService.suspendTenant(id, reason); - } - - @Post(':id/activate') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Activate tenant (Admin only)' }) - activate(@Param('id') id: string) { - return this.adminService.activateTenant(id); - } - - @Post(':id/upgrade') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Upgrade tenant plan (Admin only)' }) - upgradePlan(@Param('id') id: string, @Body('plan') plan: TenantPlan) { - return this.adminService.upgradePlan(id, plan); - } - - @Post(':id/reset-data') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Reset tenant data (Admin only)' }) - resetData(@Param('id') id: string) { - return this.adminService.resetTenantData(id); - } - - @Get(':id/export') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Export tenant data (Admin only)' }) - exportData(@Param('id') id: string) { - return this.adminService.exportTenantData(id); - } + constructor(private readonly tenancyService: TenancyService, private readonly adminService: TenantAdminService, private readonly billingService: TenantBillingService, private readonly customizationService: CustomizationService) { } + @Post() + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Create a new tenant (Admin only)' }) + create( + @Body() + createTenantDto: CreateTenantDto) { + return this.tenancyService.create(createTenantDto); + } + @Get() + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Get all tenants (Admin only)' }) + @ApiQuery({ name: 'page', required: false, type: Number }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + findAll( + @Query('page') + page?: number, + @Query('limit') + limit?: number) { + return this.tenancyService.findAll(page, limit); + } + @Get('search') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Search tenants (Admin only)' }) + @ApiQuery({ name: 'q', required: true }) + search( + @Query('q') + query: string) { + return this.adminService.searchTenants(query); + } + @Get(':id') + @ApiOperation({ summary: 'Get tenant by ID' }) + findOne( + @Param('id') + id: string) { + return this.tenancyService.findOne(id); + } + @Get(':id/full') + @ApiOperation({ summary: 'Get tenant with all related data' }) + getTenantWithRelations( + @Param('id') + id: string) { + return this.tenancyService.getTenantWithRelations(id); + } + @Get(':id/statistics') + @ApiOperation({ summary: 'Get tenant statistics' }) + getStatistics( + @Param('id') + id: string) { + return this.adminService.getTenantStatistics(id); + } + @Get(':id/health') + @ApiOperation({ summary: 'Check tenant health' }) + checkHealth( + @Param('id') + id: string) { + return this.adminService.checkTenantHealth(id); + } + @Patch(':id') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Update tenant (Admin only)' }) + update( + @Param('id') + id: string, + @Body() + updateTenantDto: UpdateTenantDto) { + return this.tenancyService.update(id, updateTenantDto); + } + @Delete(':id') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Delete tenant (Admin only)' }) + remove( + @Param('id') + id: string) { + return this.tenancyService.remove(id); + } + // Configuration endpoints + @Get(':id/config') + @ApiOperation({ summary: 'Get tenant configuration' }) + getConfig( + @Param('id') + id: string) { + return this.tenancyService.getConfig(id); + } + @Patch(':id/config') + @ApiOperation({ summary: 'Update tenant configuration' }) + updateConfig( + @Param('id') + id: string, + @Body() + updateConfigDto: UpdateTenantConfigDto) { + return this.tenancyService.updateConfig(id, updateConfigDto); + } + // Billing endpoints + @Get(':id/billing') + @ApiOperation({ summary: 'Get tenant billing information' }) + getBilling( + @Param('id') + id: string) { + return this.billingService.getBillingInfo(id); + } + @Get(':id/billing/history') + @ApiOperation({ summary: 'Get tenant billing history' }) + getBillingHistory( + @Param('id') + id: string) { + return this.billingService.getBillingHistory(id); + } + @Post(':id/billing/invoice') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Generate invoice for tenant (Admin only)' }) + generateInvoice( + @Param('id') + id: string) { + return this.billingService.generateInvoice(id); + } + // Customization endpoints + @Get(':id/customization') + @ApiOperation({ summary: 'Get tenant customization' }) + getCustomization( + @Param('id') + id: string) { + return this.customizationService.getCustomization(id); + } + @Patch(':id/customization') + @ApiOperation({ summary: 'Update tenant customization' }) + updateCustomization( + @Param('id') + id: string, + @Body() + updateCustomizationDto: UpdateTenantCustomizationDto) { + return this.customizationService.updateCustomization(id, updateCustomizationDto); + } + // Admin operations + @Post(':id/suspend') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Suspend tenant (Admin only)' }) + suspend( + @Param('id') + id: string, + @Body('reason') + reason?: string) { + return this.adminService.suspendTenant(id, reason); + } + @Post(':id/activate') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Activate tenant (Admin only)' }) + activate( + @Param('id') + id: string) { + return this.adminService.activateTenant(id); + } + @Post(':id/upgrade') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Upgrade tenant plan (Admin only)' }) + upgradePlan( + @Param('id') + id: string, + @Body('plan') + plan: TenantPlan) { + return this.adminService.upgradePlan(id, plan); + } + @Post(':id/reset-data') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Reset tenant data (Admin only)' }) + resetData( + @Param('id') + id: string) { + return this.adminService.resetTenantData(id); + } + @Get(':id/export') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Export tenant data (Admin only)' }) + exportData( + @Param('id') + id: string) { + return this.adminService.exportTenantData(id); + } } diff --git a/src/tenancy/tenancy.module.ts b/src/tenancy/tenancy.module.ts index c2441d29..24cddffa 100644 --- a/src/tenancy/tenancy.module.ts +++ b/src/tenancy/tenancy.module.ts @@ -11,25 +11,25 @@ import { TenantBillingService } from './billing/tenant-billing.service'; import { CustomizationService } from './customization/customization.service'; import { TenantAdminService } from './admin/tenant-admin.service'; import { TenantGuard } from './guards/tenant.guard'; - @Module({ - imports: [TypeOrmModule.forFeature([Tenant, TenantConfig, TenantBilling, TenantCustomization])], - controllers: [TenancyController], - providers: [ - TenancyService, - IsolationService, - TenantBillingService, - CustomizationService, - TenantAdminService, - TenantGuard, - ], - exports: [ - TenancyService, - IsolationService, - TenantBillingService, - CustomizationService, - TenantAdminService, - TenantGuard, - ], + imports: [TypeOrmModule.forFeature([Tenant, TenantConfig, TenantBilling, TenantCustomization])], + controllers: [TenancyController], + providers: [ + TenancyService, + IsolationService, + TenantBillingService, + CustomizationService, + TenantAdminService, + TenantGuard, + ], + exports: [ + TenancyService, + IsolationService, + TenantBillingService, + CustomizationService, + TenantAdminService, + TenantGuard, + ], }) -export class TenancyModule {} +export class TenancyModule { +} diff --git a/src/tenancy/tenancy.service.ts b/src/tenancy/tenancy.service.ts index f39e2728..ea88023b 100644 --- a/src/tenancy/tenancy.service.ts +++ b/src/tenancy/tenancy.service.ts @@ -8,236 +8,204 @@ import { TenantCustomization } from './entities/tenant-customization.entity'; import { CreateTenantDto, UpdateTenantDto, UpdateTenantConfigDto } from './dto/tenant.dto'; import { TenantBillingService } from './billing/tenant-billing.service'; import { CustomizationService } from './customization/customization.service'; - @Injectable() export class TenancyService { - constructor( + constructor( @InjectRepository(Tenant) - private readonly tenantRepository: Repository, + private readonly tenantRepository: Repository, @InjectRepository(TenantConfig) - private readonly configRepository: Repository, + private readonly configRepository: Repository, @InjectRepository(TenantBilling) - private readonly billingRepository: Repository, + private readonly billingRepository: Repository, @InjectRepository(TenantCustomization) - private readonly customizationRepository: Repository, - private readonly billingService: TenantBillingService, - private readonly customizationService: CustomizationService, - ) {} - - /** - * Create a new tenant - */ - async create(createTenantDto: CreateTenantDto): Promise { - // Check if slug already exists - const existingTenant = await this.tenantRepository.findOne({ - where: { slug: createTenantDto.slug }, - withDeleted: true, - }); - - if (existingTenant) { - throw new ConflictException('Tenant with this slug already exists'); + private readonly customizationRepository: Repository, private readonly billingService: TenantBillingService, private readonly customizationService: CustomizationService) { } + /** + * Create a new tenant + */ + async create(createTenantDto: CreateTenantDto): Promise { + // Check if slug already exists + const existingTenant = await this.tenantRepository.findOne({ + where: { slug: createTenantDto.slug }, + withDeleted: true, + }); + if (existingTenant) { + throw new ConflictException('Tenant with this slug already exists'); + } + // Create tenant + const tenant = this.tenantRepository.create({ + ...createTenantDto, + userLimit: createTenantDto.userLimit || 10, + storageLimit: createTenantDto.storageLimit || 1024, + }); + const savedTenant = await this.tenantRepository.save(tenant); + // Create related records + await Promise.all([ + this.createDefaultConfig(savedTenant.id), + this.billingService.createBillingRecord(savedTenant.id), + this.customizationService.createDefaultCustomization(savedTenant.id), + ]); + return savedTenant; } - - // Create tenant - const tenant = this.tenantRepository.create({ - ...createTenantDto, - userLimit: createTenantDto.userLimit || 10, - storageLimit: createTenantDto.storageLimit || 1024, - }); - - const savedTenant = await this.tenantRepository.save(tenant); - - // Create related records - await Promise.all([ - this.createDefaultConfig(savedTenant.id), - this.billingService.createBillingRecord(savedTenant.id), - this.customizationService.createDefaultCustomization(savedTenant.id), - ]); - - return savedTenant; - } - - /** - * Find all tenants - */ - async findAll( - page: number = 1, - limit: number = 10, - ): Promise<{ tenants: Tenant[]; total: number; page: number; totalPages: number }> { - const [tenants, total] = await this.tenantRepository.findAndCount({ - skip: (page - 1) * limit, - take: limit, - order: { createdAt: 'DESC' }, - }); - - return { - tenants, - total, - page, - totalPages: Math.ceil(total / limit), - }; - } - - /** - * Find tenant by ID - */ - async findOne(id: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { id } }); - if (!tenant) { - throw new NotFoundException(`Tenant with ID ${id} not found`); + /** + * Find all tenants + */ + async findAll(page: number = 1, limit: number = 10): Promise<{ + tenants: Tenant[]; + total: number; + page: number; + totalPages: number; + }> { + const [tenants, total] = await this.tenantRepository.findAndCount({ + skip: (page - 1) * limit, + take: limit, + order: { createdAt: 'DESC' }, + }); + return { + tenants, + total, + page, + totalPages: Math.ceil(total / limit), + }; } - return tenant; - } - - /** - * Find tenant by slug - */ - async findBySlug(slug: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { slug } }); - if (!tenant) { - throw new NotFoundException(`Tenant with slug ${slug} not found`); + /** + * Find tenant by ID + */ + async findOne(id: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { id } }); + if (!tenant) { + throw new NotFoundException(`Tenant with ID ${id} not found`); + } + return tenant; } - return tenant; - } - - /** - * Find tenant by domain - */ - async findByDomain(domain: string): Promise { - const tenant = await this.tenantRepository.findOne({ where: { domain } }); - if (!tenant) { - throw new NotFoundException(`Tenant with domain ${domain} not found`); + /** + * Find tenant by slug + */ + async findBySlug(slug: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { slug } }); + if (!tenant) { + throw new NotFoundException(`Tenant with slug ${slug} not found`); + } + return tenant; } - return tenant; - } - - /** - * Update tenant - */ - async update(id: string, updateTenantDto: UpdateTenantDto): Promise { - const tenant = await this.findOne(id); - - Object.assign(tenant, updateTenantDto); - - return await this.tenantRepository.save(tenant); - } - - /** - * Delete tenant - */ - async remove(id: string): Promise { - await this.findOne(id); - - await this.tenantRepository.manager.transaction(async (manager) => { - await manager.getRepository(TenantConfig).softDelete({ tenantId: id }); - await manager.getRepository(TenantBilling).softDelete({ tenantId: id }); - await manager.getRepository(TenantCustomization).softDelete({ tenantId: id }); - await manager.getRepository(Tenant).softDelete(id); - }); - } - - /** - * Get tenant configuration - */ - async getConfig(tenantId: string): Promise { - const config = await this.configRepository.findOne({ where: { tenantId } }); - if (!config) { - throw new NotFoundException(`Config not found for tenant ${tenantId}`); + /** + * Find tenant by domain + */ + async findByDomain(domain: string): Promise { + const tenant = await this.tenantRepository.findOne({ where: { domain } }); + if (!tenant) { + throw new NotFoundException(`Tenant with domain ${domain} not found`); + } + return tenant; + } + /** + * Update tenant + */ + async update(id: string, updateTenantDto: UpdateTenantDto): Promise { + const tenant = await this.findOne(id); + Object.assign(tenant, updateTenantDto); + return await this.tenantRepository.save(tenant); + } + /** + * Delete tenant + */ + async remove(id: string): Promise { + await this.findOne(id); + await this.tenantRepository.manager.transaction(async (manager) => { + await manager.getRepository(TenantConfig).softDelete({ tenantId: id }); + await manager.getRepository(TenantBilling).softDelete({ tenantId: id }); + await manager.getRepository(TenantCustomization).softDelete({ tenantId: id }); + await manager.getRepository(Tenant).softDelete(id); + }); + } + /** + * Get tenant configuration + */ + async getConfig(tenantId: string): Promise { + const config = await this.configRepository.findOne({ where: { tenantId } }); + if (!config) { + throw new NotFoundException(`Config not found for tenant ${tenantId}`); + } + return config; + } + /** + * Update tenant configuration + */ + async updateConfig(tenantId: string, updateConfigDto: UpdateTenantConfigDto): Promise { + const config = await this.getConfig(tenantId); + Object.assign(config, updateConfigDto); + return await this.configRepository.save(config); + } + /** + * Create default configuration + */ + private async createDefaultConfig(tenantId: string): Promise { + const config = this.configRepository.create({ + tenantId, + features: { + analytics: true, + messaging: true, + courses: true, + assessments: true, + recommendations: true, + }, + notifications: { + email: true, + push: true, + sms: false, + }, + security: { + mfaRequired: false, + passwordPolicy: { + minLength: 8, + requireNumbers: true, + requireSpecialChars: true, + requireUppercase: true, + }, + sessionTimeout: 3600, + }, + }); + return await this.configRepository.save(config); + } + /** + * Increment user count + */ + async incrementUserCount(tenantId: string): Promise { + await this.tenantRepository.increment({ id: tenantId }, 'currentUserCount', 1); + } + /** + * Decrement user count + */ + async decrementUserCount(tenantId: string): Promise { + await this.tenantRepository.decrement({ id: tenantId }, 'currentUserCount', 1); + } + /** + * Update storage usage + */ + async updateStorageUsage(tenantId: string, sizeInMB: number): Promise { + const tenant = await this.findOne(tenantId); + tenant.currentStorageUsage += sizeInMB; + await this.tenantRepository.save(tenant); + } + /** + * Get tenant with all related data + */ + async getTenantWithRelations(tenantId: string): Promise<{ + tenant: Tenant; + config: TenantConfig; + billing: TenantBilling; + customization: TenantCustomization; + }> { + const [tenant, config, billing, customization] = await Promise.all([ + this.findOne(tenantId), + this.getConfig(tenantId), + this.billingService.getBillingInfo(tenantId), + this.customizationService.getCustomization(tenantId), + ]); + return { + tenant, + config, + billing, + customization, + }; } - return config; - } - - /** - * Update tenant configuration - */ - async updateConfig( - tenantId: string, - updateConfigDto: UpdateTenantConfigDto, - ): Promise { - const config = await this.getConfig(tenantId); - - Object.assign(config, updateConfigDto); - - return await this.configRepository.save(config); - } - - /** - * Create default configuration - */ - private async createDefaultConfig(tenantId: string): Promise { - const config = this.configRepository.create({ - tenantId, - features: { - analytics: true, - messaging: true, - courses: true, - assessments: true, - recommendations: true, - }, - notifications: { - email: true, - push: true, - sms: false, - }, - security: { - mfaRequired: false, - passwordPolicy: { - minLength: 8, - requireNumbers: true, - requireSpecialChars: true, - requireUppercase: true, - }, - sessionTimeout: 3600, - }, - }); - - return await this.configRepository.save(config); - } - - /** - * Increment user count - */ - async incrementUserCount(tenantId: string): Promise { - await this.tenantRepository.increment({ id: tenantId }, 'currentUserCount', 1); - } - - /** - * Decrement user count - */ - async decrementUserCount(tenantId: string): Promise { - await this.tenantRepository.decrement({ id: tenantId }, 'currentUserCount', 1); - } - - /** - * Update storage usage - */ - async updateStorageUsage(tenantId: string, sizeInMB: number): Promise { - const tenant = await this.findOne(tenantId); - tenant.currentStorageUsage += sizeInMB; - await this.tenantRepository.save(tenant); - } - - /** - * Get tenant with all related data - */ - async getTenantWithRelations(tenantId: string): Promise<{ - tenant: Tenant; - config: TenantConfig; - billing: TenantBilling; - customization: TenantCustomization; - }> { - const [tenant, config, billing, customization] = await Promise.all([ - this.findOne(tenantId), - this.getConfig(tenantId), - this.billingService.getBillingInfo(tenantId), - this.customizationService.getCustomization(tenantId), - ]); - - return { - tenant, - config, - billing, - customization, - }; - } } diff --git a/src/users/dto/create-user.dto.ts b/src/users/dto/create-user.dto.ts index c6fa115d..eeab41cd 100644 --- a/src/users/dto/create-user.dto.ts +++ b/src/users/dto/create-user.dto.ts @@ -2,35 +2,29 @@ import { IsEmail, IsString, IsOptional, IsEnum, IsNotEmpty } from 'class-validat import { ApiProperty } from '@nestjs/swagger'; import { UserRole } from '../entities/user.entity'; import { IsStrongPassword } from '../../common/validators/password.validator'; - export class CreateUserDto { - @ApiProperty({ example: 'john.doe@example.com' }) - @IsEmail({}, { message: 'Please provide a valid email format' }) - @IsNotEmpty({ message: 'Email cannot be blank' }) - email: string; - - @ApiProperty({ example: 'StrongPass123!' }) - @IsString() - @IsStrongPassword() - password: string; - - @ApiProperty({ example: 'John' }) - @IsString({ message: 'First name must be a string' }) - @IsNotEmpty({ message: 'First name is required' }) - firstName: string; - - @ApiProperty({ example: 'Doe' }) - @IsString({ message: 'Last name must be a string' }) - @IsNotEmpty({ message: 'Last name is required' }) - lastName: string; - - @ApiProperty({ enum: UserRole, required: false, default: UserRole.STUDENT }) - @IsOptional() - @IsEnum(UserRole, { message: 'Invalid user role' }) - role?: UserRole; - - @ApiProperty({ required: false }) - @IsOptional() - @IsString({ message: 'Profile picture must be a valid URL/string' }) - profilePicture?: string; + @ApiProperty({ example: 'john.doe@example.com' }) + @IsEmail({}, { message: 'Please provide a valid email format' }) + @IsNotEmpty({ message: 'Email cannot be blank' }) + email: string; + @ApiProperty({ example: 'StrongPass123!' }) + @IsString() + @IsStrongPassword() + password: string; + @ApiProperty({ example: 'John' }) + @IsString({ message: 'First name must be a string' }) + @IsNotEmpty({ message: 'First name is required' }) + firstName: string; + @ApiProperty({ example: 'Doe' }) + @IsString({ message: 'Last name must be a string' }) + @IsNotEmpty({ message: 'Last name is required' }) + lastName: string; + @ApiProperty({ enum: UserRole, required: false, default: UserRole.STUDENT }) + @IsOptional() + @IsEnum(UserRole, { message: 'Invalid user role' }) + role?: UserRole; + @ApiProperty({ required: false }) + @IsOptional() + @IsString({ message: 'Profile picture must be a valid URL/string' }) + profilePicture?: string; } diff --git a/src/users/dto/get-users.dto.ts b/src/users/dto/get-users.dto.ts index 559d1e5e..f861aad0 100644 --- a/src/users/dto/get-users.dto.ts +++ b/src/users/dto/get-users.dto.ts @@ -1,12 +1,10 @@ import { IsOptional, IsString } from 'class-validator'; import { PaginationQueryDto } from '../../common/dto/pagination.dto'; - export class GetUsersDto extends PaginationQueryDto { - @IsOptional() - @IsString() - status?: string; - - @IsOptional() - @IsString() - role?: string; + @IsOptional() + @IsString() + status?: string; + @IsOptional() + @IsString() + role?: string; } diff --git a/src/users/dto/update-user.dto.ts b/src/users/dto/update-user.dto.ts index a03365ad..81e538d0 100644 --- a/src/users/dto/update-user.dto.ts +++ b/src/users/dto/update-user.dto.ts @@ -1,15 +1,13 @@ import { PartialType } from '@nestjs/swagger'; import { IsBoolean, IsOptional, IsString, MinLength, IsEmail } from 'class-validator'; import { CreateUserDto } from './create-user.dto'; - export class UpdateUserDto extends PartialType(CreateUserDto) { - @IsOptional() - @IsString() - @MinLength(6) - password?: string; - - @IsOptional() - @IsBoolean() - @IsEmail() - isEmailVerified?: boolean; + @IsOptional() + @IsString() + @MinLength(6) + password?: string; + @IsOptional() + @IsBoolean() + @IsEmail() + isEmailVerified?: boolean; } diff --git a/src/users/entities/user.entity.ts b/src/users/entities/user.entity.ts index a419d4c2..531123a6 100644 --- a/src/users/entities/user.entity.ts +++ b/src/users/entities/user.entity.ts @@ -1,107 +1,73 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - CreateDateColumn, - UpdateDateColumn, - DeleteDateColumn, - Index, - OneToMany, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, CreateDateColumn, UpdateDateColumn, DeleteDateColumn, Index, OneToMany, } from 'typeorm'; import { Course } from '../../courses/entities/course.entity'; import { Enrollment } from '../../courses/entities/enrollment.entity'; - export enum UserRole { - STUDENT = 'student', - TEACHER = 'teacher', - ADMIN = 'admin', + STUDENT = 'student', + TEACHER = 'teacher', + ADMIN = 'admin' } - export enum UserStatus { - ACTIVE = 'active', - INACTIVE = 'inactive', - SUSPENDED = 'suspended', + ACTIVE = 'active', + INACTIVE = 'inactive', + SUSPENDED = 'suspended' } - @Entity('users') export class User { - @PrimaryGeneratedColumn('uuid') - id: string; - - @Column({ unique: true }) - @Index() - email: string; - - @Column({ nullable: true }) - @Index() - username?: string; - - @Column() - password: string; - - @Column() - firstName: string; - - @Column() - lastName: string; - - @Column({ - type: 'enum', - enum: UserRole, - default: UserRole.STUDENT, - }) - role: UserRole; - - @Column({ - type: 'enum', - enum: UserStatus, - default: UserStatus.ACTIVE, - }) - status: UserStatus; - - @Column({ nullable: true }) - @Index() - tenantId?: string; - - @Column({ nullable: true }) - profilePicture?: string; - - @Column({ default: false }) - isEmailVerified: boolean; - - @Column({ nullable: true }) - emailVerificationToken?: string; - - @Column({ type: 'timestamp', nullable: true }) - emailVerificationExpires?: Date; - - @Column({ nullable: true }) - passwordResetToken?: string; - - @Column({ type: 'timestamp', nullable: true }) - passwordResetExpires?: Date; - - @Column({ nullable: true }) - refreshToken?: string; - - @Column('text', { array: true, default: [] }) - passwordHistory: string[]; - - @Column({ type: 'timestamp', nullable: true }) - lastLoginAt?: Date; - - @OneToMany(() => Course, (course) => course.instructor) - courses: Course[]; - - @OneToMany(() => Enrollment, (enrollment) => enrollment.user) - enrollments: Enrollment[]; - - @CreateDateColumn() - createdAt: Date; - - @UpdateDateColumn() - updatedAt: Date; - - @DeleteDateColumn() - deletedAt?: Date; + @PrimaryGeneratedColumn('uuid') + id: string; + @Column({ unique: true }) + @Index() + email: string; + @Column({ nullable: true }) + @Index() + username?: string; + @Column() + password: string; + @Column() + firstName: string; + @Column() + lastName: string; + @Column({ + type: 'enum', + enum: UserRole, + default: UserRole.STUDENT, + }) + role: UserRole; + @Column({ + type: 'enum', + enum: UserStatus, + default: UserStatus.ACTIVE, + }) + status: UserStatus; + @Column({ nullable: true }) + @Index() + tenantId?: string; + @Column({ nullable: true }) + profilePicture?: string; + @Column({ default: false }) + isEmailVerified: boolean; + @Column({ nullable: true }) + emailVerificationToken?: string; + @Column({ type: 'timestamp', nullable: true }) + emailVerificationExpires?: Date; + @Column({ nullable: true }) + passwordResetToken?: string; + @Column({ type: 'timestamp', nullable: true }) + passwordResetExpires?: Date; + @Column({ nullable: true }) + refreshToken?: string; + @Column('text', { array: true, default: [] }) + passwordHistory: string[]; + @Column({ type: 'timestamp', nullable: true }) + lastLoginAt?: Date; + @OneToMany(() => Course, (course) => course.instructor) + courses: Course[]; + @OneToMany(() => Enrollment, (enrollment) => enrollment.user) + enrollments: Enrollment[]; + @CreateDateColumn() + createdAt: Date; + @UpdateDateColumn() + updatedAt: Date; + @DeleteDateColumn() + deletedAt?: Date; } diff --git a/src/users/users.controller.ts b/src/users/users.controller.ts index d64da12d..802871f7 100644 --- a/src/users/users.controller.ts +++ b/src/users/users.controller.ts @@ -11,77 +11,87 @@ import { Roles } from '../auth/decorators/roles.decorator'; import { CurrentUser } from '../auth/decorators/current-user.decorator'; import { UserRole } from './entities/user.entity'; import { ExportFormat, ExportService } from '../common/export/export.service'; - @ApiTags('users') @Controller('users') @UseGuards(JwtAuthGuard, RolesGuard) @ApiBearerAuth() export class UsersController { - constructor( - private readonly usersService: UsersService, - private readonly exportService: ExportService, - ) {} - - @Post() - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Create a new user (Admin only)' }) - create(@Body() createUserDto: CreateUserDto) { - return this.usersService.create(createUserDto); - } - - @Get() - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Get all users (Admin only)' }) - findAll() { - return this.usersService.findAll(); - } - - @Get(':id') - @ApiOperation({ summary: 'Get user by ID' }) - findOne(@Param('id') id: string) { - return this.usersService.findOne(id); - } - - @Patch(':id') - @ApiOperation({ summary: 'Update user' }) - update(@Param('id') id: string, @Body() updateUserDto: UpdateUserDto) { - return this.usersService.update(id, updateUserDto); - } - - @Delete(':id') - @Roles(UserRole.ADMIN) - @ApiOperation({ summary: 'Delete user (Admin only)' }) - remove(@Param('id') id: string) { - return this.usersService.remove(id); - } - - @Post('me/export') - @Throttle({ default: { limit: 3, ttl: 3600000 } }) - @ApiOperation({ summary: 'Request user data export (JSON or PDF)' }) - requestDataExport( - @CurrentUser() user: { userId: string }, - @Body() body?: { format?: ExportFormat }, - ) { - return this.exportService.requestUserDataExport(user.userId, body?.format ?? 'json'); - } - - @Get('me/export/history') - @ApiOperation({ summary: 'Get export request history for the current user' }) - getExportHistory(@CurrentUser() user: { userId: string }) { - return this.exportService.getUserExportHistory(user.userId); - } - - @Get('me/export/:exportId') - @ApiOperation({ summary: 'Download a completed user export file' }) - async downloadExport( - @CurrentUser() user: { userId: string }, - @Param('exportId') exportId: string, - @Res() res: Response, - ) { - const file = await this.exportService.getCompletedExportFile(user.userId, exportId); - - res.setHeader('Content-Type', file.mimeType); - res.setHeader('Content-Disposition', `attachment; filename="${file.fileName}"`); - res.send(file.content); - } + constructor(private readonly usersService: UsersService, private readonly exportService: ExportService) { } + @Post() + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Create a new user (Admin only)' }) + create( + @Body() + createUserDto: CreateUserDto) { + return this.usersService.create(createUserDto); + } + @Get() + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Get all users (Admin only)' }) + findAll() { + return this.usersService.findAll(); + } + @Get(':id') + @ApiOperation({ summary: 'Get user by ID' }) + findOne( + @Param('id') + id: string) { + return this.usersService.findOne(id); + } + @Patch(':id') + @ApiOperation({ summary: 'Update user' }) + update( + @Param('id') + id: string, + @Body() + updateUserDto: UpdateUserDto) { + return this.usersService.update(id, updateUserDto); + } + @Delete(':id') + @Roles(UserRole.ADMIN) + @ApiOperation({ summary: 'Delete user (Admin only)' }) + remove( + @Param('id') + id: string) { + return this.usersService.remove(id); + } + @Post('me/export') + @Throttle({ default: { limit: 3, ttl: 3600000 } }) + @ApiOperation({ summary: 'Request user data export (JSON or PDF)' }) + requestDataExport( + @CurrentUser() + user: { + userId: string; + }, + @Body() + body?: { + format?: ExportFormat; + }) { + return this.exportService.requestUserDataExport(user.userId, body?.format ?? 'json'); + } + @Get('me/export/history') + @ApiOperation({ summary: 'Get export request history for the current user' }) + getExportHistory( + @CurrentUser() + user: { + userId: string; + }) { + return this.exportService.getUserExportHistory(user.userId); + } + @Get('me/export/:exportId') + @ApiOperation({ summary: 'Download a completed user export file' }) + async downloadExport( + @CurrentUser() + user: { + userId: string; + }, + @Param('exportId') + exportId: string, + @Res() + res: Response) { + const file = await this.exportService.getCompletedExportFile(user.userId, exportId); + res.setHeader('Content-Type', file.mimeType); + res.setHeader('Content-Disposition', `attachment; filename="${file.fileName}"`); + res.send(file.content); + } } diff --git a/src/users/users.module.ts b/src/users/users.module.ts index cc2bf8f4..60c60140 100644 --- a/src/users/users.module.ts +++ b/src/users/users.module.ts @@ -8,19 +8,15 @@ import { User } from './entities/user.entity'; import { Enrollment } from '../courses/entities/enrollment.entity'; import { RolesGuard } from '../auth/guards/roles.guard'; import { JwtAuthGuard } from '../auth/guards/jwt-auth.guard'; -import { - ExportService, - UserDataExportProcessor, - UserExportHistory, -} from '../common/export/export.service'; - +import { ExportService, UserDataExportProcessor, UserExportHistory, } from '../common/export/export.service'; @Module({ - imports: [ - TypeOrmModule.forFeature([User, Enrollment, UserExportHistory]), - BullModule.registerQueue({ name: QUEUE_NAMES.USER_DATA_EXPORT }), - ], - controllers: [UsersController], - providers: [UsersService, ExportService, UserDataExportProcessor, RolesGuard, JwtAuthGuard], - exports: [UsersService], + imports: [ + TypeOrmModule.forFeature([User, Enrollment, UserExportHistory]), + BullModule.registerQueue({ name: QUEUE_NAMES.USER_DATA_EXPORT }), + ], + controllers: [UsersController], + providers: [UsersService, ExportService, UserDataExportProcessor, RolesGuard, JwtAuthGuard], + exports: [UsersService], }) -export class UsersModule {} +export class UsersModule { +} diff --git a/src/users/users.service.spec.ts b/src/users/users.service.spec.ts index 98297d06..e44f0300 100644 --- a/src/users/users.service.spec.ts +++ b/src/users/users.service.spec.ts @@ -1,130 +1,86 @@ import { UsersService } from './users.service'; import { UserRole, UserStatus } from './entities/user.entity'; import * as bcrypt from 'bcryptjs'; -import { - createMockRepository, - createMockCachingService, - createMockQueryBuilder, - createMockEventEmitter, -} from 'test/utils/mock-factories'; +import { createMockRepository, createMockCachingService, createMockQueryBuilder, createMockEventEmitter, } from 'test/utils/mock-factories'; import { Repository } from 'typeorm'; - describe('UsersService', () => { - let service: UsersService; - let mockUserRepository: jest.Mocked>; - let mockCachingService: jest.Mocked; - let mockEventEmitter: jest.Mocked; - - beforeEach(() => { - // ─── Initialize Mocks ────────────────────────────────────────────────── - mockUserRepository = createMockRepository(); - mockCachingService = createMockCachingService(); - mockEventEmitter = createMockEventEmitter(); - - // ─── Configure Default QueryBuilder ──────────────────────────────────── - const mockQueryBuilder = createMockQueryBuilder(); - mockQueryBuilder.getMany.mockResolvedValue([ - { id: 'user-1', email: 'test@example.com' }, - ]); - mockQueryBuilder.getCount.mockResolvedValue(1); - mockUserRepository.createQueryBuilder.mockReturnValue(mockQueryBuilder as any); - - // ─── Service Instantiation ───────────────────────────────────────────── - service = new UsersService( - mockUserRepository, - mockCachingService, - mockEventEmitter, - ); - }); - - afterEach(() => { - jest.clearAllMocks(); - }); - - it('sanitizes search input and uses parameterized ILIKE', async () => { - const mockQueryBuilder = createMockQueryBuilder(); - mockUserRepository.createQueryBuilder.mockReturnValue(mockQueryBuilder as any); - - const maliciousSearch = "a%_b\\ test' OR 1=1 --"; - - await expect( - service.findAll({ - search: maliciousSearch, - role: UserRole.STUDENT, - status: UserStatus.ACTIVE, - }), - ).resolves.toBeDefined(); - - expect(mockUserRepository.createQueryBuilder).toHaveBeenCalledWith('user'); - - expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith( - "(user.email ILIKE :search ESCAPE '\\' OR user.firstName ILIKE :search ESCAPE '\\' OR user.lastName ILIKE :search ESCAPE '\\')", - { - search: "%a\\%\\_b\\\\ test' OR 1=1 --%", - }, - ); - }); - - it('blocks non-whitelisted role values', async () => { - await expect(service.findAll({ role: 'hacker' as any })).rejects.toThrow( - /Invalid value for role/, - ); - }); - - it('blocks non-whitelisted status values', async () => { - await expect(service.findAll({ status: 'hacked' as any })).rejects.toThrow( - /Invalid value for status/, - ); - }); - - it('rejects password reuse via current password', async () => { - const currentHash = await bcrypt.hash('CurrentPass1!', 10); - - mockUserRepository.findOne.mockResolvedValue({ - id: 'user-1', - email: 'test@example.com', - password: currentHash, - passwordHistory: [], + let service: UsersService; + let mockUserRepository: jest.Mocked>; + let mockCachingService: jest.Mocked; + let mockEventEmitter: jest.Mocked; + beforeEach(() => { + // ─── Initialize Mocks ────────────────────────────────────────────────── + mockUserRepository = createMockRepository(); + mockCachingService = createMockCachingService(); + mockEventEmitter = createMockEventEmitter(); + // ─── Configure Default QueryBuilder ──────────────────────────────────── + const mockQueryBuilder = createMockQueryBuilder(); + mockQueryBuilder.getMany.mockResolvedValue([ + { id: 'user-1', email: 'test@example.com' }, + ]); + mockQueryBuilder.getCount.mockResolvedValue(1); + mockUserRepository.createQueryBuilder.mockReturnValue(mockQueryBuilder as unknown); + // ─── Service Instantiation ───────────────────────────────────────────── + service = new UsersService(mockUserRepository, mockCachingService, mockEventEmitter); }); - mockUserRepository.save.mockResolvedValue(true as any); - - await expect(service.update('user-1', { password: 'CurrentPass1!' })).rejects.toThrow( - /New password must be different from the current password/, - ); - }); - - it('rejects password reuse via history', async () => { - const currentHash = await bcrypt.hash('CurrentPass1!', 10); - const oldHash = await bcrypt.hash('OldPass1!', 10); - - mockUserRepository.findOne.mockResolvedValue({ - id: 'user-1', - email: 'test@example.com', - password: currentHash, - passwordHistory: [oldHash], + afterEach(() => { + jest.clearAllMocks(); }); - mockUserRepository.save.mockResolvedValue(true as any); - - await expect(service.update('user-1', { password: 'OldPass1!' })).rejects.toThrow( - /New password must not match your last 5 passwords/, - ); - }); - - it('updates password and appends current password to history', async () => { - const currentHash = await bcrypt.hash('CurrentPass1!', 10); - - mockUserRepository.findOne.mockResolvedValue({ - id: 'user-1', - email: 'test@example.com', - password: currentHash, - passwordHistory: [], + it('sanitizes search input and uses parameterized ILIKE', async () => { + const mockQueryBuilder = createMockQueryBuilder(); + mockUserRepository.createQueryBuilder.mockReturnValue(mockQueryBuilder as unknown); + const maliciousSearch = "a%_b\\ test' OR 1=1 --"; + await expect(service.findAll({ + search: maliciousSearch, + role: UserRole.STUDENT, + status: UserStatus.ACTIVE, + })).resolves.toBeDefined(); + expect(mockUserRepository.createQueryBuilder).toHaveBeenCalledWith('user'); + expect(mockQueryBuilder.andWhere).toHaveBeenCalledWith("(user.email ILIKE :search ESCAPE '\\' OR user.firstName ILIKE :search ESCAPE '\\' OR user.lastName ILIKE :search ESCAPE '\\')", { + search: "%a\\%\\_b\\\\ test' OR 1=1 --%", + }); + }); + it('blocks non-whitelisted role values', async () => { + await expect(service.findAll({ role: 'hacker' as unknown })).rejects.toThrow(/Invalid value for role/); + }); + it('blocks non-whitelisted status values', async () => { + await expect(service.findAll({ status: 'hacked' as unknown })).rejects.toThrow(/Invalid value for status/); + }); + it('rejects password reuse via current password', async () => { + const currentHash = await bcrypt.hash('CurrentPass1!', 10); + mockUserRepository.findOne.mockResolvedValue({ + id: 'user-1', + email: 'test@example.com', + password: currentHash, + passwordHistory: [], + }); + mockUserRepository.save.mockResolvedValue(true as unknown); + await expect(service.update('user-1', { password: 'CurrentPass1!' })).rejects.toThrow(/New password must be different from the current password/); + }); + it('rejects password reuse via history', async () => { + const currentHash = await bcrypt.hash('CurrentPass1!', 10); + const oldHash = await bcrypt.hash('OldPass1!', 10); + mockUserRepository.findOne.mockResolvedValue({ + id: 'user-1', + email: 'test@example.com', + password: currentHash, + passwordHistory: [oldHash], + }); + mockUserRepository.save.mockResolvedValue(true as unknown); + await expect(service.update('user-1', { password: 'OldPass1!' })).rejects.toThrow(/New password must not match your last 5 passwords/); + }); + it('updates password and appends current password to history', async () => { + const currentHash = await bcrypt.hash('CurrentPass1!', 10); + mockUserRepository.findOne.mockResolvedValue({ + id: 'user-1', + email: 'test@example.com', + password: currentHash, + passwordHistory: [], + }); + mockUserRepository.save.mockImplementation(async (user: unknown) => user); + const result = await service.update('user-1', { password: 'NewPass1!' }); + expect(result.password).not.toBe('NewPass1!'); + expect(await bcrypt.compare('NewPass1!', result.password)).toBe(true); + expect(result.passwordHistory).toEqual([currentHash]); }); - mockUserRepository.save.mockImplementation(async (user: any) => user); - - const result = await service.update('user-1', { password: 'NewPass1!' }); - - expect(result.password).not.toBe('NewPass1!'); - expect(await bcrypt.compare('NewPass1!', result.password)).toBe(true); - expect(result.passwordHistory).toEqual([currentHash]); - }); }); diff --git a/src/users/users.service.ts b/src/users/users.service.ts index aaaceecc..852f87b5 100644 --- a/src/users/users.service.ts +++ b/src/users/users.service.ts @@ -14,187 +14,129 @@ import { CachingService } from '../caching/caching.service'; import { CACHE_TTL, CACHE_PREFIXES, CACHE_EVENTS } from '../caching/caching.constants'; import { EventEmitter2 } from '@nestjs/event-emitter'; import { ConfigService } from '@nestjs/config'; - @Injectable() export class UsersService { - constructor( + constructor( @InjectRepository(User) - private readonly userRepository: Repository, - private readonly cachingService: CachingService, - private readonly eventEmitter: EventEmitter2, - private readonly configService: ConfigService, - ) {} - - async create(createUserDto: CreateUserDto): Promise { - // Check if user already exists - const existingUser = await this.userRepository.findOne({ - where: { email: createUserDto.email }, - withDeleted: true, - }); - ensureUserDoesNotExist(existingUser, 'User with this email already exists'); - - // Hash password - const bcryptRounds = this.configService.get('BCRYPT_ROUNDS') || 10; - const hashedPassword = await bcrypt.hash(createUserDto.password, bcryptRounds); - - // Create user - const user = this.userRepository.create({ - ...createUserDto, - password: hashedPassword, - }); - - return await this.userRepository.save(user); - } - - async findAll(filter?: GetUsersDto): Promise> { - const cacheKey = `${CACHE_PREFIXES.USERS_LIST}:${JSON.stringify(filter || {})}`; - - return this.cachingService.getOrSet( - cacheKey, - async () => { - const query = this.userRepository.createQueryBuilder('user'); - - if (filter?.role) { - const role = enforceWhitelistedValue(filter.role, Object.values(UserRole), 'role'); - query.andWhere('user.role = :role', { role }); - } - - if (filter?.status) { - const status = enforceWhitelistedValue( - filter.status, - Object.values(UserStatus), - 'status', - ); - query.andWhere('user.status = :status', { status }); - } - - if (filter?.search) { - const safeSearch = sanitizeSqlLike(filter.search); - // eslint-disable-next-line quotes - query.andWhere( - "(user.email ILIKE :search ESCAPE '\\' OR user.firstName ILIKE :search ESCAPE '\\' OR user.lastName ILIKE :search ESCAPE '\\')", // eslint-disable-line quotes - { search: `%${safeSearch}%` }, - ); - } - - return await paginate(query, filter || new PaginationQueryDto()); - }, - CACHE_TTL.USER_PROFILE, - ); - } - - async findByIds(ids: string[]): Promise { - if (ids.length === 0) return []; - return await this.userRepository.findByIds(ids); - } - - /** - * Helper method to find a user by ID or throw NotFoundException. - * Can be used internally to eliminate duplication. - */ - async findUserOrThrow(id: string): Promise { - const user = await this.userRepository.findOne({ where: { id } }); - return ensureUserExists(user, 'User not found'); - } - - async findOne(id: string): Promise { - const cacheKey = `${CACHE_PREFIXES.USER_PROFILE}:${id}`; - - return this.cachingService.getOrSet( - cacheKey, - async () => { - return await this.findUserOrThrow(id); - }, - CACHE_TTL.USER_PROFILE, - ); - } - - async findByEmail(email: string): Promise { - return await this.userRepository.findOne({ where: { email } }); - } - - async findByPasswordResetToken(token: string): Promise { - return await this.userRepository.findOne({ - where: { passwordResetToken: token }, - }); - } - - async findByEmailVerificationToken(token: string): Promise { - return await this.userRepository.findOne({ - where: { emailVerificationToken: token }, - }); - } - - async update(id: string, updateUserDto: UpdateUserDto): Promise { - const user = await this.findUserOrThrow(id); - - if (updateUserDto.password) { - const plainPassword = updateUserDto.password; - - if (await bcrypt.compare(plainPassword, user.password)) { - throw new BadRequestException('New password must be different from the current password'); - } - - const recentPasswords = user.passwordHistory ?? []; - for (const oldHash of recentPasswords.slice(-5)) { - if (await bcrypt.compare(plainPassword, oldHash)) { - throw new BadRequestException('New password must not match your last 5 passwords'); + private readonly userRepository: Repository, private readonly cachingService: CachingService, private readonly eventEmitter: EventEmitter2, private readonly configService: ConfigService) { } + async create(createUserDto: CreateUserDto): Promise { + // Check if user already exists + const existingUser = await this.userRepository.findOne({ + where: { email: createUserDto.email }, + withDeleted: true, + }); + ensureUserDoesNotExist(existingUser, 'User with this email already exists'); + // Hash password + const bcryptRounds = this.configService.get('BCRYPT_ROUNDS') || 10; + const hashedPassword = await bcrypt.hash(createUserDto.password, bcryptRounds); + // Create user + const user = this.userRepository.create({ + ...createUserDto, + password: hashedPassword, + }); + return await this.userRepository.save(user); + } + async findAll(filter?: GetUsersDto): Promise> { + const cacheKey = `${CACHE_PREFIXES.USERS_LIST}:${JSON.stringify(filter || {})}`; + return this.cachingService.getOrSet(cacheKey, async () => { + const query = this.userRepository.createQueryBuilder('user'); + if (filter?.role) { + const role = enforceWhitelistedValue(filter.role, Object.values(UserRole), 'role'); + query.andWhere('user.role = :role', { role }); + } + if (filter?.status) { + const status = enforceWhitelistedValue(filter.status, Object.values(UserStatus), 'status'); + query.andWhere('user.status = :status', { status }); + } + if (filter?.search) { + const safeSearch = sanitizeSqlLike(filter.search); + // eslint-disable-next-line quotes + query.andWhere("(user.email ILIKE :search ESCAPE '\\' OR user.firstName ILIKE :search ESCAPE '\\' OR user.lastName ILIKE :search ESCAPE '\\')", // eslint-disable-line quotes + { search: `%${safeSearch}%` }); + } + return await paginate(query, filter || new PaginationQueryDto()); + }, CACHE_TTL.USER_PROFILE); + } + async findByIds(ids: string[]): Promise { + if (ids.length === 0) + return []; + return await this.userRepository.findByIds(ids); + } + /** + * Helper method to find a user by ID or throw NotFoundException. + * Can be used internally to eliminate duplication. + */ + async findUserOrThrow(id: string): Promise { + const user = await this.userRepository.findOne({ where: { id } }); + return ensureUserExists(user, 'User not found'); + } + async findOne(id: string): Promise { + const cacheKey = `${CACHE_PREFIXES.USER_PROFILE}:${id}`; + return this.cachingService.getOrSet(cacheKey, async () => { + return await this.findUserOrThrow(id); + }, CACHE_TTL.USER_PROFILE); + } + async findByEmail(email: string): Promise { + return await this.userRepository.findOne({ where: { email } }); + } + async findByPasswordResetToken(token: string): Promise { + return await this.userRepository.findOne({ + where: { passwordResetToken: token }, + }); + } + async findByEmailVerificationToken(token: string): Promise { + return await this.userRepository.findOne({ + where: { emailVerificationToken: token }, + }); + } + async update(id: string, updateUserDto: UpdateUserDto): Promise { + const user = await this.findUserOrThrow(id); + if (updateUserDto.password) { + const plainPassword = updateUserDto.password; + if (await bcrypt.compare(plainPassword, user.password)) { + throw new BadRequestException('New password must be different from the current password'); + } + const recentPasswords = user.passwordHistory ?? []; + for (const oldHash of recentPasswords.slice(-5)) { + if (await bcrypt.compare(plainPassword, oldHash)) { + throw new BadRequestException('New password must not match your last 5 passwords'); + } + } + // Append current, maintain last 5 entries + user.passwordHistory = [...recentPasswords, user.password].slice(-5); + const bcryptRounds = this.configService.get('BCRYPT_ROUNDS') || 10; + updateUserDto.password = await bcrypt.hash(plainPassword, bcryptRounds); } - } - - // Append current, maintain last 5 entries - user.passwordHistory = [...recentPasswords, user.password].slice(-5); - - const bcryptRounds = this.configService.get('BCRYPT_ROUNDS') || 10; - updateUserDto.password = await bcrypt.hash(plainPassword, bcryptRounds); + Object.assign(user, updateUserDto); + const saved = await this.userRepository.save(user); + // Invalidate cache after update + this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId: id }); + return saved; + } + async updateRefreshToken(userId: string, refreshToken: string | null): Promise { + await this.userRepository.update(userId, { refreshToken: refreshToken as unknown as string }); + // Invalidate user cache + this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId }); + } + async updatePasswordResetToken(userId: string, token: string | null, expires: Date | null): Promise { + await this.userRepository.update(userId, { + passwordResetToken: token as unknown as string, + passwordResetExpires: expires as unknown as Date, + }); + } + async updateEmailVerificationToken(userId: string, token: string | null, expires: Date | null): Promise { + await this.userRepository.update(userId, { + emailVerificationToken: token as unknown as string, + emailVerificationExpires: expires as unknown as Date, + }); + } + async updateLastLogin(userId: string): Promise { + await this.userRepository.update(userId, { lastLoginAt: new Date() }); + } + async remove(id: string): Promise { + await this.findUserOrThrow(id); + await this.userRepository.softDelete(id); + // Invalidate cache after delete + this.eventEmitter.emit(CACHE_EVENTS.USER_DELETED, { userId: id }); } - - Object.assign(user, updateUserDto); - const saved = await this.userRepository.save(user); - - // Invalidate cache after update - this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId: id }); - - return saved; - } - - async updateRefreshToken(userId: string, refreshToken: string | null): Promise { - await this.userRepository.update(userId, { refreshToken: refreshToken as unknown as string }); - // Invalidate user cache - this.eventEmitter.emit(CACHE_EVENTS.USER_UPDATED, { userId }); - } - - async updatePasswordResetToken( - userId: string, - token: string | null, - expires: Date | null, - ): Promise { - await this.userRepository.update(userId, { - passwordResetToken: token as unknown as string, - passwordResetExpires: expires as unknown as Date, - }); - } - - async updateEmailVerificationToken( - userId: string, - token: string | null, - expires: Date | null, - ): Promise { - await this.userRepository.update(userId, { - emailVerificationToken: token as unknown as string, - emailVerificationExpires: expires as unknown as Date, - }); - } - - async updateLastLogin(userId: string): Promise { - await this.userRepository.update(userId, { lastLoginAt: new Date() }); - } - - async remove(id: string): Promise { - await this.findUserOrThrow(id); - await this.userRepository.softDelete(id); - - // Invalidate cache after delete - this.eventEmitter.emit(CACHE_EVENTS.USER_DELETED, { userId: id }); - } } diff --git a/tsconfig.build.tsbuildinfo b/tsconfig.build.tsbuildinfo index 8c70bb93..b83c780f 100644 --- a/tsconfig.build.tsbuildinfo +++ b/tsconfig.build.tsbuildinfo @@ -1 +1 @@ -{"fileNames":["./node_modules/typescript/lib/lib.es5.d.ts","./node_modules/typescript/lib/lib.es2015.d.ts","./node_modules/typescript/lib/lib.es2016.d.ts","./node_modules/typescript/lib/lib.es2017.d.ts","./node_modules/typescript/lib/lib.es2018.d.ts","./node_modules/typescript/lib/lib.es2019.d.ts","./node_modules/typescript/lib/lib.es2020.d.ts","./node_modules/typescript/lib/lib.es2021.d.ts","./node_modules/typescript/lib/lib.dom.d.ts","./node_modules/typescript/lib/lib.es2015.core.d.ts","./node_modules/typescript/lib/lib.es2015.collection.d.ts","./node_modules/typescript/lib/lib.es2015.generator.d.ts","./node_modules/typescript/lib/lib.es2015.iterable.d.ts","./node_modules/typescript/lib/lib.es2015.promise.d.ts","./node_modules/typescript/lib/lib.es2015.proxy.d.ts","./node_modules/typescript/lib/lib.es2015.reflect.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2016.array.include.d.ts","./node_modules/typescript/lib/lib.es2016.intl.d.ts","./node_modules/typescript/lib/lib.es2017.arraybuffer.d.ts","./node_modules/typescript/lib/lib.es2017.date.d.ts","./node_modules/typescript/lib/lib.es2017.object.d.ts","./node_modules/typescript/lib/lib.es2017.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2017.string.d.ts","./node_modules/typescript/lib/lib.es2017.intl.d.ts","./node_modules/typescript/lib/lib.es2017.typedarrays.d.ts","./node_modules/typescript/lib/lib.es2018.asyncgenerator.d.ts","./node_modules/typescript/lib/lib.es2018.asynciterable.d.ts","./node_modules/typescript/lib/lib.es2018.intl.d.ts","./node_modules/typescript/lib/lib.es2018.promise.d.ts","./node_modules/typescript/lib/lib.es2018.regexp.d.ts","./node_modules/typescript/lib/lib.es2019.array.d.ts","./node_modules/typescript/lib/lib.es2019.object.d.ts","./node_modules/typescript/lib/lib.es2019.string.d.ts","./node_modules/typescript/lib/lib.es2019.symbol.d.ts","./node_modules/typescript/lib/lib.es2019.intl.d.ts","./node_modules/typescript/lib/lib.es2020.bigint.d.ts","./node_modules/typescript/lib/lib.es2020.date.d.ts","./node_modules/typescript/lib/lib.es2020.promise.d.ts","./node_modules/typescript/lib/lib.es2020.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2020.string.d.ts","./node_modules/typescript/lib/lib.es2020.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2020.intl.d.ts","./node_modules/typescript/lib/lib.es2020.number.d.ts","./node_modules/typescript/lib/lib.es2021.promise.d.ts","./node_modules/typescript/lib/lib.es2021.string.d.ts","./node_modules/typescript/lib/lib.es2021.weakref.d.ts","./node_modules/typescript/lib/lib.es2021.intl.d.ts","./node_modules/typescript/lib/lib.decorators.d.ts","./node_modules/typescript/lib/lib.decorators.legacy.d.ts","./node_modules/reflect-metadata/index.d.ts","./node_modules/@nestjs/common/decorators/core/bind.decorator.d.ts","./node_modules/@nestjs/common/interfaces/abstract.interface.d.ts","./node_modules/@nestjs/common/interfaces/controllers/controller-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/controllers/controller.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/arguments-host.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/exception-filter.interface.d.ts","./node_modules/rxjs/dist/types/internal/subscription.d.ts","./node_modules/rxjs/dist/types/internal/subscriber.d.ts","./node_modules/rxjs/dist/types/internal/operator.d.ts","./node_modules/rxjs/dist/types/internal/observable.d.ts","./node_modules/rxjs/dist/types/internal/types.d.ts","./node_modules/rxjs/dist/types/internal/operators/audit.d.ts","./node_modules/rxjs/dist/types/internal/operators/audittime.d.ts","./node_modules/rxjs/dist/types/internal/operators/buffer.d.ts","./node_modules/rxjs/dist/types/internal/operators/buffercount.d.ts","./node_modules/rxjs/dist/types/internal/operators/buffertime.d.ts","./node_modules/rxjs/dist/types/internal/operators/buffertoggle.d.ts","./node_modules/rxjs/dist/types/internal/operators/bufferwhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/catcherror.d.ts","./node_modules/rxjs/dist/types/internal/operators/combinelatestall.d.ts","./node_modules/rxjs/dist/types/internal/operators/combineall.d.ts","./node_modules/rxjs/dist/types/internal/operators/combinelatest.d.ts","./node_modules/rxjs/dist/types/internal/operators/combinelatestwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/concat.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatall.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatmap.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatmapto.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/connect.d.ts","./node_modules/rxjs/dist/types/internal/operators/count.d.ts","./node_modules/rxjs/dist/types/internal/operators/debounce.d.ts","./node_modules/rxjs/dist/types/internal/operators/debouncetime.d.ts","./node_modules/rxjs/dist/types/internal/operators/defaultifempty.d.ts","./node_modules/rxjs/dist/types/internal/operators/delay.d.ts","./node_modules/rxjs/dist/types/internal/operators/delaywhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/dematerialize.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinct.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinctuntilchanged.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinctuntilkeychanged.d.ts","./node_modules/rxjs/dist/types/internal/operators/elementat.d.ts","./node_modules/rxjs/dist/types/internal/operators/endwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/every.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaustall.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaust.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaustmap.d.ts","./node_modules/rxjs/dist/types/internal/operators/expand.d.ts","./node_modules/rxjs/dist/types/internal/operators/filter.d.ts","./node_modules/rxjs/dist/types/internal/operators/finalize.d.ts","./node_modules/rxjs/dist/types/internal/operators/find.d.ts","./node_modules/rxjs/dist/types/internal/operators/findindex.d.ts","./node_modules/rxjs/dist/types/internal/operators/first.d.ts","./node_modules/rxjs/dist/types/internal/subject.d.ts","./node_modules/rxjs/dist/types/internal/operators/groupby.d.ts","./node_modules/rxjs/dist/types/internal/operators/ignoreelements.d.ts","./node_modules/rxjs/dist/types/internal/operators/isempty.d.ts","./node_modules/rxjs/dist/types/internal/operators/last.d.ts","./node_modules/rxjs/dist/types/internal/operators/map.d.ts","./node_modules/rxjs/dist/types/internal/operators/mapto.d.ts","./node_modules/rxjs/dist/types/internal/notification.d.ts","./node_modules/rxjs/dist/types/internal/operators/materialize.d.ts","./node_modules/rxjs/dist/types/internal/operators/max.d.ts","./node_modules/rxjs/dist/types/internal/operators/merge.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeall.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergemap.d.ts","./node_modules/rxjs/dist/types/internal/operators/flatmap.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergemapto.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergescan.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergewith.d.ts","./node_modules/rxjs/dist/types/internal/operators/min.d.ts","./node_modules/rxjs/dist/types/internal/observable/connectableobservable.d.ts","./node_modules/rxjs/dist/types/internal/operators/multicast.d.ts","./node_modules/rxjs/dist/types/internal/operators/observeon.d.ts","./node_modules/rxjs/dist/types/internal/operators/onerrorresumenextwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/pairwise.d.ts","./node_modules/rxjs/dist/types/internal/operators/partition.d.ts","./node_modules/rxjs/dist/types/internal/operators/pluck.d.ts","./node_modules/rxjs/dist/types/internal/operators/publish.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishbehavior.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishlast.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishreplay.d.ts","./node_modules/rxjs/dist/types/internal/operators/race.d.ts","./node_modules/rxjs/dist/types/internal/operators/racewith.d.ts","./node_modules/rxjs/dist/types/internal/operators/reduce.d.ts","./node_modules/rxjs/dist/types/internal/operators/repeat.d.ts","./node_modules/rxjs/dist/types/internal/operators/repeatwhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/retry.d.ts","./node_modules/rxjs/dist/types/internal/operators/retrywhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/refcount.d.ts","./node_modules/rxjs/dist/types/internal/operators/sample.d.ts","./node_modules/rxjs/dist/types/internal/operators/sampletime.d.ts","./node_modules/rxjs/dist/types/internal/operators/scan.d.ts","./node_modules/rxjs/dist/types/internal/operators/sequenceequal.d.ts","./node_modules/rxjs/dist/types/internal/operators/share.d.ts","./node_modules/rxjs/dist/types/internal/operators/sharereplay.d.ts","./node_modules/rxjs/dist/types/internal/operators/single.d.ts","./node_modules/rxjs/dist/types/internal/operators/skip.d.ts","./node_modules/rxjs/dist/types/internal/operators/skiplast.d.ts","./node_modules/rxjs/dist/types/internal/operators/skipuntil.d.ts","./node_modules/rxjs/dist/types/internal/operators/skipwhile.d.ts","./node_modules/rxjs/dist/types/internal/operators/startwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/subscribeon.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchall.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchmap.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchmapto.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchscan.d.ts","./node_modules/rxjs/dist/types/internal/operators/take.d.ts","./node_modules/rxjs/dist/types/internal/operators/takelast.d.ts","./node_modules/rxjs/dist/types/internal/operators/takeuntil.d.ts","./node_modules/rxjs/dist/types/internal/operators/takewhile.d.ts","./node_modules/rxjs/dist/types/internal/operators/tap.d.ts","./node_modules/rxjs/dist/types/internal/operators/throttle.d.ts","./node_modules/rxjs/dist/types/internal/operators/throttletime.d.ts","./node_modules/rxjs/dist/types/internal/operators/throwifempty.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeinterval.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeout.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeoutwith.d.ts","./node_modules/rxjs/dist/types/internal/operators/timestamp.d.ts","./node_modules/rxjs/dist/types/internal/operators/toarray.d.ts","./node_modules/rxjs/dist/types/internal/operators/window.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowcount.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowtime.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowtoggle.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowwhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/withlatestfrom.d.ts","./node_modules/rxjs/dist/types/internal/operators/zip.d.ts","./node_modules/rxjs/dist/types/internal/operators/zipall.d.ts","./node_modules/rxjs/dist/types/internal/operators/zipwith.d.ts","./node_modules/rxjs/dist/types/operators/index.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/action.d.ts","./node_modules/rxjs/dist/types/internal/scheduler.d.ts","./node_modules/rxjs/dist/types/internal/testing/testmessage.d.ts","./node_modules/rxjs/dist/types/internal/testing/subscriptionlog.d.ts","./node_modules/rxjs/dist/types/internal/testing/subscriptionloggable.d.ts","./node_modules/rxjs/dist/types/internal/testing/coldobservable.d.ts","./node_modules/rxjs/dist/types/internal/testing/hotobservable.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/asyncscheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/timerhandle.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/asyncaction.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/virtualtimescheduler.d.ts","./node_modules/rxjs/dist/types/internal/testing/testscheduler.d.ts","./node_modules/rxjs/dist/types/testing/index.d.ts","./node_modules/rxjs/dist/types/internal/symbol/observable.d.ts","./node_modules/rxjs/dist/types/internal/observable/dom/animationframes.d.ts","./node_modules/rxjs/dist/types/internal/behaviorsubject.d.ts","./node_modules/rxjs/dist/types/internal/replaysubject.d.ts","./node_modules/rxjs/dist/types/internal/asyncsubject.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/asapscheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/asap.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/async.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/queuescheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/queue.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/animationframescheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/animationframe.d.ts","./node_modules/rxjs/dist/types/internal/util/identity.d.ts","./node_modules/rxjs/dist/types/internal/util/pipe.d.ts","./node_modules/rxjs/dist/types/internal/util/noop.d.ts","./node_modules/rxjs/dist/types/internal/util/isobservable.d.ts","./node_modules/rxjs/dist/types/internal/lastvaluefrom.d.ts","./node_modules/rxjs/dist/types/internal/firstvaluefrom.d.ts","./node_modules/rxjs/dist/types/internal/util/argumentoutofrangeerror.d.ts","./node_modules/rxjs/dist/types/internal/util/emptyerror.d.ts","./node_modules/rxjs/dist/types/internal/util/notfounderror.d.ts","./node_modules/rxjs/dist/types/internal/util/objectunsubscribederror.d.ts","./node_modules/rxjs/dist/types/internal/util/sequenceerror.d.ts","./node_modules/rxjs/dist/types/internal/util/unsubscriptionerror.d.ts","./node_modules/rxjs/dist/types/internal/observable/bindcallback.d.ts","./node_modules/rxjs/dist/types/internal/observable/bindnodecallback.d.ts","./node_modules/rxjs/dist/types/internal/anycatcher.d.ts","./node_modules/rxjs/dist/types/internal/observable/combinelatest.d.ts","./node_modules/rxjs/dist/types/internal/observable/concat.d.ts","./node_modules/rxjs/dist/types/internal/observable/connectable.d.ts","./node_modules/rxjs/dist/types/internal/observable/defer.d.ts","./node_modules/rxjs/dist/types/internal/observable/empty.d.ts","./node_modules/rxjs/dist/types/internal/observable/forkjoin.d.ts","./node_modules/rxjs/dist/types/internal/observable/from.d.ts","./node_modules/rxjs/dist/types/internal/observable/fromevent.d.ts","./node_modules/rxjs/dist/types/internal/observable/fromeventpattern.d.ts","./node_modules/rxjs/dist/types/internal/observable/generate.d.ts","./node_modules/rxjs/dist/types/internal/observable/iif.d.ts","./node_modules/rxjs/dist/types/internal/observable/interval.d.ts","./node_modules/rxjs/dist/types/internal/observable/merge.d.ts","./node_modules/rxjs/dist/types/internal/observable/never.d.ts","./node_modules/rxjs/dist/types/internal/observable/of.d.ts","./node_modules/rxjs/dist/types/internal/observable/onerrorresumenext.d.ts","./node_modules/rxjs/dist/types/internal/observable/pairs.d.ts","./node_modules/rxjs/dist/types/internal/observable/partition.d.ts","./node_modules/rxjs/dist/types/internal/observable/race.d.ts","./node_modules/rxjs/dist/types/internal/observable/range.d.ts","./node_modules/rxjs/dist/types/internal/observable/throwerror.d.ts","./node_modules/rxjs/dist/types/internal/observable/timer.d.ts","./node_modules/rxjs/dist/types/internal/observable/using.d.ts","./node_modules/rxjs/dist/types/internal/observable/zip.d.ts","./node_modules/rxjs/dist/types/internal/scheduled/scheduled.d.ts","./node_modules/rxjs/dist/types/internal/config.d.ts","./node_modules/rxjs/dist/types/index.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/rpc-exception-filter.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/ws-exception-filter.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validation-error.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/execution-context.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/can-activate.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/custom-route-param-factory.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/nest-interceptor.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/paramtype.interface.d.ts","./node_modules/@nestjs/common/interfaces/type.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/pipe-transform.interface.d.ts","./node_modules/@nestjs/common/enums/request-method.enum.d.ts","./node_modules/@nestjs/common/enums/http-status.enum.d.ts","./node_modules/@nestjs/common/enums/shutdown-signal.enum.d.ts","./node_modules/@nestjs/common/enums/version-type.enum.d.ts","./node_modules/@nestjs/common/enums/index.d.ts","./node_modules/@nestjs/common/interfaces/version-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-configuration.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-consumer.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-config-proxy.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/nest-middleware.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/index.d.ts","./node_modules/@nestjs/common/interfaces/global-prefix-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/before-application-shutdown.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-application-bootstrap.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-application-shutdown.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-destroy.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-init.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/index.d.ts","./node_modules/@nestjs/common/interfaces/http/http-exception-body.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/http-redirect-response.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/cors-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/https-options.interface.d.ts","./node_modules/@nestjs/common/services/logger.service.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-context-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/http-server.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/message-event.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/raw-body-request.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/index.d.ts","./node_modules/@nestjs/common/interfaces/injectable.interface.d.ts","./node_modules/@nestjs/common/interfaces/microservices/nest-hybrid-application-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/forward-reference.interface.d.ts","./node_modules/@nestjs/common/interfaces/scope-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/injection-token.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/optional-factory-dependency.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/provider.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/module-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/dynamic-module.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/introspection-result.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/nest-module.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/index.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-context.interface.d.ts","./node_modules/@nestjs/common/interfaces/websockets/web-socket-adapter.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-application.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-microservice.interface.d.ts","./node_modules/@nestjs/common/interfaces/index.d.ts","./node_modules/@nestjs/common/decorators/core/catch.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/controller.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/dependencies.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/exception-filters.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/inject.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/injectable.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/optional.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/set-metadata.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-guards.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-interceptors.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-pipes.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/apply-decorators.d.ts","./node_modules/@nestjs/common/decorators/core/version.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/index.d.ts","./node_modules/@nestjs/common/decorators/modules/global.decorator.d.ts","./node_modules/@nestjs/common/decorators/modules/module.decorator.d.ts","./node_modules/@nestjs/common/decorators/modules/index.d.ts","./node_modules/@nestjs/common/decorators/http/request-mapping.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/route-params.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/http-code.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/create-route-param-metadata.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/render.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/header.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/redirect.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/sse.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/index.d.ts","./node_modules/@nestjs/common/decorators/index.d.ts","./node_modules/@nestjs/common/exceptions/http.exception.d.ts","./node_modules/@nestjs/common/exceptions/bad-request.exception.d.ts","./node_modules/@nestjs/common/exceptions/unauthorized.exception.d.ts","./node_modules/@nestjs/common/exceptions/method-not-allowed.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-found.exception.d.ts","./node_modules/@nestjs/common/exceptions/forbidden.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-acceptable.exception.d.ts","./node_modules/@nestjs/common/exceptions/request-timeout.exception.d.ts","./node_modules/@nestjs/common/exceptions/conflict.exception.d.ts","./node_modules/@nestjs/common/exceptions/gone.exception.d.ts","./node_modules/@nestjs/common/exceptions/payload-too-large.exception.d.ts","./node_modules/@nestjs/common/exceptions/unsupported-media-type.exception.d.ts","./node_modules/@nestjs/common/exceptions/unprocessable-entity.exception.d.ts","./node_modules/@nestjs/common/exceptions/internal-server-error.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-implemented.exception.d.ts","./node_modules/@nestjs/common/exceptions/http-version-not-supported.exception.d.ts","./node_modules/@nestjs/common/exceptions/bad-gateway.exception.d.ts","./node_modules/@nestjs/common/exceptions/service-unavailable.exception.d.ts","./node_modules/@nestjs/common/exceptions/gateway-timeout.exception.d.ts","./node_modules/@nestjs/common/exceptions/im-a-teapot.exception.d.ts","./node_modules/@nestjs/common/exceptions/precondition-failed.exception.d.ts","./node_modules/@nestjs/common/exceptions/misdirected.exception.d.ts","./node_modules/@nestjs/common/exceptions/index.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/streamable-options.interface.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/streamable-handler-response.interface.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/index.d.ts","./node_modules/@nestjs/common/services/console-logger.service.d.ts","./node_modules/@nestjs/common/services/index.d.ts","./node_modules/@nestjs/common/file-stream/streamable-file.d.ts","./node_modules/@nestjs/common/file-stream/index.d.ts","./node_modules/@nestjs/common/module-utils/constants.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-async-options.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-cls.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-host.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/index.d.ts","./node_modules/@nestjs/common/module-utils/configurable-module.builder.d.ts","./node_modules/@nestjs/common/module-utils/index.d.ts","./node_modules/@nestjs/common/pipes/default-value.pipe.d.ts","./node_modules/@nestjs/common/interfaces/external/class-transform-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/transformer-package.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validator-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validator-package.interface.d.ts","./node_modules/@nestjs/common/utils/http-error-by-code.util.d.ts","./node_modules/@nestjs/common/pipes/validation.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-array.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-bool.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-int.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-float.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-enum.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-uuid.pipe.d.ts","./node_modules/@nestjs/common/pipes/file/interfaces/file.interface.d.ts","./node_modules/@nestjs/common/pipes/file/interfaces/index.d.ts","./node_modules/@nestjs/common/pipes/file/file-validator.interface.d.ts","./node_modules/@nestjs/common/pipes/file/file-type.validator.d.ts","./node_modules/@nestjs/common/pipes/file/max-file-size.validator.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file-options.interface.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file.pipe.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file-pipe.builder.d.ts","./node_modules/@nestjs/common/pipes/file/index.d.ts","./node_modules/@nestjs/common/pipes/index.d.ts","./node_modules/@nestjs/common/serializer/class-serializer.interfaces.d.ts","./node_modules/@nestjs/common/serializer/class-serializer.interceptor.d.ts","./node_modules/@nestjs/common/serializer/decorators/serialize-options.decorator.d.ts","./node_modules/@nestjs/common/serializer/decorators/index.d.ts","./node_modules/@nestjs/common/serializer/index.d.ts","./node_modules/@nestjs/common/utils/forward-ref.util.d.ts","./node_modules/@nestjs/common/utils/index.d.ts","./node_modules/@nestjs/common/index.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-basic.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-bearer.decorator.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/open-api-spec.interface.d.ts","./node_modules/@nestjs/swagger/dist/types/swagger-enum.type.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-body.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-consumes.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-cookie.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-exclude-endpoint.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-exclude-controller.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-extra-models.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-header.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-hide-property.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-oauth2.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-operation.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-param.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-produces.decorator.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/schema-object-metadata.interface.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-property.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-query.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-response.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-security.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-use-tags.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-extension.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/index.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-ui-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-custom-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-document-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/index.d.ts","./node_modules/@nestjs/swagger/dist/document-builder.d.ts","./node_modules/@nestjs/swagger/dist/swagger-module.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/intersection-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/omit-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/partial-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/pick-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/index.d.ts","./node_modules/@nestjs/swagger/dist/utils/get-schema-path.util.d.ts","./node_modules/@nestjs/swagger/dist/utils/index.d.ts","./node_modules/@nestjs/swagger/dist/index.d.ts","./node_modules/@nestjs/swagger/index.d.ts","./src/app.service.ts","./src/app.controller.ts","./node_modules/@nestjs/core/adapters/http-adapter.d.ts","./node_modules/@nestjs/core/adapters/index.d.ts","./node_modules/@nestjs/common/constants.d.ts","./node_modules/@nestjs/core/inspector/interfaces/edge.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/entrypoint.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/extras.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/node.interface.d.ts","./node_modules/@nestjs/core/injector/settlement-signal.d.ts","./node_modules/@nestjs/core/injector/injector.d.ts","./node_modules/@nestjs/core/inspector/interfaces/serialized-graph-metadata.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/serialized-graph-json.interface.d.ts","./node_modules/@nestjs/core/inspector/serialized-graph.d.ts","./node_modules/@nestjs/core/injector/module-token-factory.d.ts","./node_modules/@nestjs/core/injector/compiler.d.ts","./node_modules/@nestjs/core/injector/modules-container.d.ts","./node_modules/@nestjs/core/injector/container.d.ts","./node_modules/@nestjs/core/injector/instance-links-host.d.ts","./node_modules/@nestjs/core/injector/abstract-instance-resolver.d.ts","./node_modules/@nestjs/core/injector/module-ref.d.ts","./node_modules/@nestjs/core/injector/module.d.ts","./node_modules/@nestjs/core/injector/instance-wrapper.d.ts","./node_modules/@nestjs/core/router/interfaces/exclude-route-metadata.interface.d.ts","./node_modules/@nestjs/core/application-config.d.ts","./node_modules/@nestjs/core/constants.d.ts","./node_modules/@nestjs/core/discovery/discovery-module.d.ts","./node_modules/@nestjs/core/discovery/discovery-service.d.ts","./node_modules/@nestjs/core/discovery/index.d.ts","./node_modules/@nestjs/core/helpers/http-adapter-host.d.ts","./node_modules/@nestjs/core/exceptions/base-exception-filter.d.ts","./node_modules/@nestjs/core/exceptions/index.d.ts","./node_modules/@nestjs/core/helpers/context-id-factory.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/exception-filter-metadata.interface.d.ts","./node_modules/@nestjs/core/exceptions/exceptions-handler.d.ts","./node_modules/@nestjs/core/router/router-proxy.d.ts","./node_modules/@nestjs/core/helpers/context-creator.d.ts","./node_modules/@nestjs/core/exceptions/base-exception-filter-context.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/rpc-exception-filter-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/index.d.ts","./node_modules/@nestjs/core/exceptions/external-exception-filter.d.ts","./node_modules/@nestjs/core/exceptions/external-exceptions-handler.d.ts","./node_modules/@nestjs/core/exceptions/external-exception-filter-context.d.ts","./node_modules/@nestjs/core/guards/constants.d.ts","./node_modules/@nestjs/core/helpers/execution-context-host.d.ts","./node_modules/@nestjs/core/guards/guards-consumer.d.ts","./node_modules/@nestjs/core/guards/guards-context-creator.d.ts","./node_modules/@nestjs/core/guards/index.d.ts","./node_modules/@nestjs/core/interceptors/interceptors-consumer.d.ts","./node_modules/@nestjs/core/interceptors/interceptors-context-creator.d.ts","./node_modules/@nestjs/core/interceptors/index.d.ts","./node_modules/@nestjs/common/enums/route-paramtypes.enum.d.ts","./node_modules/@nestjs/core/pipes/params-token-factory.d.ts","./node_modules/@nestjs/core/pipes/pipes-consumer.d.ts","./node_modules/@nestjs/core/pipes/pipes-context-creator.d.ts","./node_modules/@nestjs/core/pipes/index.d.ts","./node_modules/@nestjs/core/helpers/context-utils.d.ts","./node_modules/@nestjs/core/injector/inquirer/inquirer-constants.d.ts","./node_modules/@nestjs/core/injector/inquirer/index.d.ts","./node_modules/@nestjs/core/interfaces/module-definition.interface.d.ts","./node_modules/@nestjs/core/interfaces/module-override.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/enhancer-metadata-cache-entry.interface.d.ts","./node_modules/@nestjs/core/inspector/graph-inspector.d.ts","./node_modules/@nestjs/core/metadata-scanner.d.ts","./node_modules/@nestjs/core/scanner.d.ts","./node_modules/@nestjs/core/injector/instance-loader.d.ts","./node_modules/@nestjs/core/injector/lazy-module-loader/lazy-module-loader-options.interface.d.ts","./node_modules/@nestjs/core/injector/lazy-module-loader/lazy-module-loader.d.ts","./node_modules/@nestjs/core/injector/index.d.ts","./node_modules/@nestjs/core/helpers/interfaces/external-handler-metadata.interface.d.ts","./node_modules/@nestjs/core/helpers/interfaces/params-metadata.interface.d.ts","./node_modules/@nestjs/core/helpers/external-context-creator.d.ts","./node_modules/@nestjs/core/helpers/index.d.ts","./node_modules/@nestjs/core/inspector/initialize-on-preview.allowlist.d.ts","./node_modules/@nestjs/core/inspector/partial-graph.host.d.ts","./node_modules/@nestjs/core/inspector/index.d.ts","./node_modules/@nestjs/core/middleware/route-info-path-extractor.d.ts","./node_modules/@nestjs/core/middleware/routes-mapper.d.ts","./node_modules/@nestjs/core/middleware/builder.d.ts","./node_modules/@nestjs/core/middleware/index.d.ts","./node_modules/@nestjs/core/nest-application-context.d.ts","./node_modules/@nestjs/core/nest-application.d.ts","./node_modules/@nestjs/common/interfaces/microservices/nest-microservice-options.interface.d.ts","./node_modules/@nestjs/core/nest-factory.d.ts","./node_modules/@nestjs/core/repl/repl.d.ts","./node_modules/@nestjs/core/repl/index.d.ts","./node_modules/@nestjs/core/router/interfaces/routes.interface.d.ts","./node_modules/@nestjs/core/router/interfaces/index.d.ts","./node_modules/@nestjs/core/router/request/request-constants.d.ts","./node_modules/@nestjs/core/router/request/index.d.ts","./node_modules/@nestjs/core/router/router-module.d.ts","./node_modules/@nestjs/core/router/index.d.ts","./node_modules/@nestjs/core/services/reflector.service.d.ts","./node_modules/@nestjs/core/services/index.d.ts","./node_modules/@nestjs/core/index.d.ts","./node_modules/@nestjs/config/dist/conditional.module.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-change-event.interface.d.ts","./node_modules/@nestjs/config/dist/types/config-object.type.d.ts","./node_modules/@nestjs/config/dist/types/config.type.d.ts","./node_modules/@nestjs/config/dist/types/no-infer.type.d.ts","./node_modules/@nestjs/config/dist/types/path-value.type.d.ts","./node_modules/@nestjs/config/dist/types/index.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-factory.interface.d.ts","./node_modules/@types/node/compatibility/disposable.d.ts","./node_modules/@types/node/compatibility/indexable.d.ts","./node_modules/@types/node/compatibility/iterators.d.ts","./node_modules/@types/node/compatibility/index.d.ts","./node_modules/@types/node/globals.typedarray.d.ts","./node_modules/@types/node/buffer.buffer.d.ts","./node_modules/@types/node/globals.d.ts","./node_modules/@types/node/web-globals/abortcontroller.d.ts","./node_modules/@types/node/web-globals/domexception.d.ts","./node_modules/@types/node/web-globals/events.d.ts","./node_modules/buffer/index.d.ts","./node_modules/undici-types/header.d.ts","./node_modules/undici-types/readable.d.ts","./node_modules/undici-types/file.d.ts","./node_modules/undici-types/fetch.d.ts","./node_modules/undici-types/formdata.d.ts","./node_modules/undici-types/connector.d.ts","./node_modules/undici-types/client.d.ts","./node_modules/undici-types/errors.d.ts","./node_modules/undici-types/dispatcher.d.ts","./node_modules/undici-types/global-dispatcher.d.ts","./node_modules/undici-types/global-origin.d.ts","./node_modules/undici-types/pool-stats.d.ts","./node_modules/undici-types/pool.d.ts","./node_modules/undici-types/handlers.d.ts","./node_modules/undici-types/balanced-pool.d.ts","./node_modules/undici-types/agent.d.ts","./node_modules/undici-types/mock-interceptor.d.ts","./node_modules/undici-types/mock-agent.d.ts","./node_modules/undici-types/mock-client.d.ts","./node_modules/undici-types/mock-pool.d.ts","./node_modules/undici-types/mock-errors.d.ts","./node_modules/undici-types/proxy-agent.d.ts","./node_modules/undici-types/env-http-proxy-agent.d.ts","./node_modules/undici-types/retry-handler.d.ts","./node_modules/undici-types/retry-agent.d.ts","./node_modules/undici-types/api.d.ts","./node_modules/undici-types/interceptors.d.ts","./node_modules/undici-types/util.d.ts","./node_modules/undici-types/cookies.d.ts","./node_modules/undici-types/patch.d.ts","./node_modules/undici-types/websocket.d.ts","./node_modules/undici-types/eventsource.d.ts","./node_modules/undici-types/filereader.d.ts","./node_modules/undici-types/diagnostics-channel.d.ts","./node_modules/undici-types/content-type.d.ts","./node_modules/undici-types/cache.d.ts","./node_modules/undici-types/index.d.ts","./node_modules/@types/node/web-globals/fetch.d.ts","./node_modules/@types/node/assert.d.ts","./node_modules/@types/node/assert/strict.d.ts","./node_modules/@types/node/async_hooks.d.ts","./node_modules/@types/node/buffer.d.ts","./node_modules/@types/node/child_process.d.ts","./node_modules/@types/node/cluster.d.ts","./node_modules/@types/node/console.d.ts","./node_modules/@types/node/constants.d.ts","./node_modules/@types/node/crypto.d.ts","./node_modules/@types/node/dgram.d.ts","./node_modules/@types/node/diagnostics_channel.d.ts","./node_modules/@types/node/dns.d.ts","./node_modules/@types/node/dns/promises.d.ts","./node_modules/@types/node/domain.d.ts","./node_modules/@types/node/events.d.ts","./node_modules/@types/node/fs.d.ts","./node_modules/@types/node/fs/promises.d.ts","./node_modules/@types/node/http.d.ts","./node_modules/@types/node/http2.d.ts","./node_modules/@types/node/https.d.ts","./node_modules/@types/node/inspector.generated.d.ts","./node_modules/@types/node/module.d.ts","./node_modules/@types/node/net.d.ts","./node_modules/@types/node/os.d.ts","./node_modules/@types/node/path.d.ts","./node_modules/@types/node/perf_hooks.d.ts","./node_modules/@types/node/process.d.ts","./node_modules/@types/node/punycode.d.ts","./node_modules/@types/node/querystring.d.ts","./node_modules/@types/node/readline.d.ts","./node_modules/@types/node/readline/promises.d.ts","./node_modules/@types/node/repl.d.ts","./node_modules/@types/node/sea.d.ts","./node_modules/@types/node/stream.d.ts","./node_modules/@types/node/stream/promises.d.ts","./node_modules/@types/node/stream/consumers.d.ts","./node_modules/@types/node/stream/web.d.ts","./node_modules/@types/node/string_decoder.d.ts","./node_modules/@types/node/test.d.ts","./node_modules/@types/node/timers.d.ts","./node_modules/@types/node/timers/promises.d.ts","./node_modules/@types/node/tls.d.ts","./node_modules/@types/node/trace_events.d.ts","./node_modules/@types/node/tty.d.ts","./node_modules/@types/node/url.d.ts","./node_modules/@types/node/util.d.ts","./node_modules/@types/node/v8.d.ts","./node_modules/@types/node/vm.d.ts","./node_modules/@types/node/wasi.d.ts","./node_modules/@types/node/worker_threads.d.ts","./node_modules/@types/node/zlib.d.ts","./node_modules/@types/node/index.d.ts","./node_modules/dotenv-expand/lib/main.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-module-options.interface.d.ts","./node_modules/@nestjs/config/dist/interfaces/index.d.ts","./node_modules/@nestjs/config/dist/config.module.d.ts","./node_modules/@nestjs/config/dist/config.service.d.ts","./node_modules/@nestjs/config/dist/utils/register-as.util.d.ts","./node_modules/@nestjs/config/dist/utils/get-config-token.util.d.ts","./node_modules/@nestjs/config/dist/utils/index.d.ts","./node_modules/@nestjs/config/dist/index.d.ts","./node_modules/@nestjs/config/index.d.ts","./node_modules/typeorm/metadata/types/relationtypes.d.ts","./node_modules/typeorm/metadata/types/deferrabletype.d.ts","./node_modules/typeorm/metadata/types/ondeletetype.d.ts","./node_modules/typeorm/metadata/types/onupdatetype.d.ts","./node_modules/typeorm/decorator/options/relationoptions.d.ts","./node_modules/typeorm/metadata/types/propertytypeinfunction.d.ts","./node_modules/typeorm/common/objecttype.d.ts","./node_modules/typeorm/common/entitytarget.d.ts","./node_modules/typeorm/metadata/types/relationtypeinfunction.d.ts","./node_modules/typeorm/metadata-args/relationmetadataargs.d.ts","./node_modules/typeorm/driver/types/columntypes.d.ts","./node_modules/typeorm/decorator/options/valuetransformer.d.ts","./node_modules/typeorm/decorator/options/columncommonoptions.d.ts","./node_modules/typeorm/decorator/options/columnoptions.d.ts","./node_modules/typeorm/metadata-args/types/columnmode.d.ts","./node_modules/typeorm/metadata-args/columnmetadataargs.d.ts","./node_modules/typeorm/common/objectliteral.d.ts","./node_modules/typeorm/schema-builder/options/tablecolumnoptions.d.ts","./node_modules/typeorm/schema-builder/table/tablecolumn.d.ts","./node_modules/typeorm/schema-builder/options/viewoptions.d.ts","./node_modules/typeorm/schema-builder/view/view.d.ts","./node_modules/typeorm/naming-strategy/namingstrategyinterface.d.ts","./node_modules/typeorm/metadata/foreignkeymetadata.d.ts","./node_modules/typeorm/metadata/relationmetadata.d.ts","./node_modules/typeorm/metadata-args/embeddedmetadataargs.d.ts","./node_modules/typeorm/metadata-args/relationidmetadataargs.d.ts","./node_modules/typeorm/metadata/relationidmetadata.d.ts","./node_modules/typeorm/metadata/relationcountmetadata.d.ts","./node_modules/typeorm/metadata/types/eventlistenertypes.d.ts","./node_modules/typeorm/metadata-args/entitylistenermetadataargs.d.ts","./node_modules/typeorm/metadata/entitylistenermetadata.d.ts","./node_modules/typeorm/metadata-args/uniquemetadataargs.d.ts","./node_modules/typeorm/metadata/uniquemetadata.d.ts","./node_modules/typeorm/metadata/embeddedmetadata.d.ts","./node_modules/typeorm/metadata/columnmetadata.d.ts","./node_modules/typeorm/driver/types/ctecapabilities.d.ts","./node_modules/typeorm/driver/types/mappedcolumntypes.d.ts","./node_modules/typeorm/driver/query.d.ts","./node_modules/typeorm/driver/sqlinmemory.d.ts","./node_modules/typeorm/schema-builder/schemabuilder.d.ts","./node_modules/typeorm/driver/types/datatypedefaults.d.ts","./node_modules/typeorm/entity-schema/entityschemaindexoptions.d.ts","./node_modules/typeorm/driver/types/geojsontypes.d.ts","./node_modules/typeorm/decorator/options/spatialcolumnoptions.d.ts","./node_modules/typeorm/decorator/options/foreignkeyoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemacolumnforeignkeyoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemacolumnoptions.d.ts","./node_modules/typeorm/decorator/options/joincolumnoptions.d.ts","./node_modules/typeorm/decorator/options/jointablemultiplecolumnsoptions.d.ts","./node_modules/typeorm/decorator/options/jointableoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemarelationoptions.d.ts","./node_modules/typeorm/find-options/orderbycondition.d.ts","./node_modules/typeorm/metadata/types/tabletypes.d.ts","./node_modules/typeorm/entity-schema/entityschemauniqueoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemacheckoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemaexclusionoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemainheritanceoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemarelationidoptions.d.ts","./node_modules/typeorm/entity-schema/entityschemaforeignkeyoptions.d.ts","./node_modules/typeorm/metadata/types/treetypes.d.ts","./node_modules/typeorm/metadata/types/closuretreeoptions.d.ts","./node_modules/typeorm/metadata-args/treemetadataargs.d.ts","./node_modules/typeorm/entity-schema/entityschemaoptions.d.ts","./node_modules/typeorm/entity-schema/entityschema.d.ts","./node_modules/typeorm/logger/logger.d.ts","./node_modules/typeorm/logger/loggeroptions.d.ts","./node_modules/typeorm/driver/types/databasetype.d.ts","./node_modules/typeorm/cache/queryresultcacheoptions.d.ts","./node_modules/typeorm/cache/queryresultcache.d.ts","./node_modules/typeorm/common/mixedlist.d.ts","./node_modules/typeorm/data-source/basedatasourceoptions.d.ts","./node_modules/typeorm/driver/types/replicationmode.d.ts","./node_modules/typeorm/schema-builder/options/tableforeignkeyoptions.d.ts","./node_modules/typeorm/schema-builder/table/tableforeignkey.d.ts","./node_modules/typeorm/driver/types/upserttype.d.ts","./node_modules/typeorm/driver/driver.d.ts","./node_modules/typeorm/find-options/joinoptions.d.ts","./node_modules/typeorm/find-options/findoperatortype.d.ts","./node_modules/typeorm/find-options/findoperator.d.ts","./node_modules/typeorm/platform/platformtools.d.ts","./node_modules/typeorm/driver/mongodb/bson.typings.d.ts","./node_modules/typeorm/driver/mongodb/typings.d.ts","./node_modules/typeorm/find-options/equaloperator.d.ts","./node_modules/typeorm/find-options/findoptionswhere.d.ts","./node_modules/typeorm/find-options/findoptionsselect.d.ts","./node_modules/typeorm/find-options/findoptionsrelations.d.ts","./node_modules/typeorm/find-options/findoptionsorder.d.ts","./node_modules/typeorm/find-options/findoneoptions.d.ts","./node_modules/typeorm/find-options/findmanyoptions.d.ts","./node_modules/typeorm/common/deeppartial.d.ts","./node_modules/typeorm/repository/saveoptions.d.ts","./node_modules/typeorm/repository/removeoptions.d.ts","./node_modules/typeorm/find-options/mongodb/mongofindoneoptions.d.ts","./node_modules/typeorm/find-options/mongodb/mongofindmanyoptions.d.ts","./node_modules/typeorm/schema-builder/options/tableuniqueoptions.d.ts","./node_modules/typeorm/schema-builder/table/tableunique.d.ts","./node_modules/typeorm/subscriber/broadcasterresult.d.ts","./node_modules/typeorm/subscriber/event/transactioncommitevent.d.ts","./node_modules/typeorm/subscriber/event/transactionrollbackevent.d.ts","./node_modules/typeorm/subscriber/event/transactionstartevent.d.ts","./node_modules/typeorm/subscriber/event/updateevent.d.ts","./node_modules/typeorm/subscriber/event/removeevent.d.ts","./node_modules/typeorm/subscriber/event/insertevent.d.ts","./node_modules/typeorm/subscriber/event/loadevent.d.ts","./node_modules/typeorm/subscriber/event/softremoveevent.d.ts","./node_modules/typeorm/subscriber/event/recoverevent.d.ts","./node_modules/typeorm/subscriber/event/queryevent.d.ts","./node_modules/typeorm/subscriber/entitysubscriberinterface.d.ts","./node_modules/typeorm/subscriber/broadcaster.d.ts","./node_modules/typeorm/schema-builder/options/tablecheckoptions.d.ts","./node_modules/typeorm/metadata-args/checkmetadataargs.d.ts","./node_modules/typeorm/metadata/checkmetadata.d.ts","./node_modules/typeorm/schema-builder/table/tablecheck.d.ts","./node_modules/typeorm/schema-builder/options/tableexclusionoptions.d.ts","./node_modules/typeorm/metadata-args/exclusionmetadataargs.d.ts","./node_modules/typeorm/metadata/exclusionmetadata.d.ts","./node_modules/typeorm/schema-builder/table/tableexclusion.d.ts","./node_modules/typeorm/driver/mongodb/mongoqueryrunner.d.ts","./node_modules/typeorm/query-builder/querypartialentity.d.ts","./node_modules/typeorm/query-runner/queryresult.d.ts","./node_modules/typeorm/query-builder/result/insertresult.d.ts","./node_modules/typeorm/query-builder/result/updateresult.d.ts","./node_modules/typeorm/query-builder/result/deleteresult.d.ts","./node_modules/typeorm/entity-manager/mongoentitymanager.d.ts","./node_modules/typeorm/repository/mongorepository.d.ts","./node_modules/typeorm/find-options/findtreeoptions.d.ts","./node_modules/typeorm/repository/treerepository.d.ts","./node_modules/typeorm/query-builder/transformer/plainobjecttonewentitytransformer.d.ts","./node_modules/typeorm/driver/types/isolationlevel.d.ts","./node_modules/typeorm/query-builder/whereexpressionbuilder.d.ts","./node_modules/typeorm/query-builder/brackets.d.ts","./node_modules/typeorm/query-builder/insertorupdateoptions.d.ts","./node_modules/typeorm/repository/upsertoptions.d.ts","./node_modules/typeorm/common/pickkeysbytype.d.ts","./node_modules/typeorm/entity-manager/entitymanager.d.ts","./node_modules/typeorm/repository/repository.d.ts","./node_modules/typeorm/migration/migrationinterface.d.ts","./node_modules/typeorm/migration/migration.d.ts","./node_modules/typeorm/driver/cockroachdb/cockroachconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/cockroachdb/cockroachconnectionoptions.d.ts","./node_modules/typeorm/driver/mysql/mysqlconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/mysql/mysqlconnectionoptions.d.ts","./node_modules/typeorm/driver/postgres/postgresconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/postgres/postgresconnectionoptions.d.ts","./node_modules/typeorm/driver/sqlite/sqliteconnectionoptions.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/defaultauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectoryaccesstokenauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectorydefaultauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectorymsiappserviceauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectorymsivmauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectorypasswordauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/azureactivedirectoryserviceprincipalsecret.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/ntlmauthentication.d.ts","./node_modules/typeorm/driver/sqlserver/sqlserverconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/sqlserver/sqlserverconnectionoptions.d.ts","./node_modules/typeorm/driver/oracle/oracleconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/oracle/oracleconnectionoptions.d.ts","./node_modules/typeorm/driver/mongodb/mongoconnectionoptions.d.ts","./node_modules/typeorm/driver/cordova/cordovaconnectionoptions.d.ts","./node_modules/typeorm/driver/sqljs/sqljsconnectionoptions.d.ts","./node_modules/typeorm/driver/react-native/reactnativeconnectionoptions.d.ts","./node_modules/typeorm/driver/nativescript/nativescriptconnectionoptions.d.ts","./node_modules/typeorm/driver/expo/expoconnectionoptions.d.ts","./node_modules/typeorm/driver/aurora-mysql/auroramysqlconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/aurora-mysql/auroramysqlconnectionoptions.d.ts","./node_modules/typeorm/driver/sap/sapconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/sap/sapconnectionoptions.d.ts","./node_modules/typeorm/driver/aurora-postgres/aurorapostgresconnectionoptions.d.ts","./node_modules/typeorm/driver/better-sqlite3/bettersqlite3connectionoptions.d.ts","./node_modules/typeorm/driver/capacitor/capacitorconnectionoptions.d.ts","./node_modules/typeorm/connection/baseconnectionoptions.d.ts","./node_modules/typeorm/driver/spanner/spannerconnectioncredentialsoptions.d.ts","./node_modules/typeorm/driver/spanner/spannerconnectionoptions.d.ts","./node_modules/typeorm/data-source/datasourceoptions.d.ts","./node_modules/typeorm/entity-manager/sqljsentitymanager.d.ts","./node_modules/typeorm/query-builder/relationloader.d.ts","./node_modules/typeorm/query-builder/relationidloader.d.ts","./node_modules/typeorm/data-source/datasource.d.ts","./node_modules/typeorm/metadata-args/tablemetadataargs.d.ts","./node_modules/typeorm/metadata/entitymetadata.d.ts","./node_modules/typeorm/metadata-args/indexmetadataargs.d.ts","./node_modules/typeorm/metadata/indexmetadata.d.ts","./node_modules/typeorm/schema-builder/options/tableindexoptions.d.ts","./node_modules/typeorm/schema-builder/table/tableindex.d.ts","./node_modules/typeorm/schema-builder/options/tableoptions.d.ts","./node_modules/typeorm/schema-builder/table/table.d.ts","./node_modules/typeorm/query-runner/queryrunner.d.ts","./node_modules/typeorm/query-builder/querybuildercte.d.ts","./node_modules/typeorm/query-builder/alias.d.ts","./node_modules/typeorm/query-builder/joinattribute.d.ts","./node_modules/typeorm/query-builder/relation-id/relationidattribute.d.ts","./node_modules/typeorm/query-builder/relation-count/relationcountattribute.d.ts","./node_modules/typeorm/query-builder/selectquery.d.ts","./node_modules/typeorm/query-builder/selectquerybuilderoption.d.ts","./node_modules/typeorm/query-builder/whereclause.d.ts","./node_modules/typeorm/query-builder/queryexpressionmap.d.ts","./node_modules/typeorm/query-builder/updatequerybuilder.d.ts","./node_modules/typeorm/query-builder/deletequerybuilder.d.ts","./node_modules/typeorm/query-builder/softdeletequerybuilder.d.ts","./node_modules/typeorm/query-builder/insertquerybuilder.d.ts","./node_modules/typeorm/query-builder/relationquerybuilder.d.ts","./node_modules/typeorm/query-builder/notbrackets.d.ts","./node_modules/typeorm/query-builder/querybuilder.d.ts","./node_modules/typeorm/query-builder/selectquerybuilder.d.ts","./node_modules/typeorm/metadata-args/relationcountmetadataargs.d.ts","./node_modules/typeorm/metadata-args/namingstrategymetadataargs.d.ts","./node_modules/typeorm/metadata-args/joincolumnmetadataargs.d.ts","./node_modules/typeorm/metadata-args/jointablemetadataargs.d.ts","./node_modules/typeorm/metadata-args/entitysubscribermetadataargs.d.ts","./node_modules/typeorm/metadata-args/inheritancemetadataargs.d.ts","./node_modules/typeorm/metadata-args/discriminatorvaluemetadataargs.d.ts","./node_modules/typeorm/metadata-args/entityrepositorymetadataargs.d.ts","./node_modules/typeorm/metadata-args/transactionentitymetadataargs.d.ts","./node_modules/typeorm/metadata-args/transactionrepositorymetadataargs.d.ts","./node_modules/typeorm/metadata-args/generatedmetadataargs.d.ts","./node_modules/typeorm/metadata-args/foreignkeymetadataargs.d.ts","./node_modules/typeorm/metadata-args/metadataargsstorage.d.ts","./node_modules/typeorm/connection/connectionmanager.d.ts","./node_modules/typeorm/globals.d.ts","./node_modules/typeorm/container.d.ts","./node_modules/typeorm/common/relationtype.d.ts","./node_modules/typeorm/error/typeormerror.d.ts","./node_modules/typeorm/error/cannotreflectmethodparametertypeerror.d.ts","./node_modules/typeorm/error/alreadyhasactiveconnectionerror.d.ts","./node_modules/typeorm/persistence/subjectchangemap.d.ts","./node_modules/typeorm/persistence/subject.d.ts","./node_modules/typeorm/error/subjectwithoutidentifiererror.d.ts","./node_modules/typeorm/error/cannotconnectalreadyconnectederror.d.ts","./node_modules/typeorm/error/locknotsupportedongivendrivererror.d.ts","./node_modules/typeorm/error/connectionisnotseterror.d.ts","./node_modules/typeorm/error/cannotcreateentityidmaperror.d.ts","./node_modules/typeorm/error/metadataalreadyexistserror.d.ts","./node_modules/typeorm/error/cannotdetermineentityerror.d.ts","./node_modules/typeorm/error/updatevaluesmissingerror.d.ts","./node_modules/typeorm/error/treerepositorynotsupportederror.d.ts","./node_modules/typeorm/error/customrepositorynotfounderror.d.ts","./node_modules/typeorm/error/transactionnotstartederror.d.ts","./node_modules/typeorm/error/transactionalreadystartederror.d.ts","./node_modules/typeorm/error/entitynotfounderror.d.ts","./node_modules/typeorm/error/entitymetadatanotfounderror.d.ts","./node_modules/typeorm/error/mustbeentityerror.d.ts","./node_modules/typeorm/error/optimisticlockversionmismatcherror.d.ts","./node_modules/typeorm/error/limitonupdatenotsupportederror.d.ts","./node_modules/typeorm/error/primarycolumncannotbenullableerror.d.ts","./node_modules/typeorm/error/customrepositorycannotinheritrepositoryerror.d.ts","./node_modules/typeorm/error/queryrunnerprovideralreadyreleasederror.d.ts","./node_modules/typeorm/error/cannotattachtreechildrenentityerror.d.ts","./node_modules/typeorm/error/customrepositorydoesnothaveentityerror.d.ts","./node_modules/typeorm/error/missingdeletedatecolumnerror.d.ts","./node_modules/typeorm/error/noconnectionforrepositoryerror.d.ts","./node_modules/typeorm/error/circularrelationserror.d.ts","./node_modules/typeorm/error/returningstatementnotsupportederror.d.ts","./node_modules/typeorm/error/usingjointableisnotallowederror.d.ts","./node_modules/typeorm/error/missingjoincolumnerror.d.ts","./node_modules/typeorm/error/missingprimarycolumnerror.d.ts","./node_modules/typeorm/error/entitypropertynotfounderror.d.ts","./node_modules/typeorm/error/missingdrivererror.d.ts","./node_modules/typeorm/error/driverpackagenotinstallederror.d.ts","./node_modules/typeorm/error/cannotgetentitymanagernotconnectederror.d.ts","./node_modules/typeorm/error/connectionnotfounderror.d.ts","./node_modules/typeorm/error/noversionorupdatedatecolumnerror.d.ts","./node_modules/typeorm/error/insertvaluesmissingerror.d.ts","./node_modules/typeorm/error/optimisticlockcannotbeusederror.d.ts","./node_modules/typeorm/error/metadatawithsuchnamealreadyexistserror.d.ts","./node_modules/typeorm/error/driveroptionnotseterror.d.ts","./node_modules/typeorm/error/findrelationsnotfounderror.d.ts","./node_modules/typeorm/error/pessimisticlocktransactionrequirederror.d.ts","./node_modules/typeorm/error/repositorynottreeerror.d.ts","./node_modules/typeorm/error/datatypenotsupportederror.d.ts","./node_modules/typeorm/error/initializedrelationerror.d.ts","./node_modules/typeorm/error/missingjointableerror.d.ts","./node_modules/typeorm/error/queryfailederror.d.ts","./node_modules/typeorm/error/noneedtoreleaseentitymanagererror.d.ts","./node_modules/typeorm/error/usingjoincolumnonlyononesideallowederror.d.ts","./node_modules/typeorm/error/usingjointableonlyononesideallowederror.d.ts","./node_modules/typeorm/error/subjectremovedandupdatederror.d.ts","./node_modules/typeorm/error/persistedentitynotfounderror.d.ts","./node_modules/typeorm/error/usingjoincolumnisnotallowederror.d.ts","./node_modules/typeorm/error/columntypeundefinederror.d.ts","./node_modules/typeorm/error/queryrunneralreadyreleasederror.d.ts","./node_modules/typeorm/error/offsetwithoutlimitnotsupportederror.d.ts","./node_modules/typeorm/error/cannotexecutenotconnectederror.d.ts","./node_modules/typeorm/error/noconnectionoptionerror.d.ts","./node_modules/typeorm/error/forbiddentransactionmodeoverrideerror.d.ts","./node_modules/typeorm/error/index.d.ts","./node_modules/typeorm/decorator/options/columnembeddedoptions.d.ts","./node_modules/typeorm/decorator/options/columnenumoptions.d.ts","./node_modules/typeorm/decorator/options/columnhstoreoptions.d.ts","./node_modules/typeorm/decorator/options/columnnumericoptions.d.ts","./node_modules/typeorm/decorator/options/columnunsignedoptions.d.ts","./node_modules/typeorm/decorator/options/columnwithlengthoptions.d.ts","./node_modules/typeorm/decorator/columns/column.d.ts","./node_modules/typeorm/decorator/columns/createdatecolumn.d.ts","./node_modules/typeorm/decorator/columns/deletedatecolumn.d.ts","./node_modules/typeorm/decorator/options/primarygeneratedcolumnnumericoptions.d.ts","./node_modules/typeorm/decorator/options/primarygeneratedcolumnuuidoptions.d.ts","./node_modules/typeorm/decorator/options/primarygeneratedcolumnidentityoptions.d.ts","./node_modules/typeorm/decorator/columns/primarygeneratedcolumn.d.ts","./node_modules/typeorm/decorator/columns/primarycolumn.d.ts","./node_modules/typeorm/decorator/columns/updatedatecolumn.d.ts","./node_modules/typeorm/decorator/columns/versioncolumn.d.ts","./node_modules/typeorm/decorator/options/virtualcolumnoptions.d.ts","./node_modules/typeorm/decorator/columns/virtualcolumn.d.ts","./node_modules/typeorm/decorator/options/viewcolumnoptions.d.ts","./node_modules/typeorm/decorator/columns/viewcolumn.d.ts","./node_modules/typeorm/decorator/columns/objectidcolumn.d.ts","./node_modules/typeorm/decorator/listeners/afterinsert.d.ts","./node_modules/typeorm/decorator/listeners/afterload.d.ts","./node_modules/typeorm/decorator/listeners/afterremove.d.ts","./node_modules/typeorm/decorator/listeners/aftersoftremove.d.ts","./node_modules/typeorm/decorator/listeners/afterrecover.d.ts","./node_modules/typeorm/decorator/listeners/afterupdate.d.ts","./node_modules/typeorm/decorator/listeners/beforeinsert.d.ts","./node_modules/typeorm/decorator/listeners/beforeremove.d.ts","./node_modules/typeorm/decorator/listeners/beforesoftremove.d.ts","./node_modules/typeorm/decorator/listeners/beforerecover.d.ts","./node_modules/typeorm/decorator/listeners/beforeupdate.d.ts","./node_modules/typeorm/decorator/listeners/eventsubscriber.d.ts","./node_modules/typeorm/decorator/options/indexoptions.d.ts","./node_modules/typeorm/decorator/options/entityoptions.d.ts","./node_modules/typeorm/decorator/relations/joincolumn.d.ts","./node_modules/typeorm/decorator/relations/jointable.d.ts","./node_modules/typeorm/decorator/relations/manytomany.d.ts","./node_modules/typeorm/decorator/relations/manytoone.d.ts","./node_modules/typeorm/decorator/relations/onetomany.d.ts","./node_modules/typeorm/decorator/relations/onetoone.d.ts","./node_modules/typeorm/decorator/relations/relationcount.d.ts","./node_modules/typeorm/decorator/relations/relationid.d.ts","./node_modules/typeorm/decorator/entity/entity.d.ts","./node_modules/typeorm/decorator/entity/childentity.d.ts","./node_modules/typeorm/decorator/entity/tableinheritance.d.ts","./node_modules/typeorm/decorator/options/viewentityoptions.d.ts","./node_modules/typeorm/decorator/entity-view/viewentity.d.ts","./node_modules/typeorm/decorator/tree/treelevelcolumn.d.ts","./node_modules/typeorm/decorator/tree/treeparent.d.ts","./node_modules/typeorm/decorator/tree/treechildren.d.ts","./node_modules/typeorm/decorator/tree/tree.d.ts","./node_modules/typeorm/decorator/index.d.ts","./node_modules/typeorm/decorator/foreignkey.d.ts","./node_modules/typeorm/decorator/options/uniqueoptions.d.ts","./node_modules/typeorm/decorator/unique.d.ts","./node_modules/typeorm/decorator/check.d.ts","./node_modules/typeorm/decorator/exclusion.d.ts","./node_modules/typeorm/decorator/generated.d.ts","./node_modules/typeorm/decorator/entityrepository.d.ts","./node_modules/typeorm/find-options/operator/and.d.ts","./node_modules/typeorm/find-options/operator/or.d.ts","./node_modules/typeorm/find-options/operator/any.d.ts","./node_modules/typeorm/find-options/operator/arraycontainedby.d.ts","./node_modules/typeorm/find-options/operator/arraycontains.d.ts","./node_modules/typeorm/find-options/operator/arrayoverlap.d.ts","./node_modules/typeorm/find-options/operator/between.d.ts","./node_modules/typeorm/find-options/operator/equal.d.ts","./node_modules/typeorm/find-options/operator/in.d.ts","./node_modules/typeorm/find-options/operator/isnull.d.ts","./node_modules/typeorm/find-options/operator/lessthan.d.ts","./node_modules/typeorm/find-options/operator/lessthanorequal.d.ts","./node_modules/typeorm/find-options/operator/ilike.d.ts","./node_modules/typeorm/find-options/operator/like.d.ts","./node_modules/typeorm/find-options/operator/morethan.d.ts","./node_modules/typeorm/find-options/operator/morethanorequal.d.ts","./node_modules/typeorm/find-options/operator/not.d.ts","./node_modules/typeorm/find-options/operator/raw.d.ts","./node_modules/typeorm/find-options/operator/jsoncontains.d.ts","./node_modules/typeorm/find-options/findoptionsutils.d.ts","./node_modules/typeorm/logger/abstractlogger.d.ts","./node_modules/typeorm/logger/advancedconsolelogger.d.ts","./node_modules/typeorm/logger/formattedconsolelogger.d.ts","./node_modules/typeorm/logger/simpleconsolelogger.d.ts","./node_modules/typeorm/logger/filelogger.d.ts","./node_modules/typeorm/repository/abstractrepository.d.ts","./node_modules/typeorm/data-source/index.d.ts","./node_modules/typeorm/repository/baseentity.d.ts","./node_modules/typeorm/driver/sqlserver/mssqlparameter.d.ts","./node_modules/typeorm/connection/connectionoptionsreader.d.ts","./node_modules/typeorm/connection/connectionoptions.d.ts","./node_modules/typeorm/connection/connection.d.ts","./node_modules/typeorm/migration/migrationexecutor.d.ts","./node_modules/typeorm/naming-strategy/defaultnamingstrategy.d.ts","./node_modules/typeorm/naming-strategy/legacyoraclenamingstrategy.d.ts","./node_modules/typeorm/entity-schema/entityschemaembeddedcolumnoptions.d.ts","./node_modules/typeorm/schema-builder/rdbmsschemabuilder.d.ts","./node_modules/typeorm/util/instancechecker.d.ts","./node_modules/typeorm/repository/findtreesoptions.d.ts","./node_modules/typeorm/util/treerepositoryutils.d.ts","./node_modules/typeorm/index.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/entity-class-or-schema.type.d.ts","./node_modules/@nestjs/typeorm/dist/common/typeorm.decorators.d.ts","./node_modules/@nestjs/typeorm/dist/common/typeorm.utils.d.ts","./node_modules/@nestjs/typeorm/dist/common/index.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/typeorm-options.interface.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/index.d.ts","./node_modules/@nestjs/typeorm/dist/typeorm.module.d.ts","./node_modules/@nestjs/typeorm/dist/index.d.ts","./node_modules/@nestjs/typeorm/index.d.ts","./node_modules/keyv/dist/index.d.cts","./node_modules/cache-manager/dist/index.d.cts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/command-options.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/lua-script.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_cat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_deluser.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_dryrun.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_genpass.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_getuser.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_load.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_log_reset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_log.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_save.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_setuser.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_users.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/acl_whoami.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/asking.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/auth.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bgrewriteaof.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bgsave.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_caching.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_getname.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_getredir.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_id.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_kill.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_no-evict.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_no-touch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_pause.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_setname.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_tracking.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_trackinginfo.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/client_unpause.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_addslots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/generic-transformers.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_addslotsrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_bumpepoch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_count-failure-reports.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_countkeysinslot.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_delslots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_delslotsrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_failover.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_flushslots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_forget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_getkeysinslot.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_keyslot.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_links.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_meet.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_myid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_myshardid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_nodes.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_replicas.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_replicate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_reset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_saveconfig.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_set-config-epoch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_setslot.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/cluster_slots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command_getkeys.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command_getkeysandflags.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command_info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/command.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/config_get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/config_resetstat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/config_rewrite.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/config_set.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/dbsize.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/discard.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/echo.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/failover.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/flushall.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/flushdb.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_delete.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_dump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_flush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_kill.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_list_withcode.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_load.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_restore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/function_stats.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hello.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/keys.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lastsave.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/latency_doctor.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/latency_graph.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/latency_history.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/latency_latest.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lolwut.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/memory_doctor.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/memory_malloc-stats.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/memory_purge.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/memory_stats.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/memory_usage.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/module_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/module_load.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/module_unload.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/move.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ping.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pubsub_channels.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pubsub_numpat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pubsub_numsub.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pubsub_shardchannels.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pubsub_shardnumsub.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/randomkey.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/readonly.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/readwrite.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/replicaof.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/restore-asking.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/role.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/save.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/scan.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/script_debug.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/script_exists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/script_flush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/script_kill.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/script_load.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/shutdown.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/swapdb.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/time.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/unwatch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/wait.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/append.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bitcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bitfield.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bitfield_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bitop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bitpos.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/blmove.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lmpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/blmpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/blpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/brpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/brpoplpush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zmpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bzmpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bzpopmax.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/bzpopmin.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/copy.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/decr.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/decrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/del.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/dump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/eval_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/eval.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/evalsha.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/evalsha_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/exists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/expire.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/expireat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/expiretime.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/fcall_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/fcall.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geoadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geodist.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geohash.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geopos.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadius_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadius_ro_with.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadius.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadius_with.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusbymember_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusbymember_ro_with.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusbymember.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusbymember_with.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusbymemberstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/georadiusstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geosearch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geosearch_with.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/geosearchstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/getbit.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/getdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/getex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/getrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/getset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hexists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hexpire.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hexpireat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hexpiretime.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hgetall.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hincrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hincrbyfloat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hkeys.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hmget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hpersist.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hpexpire.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hpexpireat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hpexpiretime.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hpttl.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hrandfield.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hrandfield_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hrandfield_count_withvalues.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hscan.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hscan_novalues.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hsetnx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hstrlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/httl.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/hvals.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/incr.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/incrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/incrbyfloat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lcs.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lcs_idx_withmatchlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lcs_idx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lcs_len.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lindex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/linsert.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/llen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lmove.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpop_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpos.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpos_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lpushx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lrem.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/lset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ltrim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/mget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/migrate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/mset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/msetnx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/object_encoding.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/object_freq.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/object_idletime.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/object_refcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/persist.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pexpire.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pexpireat.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pexpiretime.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pfadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pfcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pfmerge.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/psetex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/pttl.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/publish.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rename.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/renamenx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/restore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rpop_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rpoplpush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rpush.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/rpushx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/scard.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sdiff.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sdiffstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sinter.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sintercard.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sinterstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/set.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/setbit.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/setex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/setnx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/setrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sismember.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/smembers.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/smismember.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/smove.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sort_ro.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sort_store.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sort.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/spop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/spublish.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/srandmember.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/srandmember_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/srem.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sscan.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/strlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sunion.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/sunionstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/touch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ttl.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/type.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/unlink.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/watch.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xack.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xautoclaim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xautoclaim_justid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xclaim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xclaim_justid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xgroup_create.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xgroup_createconsumer.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xgroup_delconsumer.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xgroup_destroy.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xgroup_setid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xinfo_consumers.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xinfo_groups.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xinfo_stream.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xpending_range.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xpending.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xread.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xreadgroup.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xrevrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xsetid.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/xtrim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zcard.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zdiff.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zdiff_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zdiffstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zincrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zinter.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zinter_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zintercard.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zinterstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zlexcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zmscore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zpopmax.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zpopmax_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zpopmin.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zpopmin_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrandmember.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrandmember_count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrandmember_count_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrange_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrangebylex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrangebyscore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrangebyscore_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrangestore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrem.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zremrangebylex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zremrangebyrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zremrangebyscore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zrevrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zscan.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zscore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zunion.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zunion_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/zunionstore.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/commands.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/socket.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/pub-sub.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/commands-queue.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/errors.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/multi-command.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/multi-command.d.ts","./node_modules/generic-pool/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/commands.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/cluster-slots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/multi-command.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/add.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/card.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/exists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/insert.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/loadchunk.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/madd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/mexists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/reserve.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/scandump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/incrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/initbydim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/initbyprob.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/merge.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/query.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/add.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/addnx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/del.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/exists.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/insertnx.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/loadchunk.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/reserve.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/scandump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/insert.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/add.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/byrevrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/cdf.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/create.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/max.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/merge.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/min.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/quantile.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/rank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/reset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/revrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/trimmed_mean.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/byrank.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/add.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/count.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/incrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/list_withcount.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/query.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/reserve.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/config_get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/config_set.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/delete.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/explain.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/profile.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/query.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/ro_query.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/slowlog.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/graph.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrappend.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrindex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrinsert.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrpop.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/arrtrim.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/debug_memory.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/del.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/forget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/merge.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/mget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/mset.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/numincrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/nummultby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/objkeys.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/objlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/resp.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/set.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/strappend.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/strlen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/type.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/_list.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/alter.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/aggregate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/aggregate_withcursor.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/aliasadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/aliasdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/aliasupdate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/config_get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/config_set.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/create.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/cursor_del.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/cursor_read.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/dictadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/dictdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/dictdump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/dropindex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/explain.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/explaincli.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/search.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/profile_search.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/profile_aggregate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/search_nocontent.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/spellcheck.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugadd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugdel.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugget_withpayloads.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugget_withscores.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/sugget_withscores_withpayloads.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/suglen.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/syndump.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/synupdate.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/tagvals.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/add.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/alter.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/create.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/createrule.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/decrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/del.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/deleterule.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/get.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/incrby.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/info.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/info_debug.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/madd.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mget.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mget_withlabels.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/queryindex.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/range.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/revrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mrange_withlabels.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mrevrange.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/mrevrange_withlabels.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/redis/dist/index.d.ts","./node_modules/cache-manager-redis-store/dist/index.d.ts","./node_modules/@nestjs/schedule/dist/enums/cron-expression.enum.d.ts","./node_modules/@nestjs/schedule/dist/enums/index.d.ts","./node_modules/@types/luxon/src/zone.d.ts","./node_modules/@types/luxon/src/settings.d.ts","./node_modules/@types/luxon/src/_util.d.ts","./node_modules/@types/luxon/src/misc.d.ts","./node_modules/@types/luxon/src/duration.d.ts","./node_modules/@types/luxon/src/interval.d.ts","./node_modules/@types/luxon/src/datetime.d.ts","./node_modules/@types/luxon/src/info.d.ts","./node_modules/@types/luxon/src/luxon.d.ts","./node_modules/@types/luxon/index.d.ts","./node_modules/cron/dist/errors.d.ts","./node_modules/cron/dist/constants.d.ts","./node_modules/cron/dist/job.d.ts","./node_modules/cron/dist/types/utils.d.ts","./node_modules/cron/dist/types/cron.types.d.ts","./node_modules/cron/dist/time.d.ts","./node_modules/cron/dist/index.d.ts","./node_modules/@nestjs/schedule/dist/decorators/cron.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/interval.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/timeout.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/index.d.ts","./node_modules/@nestjs/schedule/dist/interfaces/schedule-module-options.interface.d.ts","./node_modules/@nestjs/schedule/dist/schedule.module.d.ts","./node_modules/@nestjs/schedule/dist/scheduler.registry.d.ts","./node_modules/@nestjs/schedule/dist/index.d.ts","./node_modules/@nestjs/schedule/index.d.ts","./node_modules/prom-client/index.d.ts","./src/monitoring/metrics/metrics-collection.service.ts","./node_modules/@types/send/index.d.ts","./node_modules/@types/qs/index.d.ts","./node_modules/@types/range-parser/index.d.ts","./node_modules/@types/express-serve-static-core/index.d.ts","./node_modules/@types/http-errors/index.d.ts","./node_modules/@types/serve-static/index.d.ts","./node_modules/@types/connect/index.d.ts","./node_modules/@types/body-parser/index.d.ts","./node_modules/@types/express/node_modules/@types/express-serve-static-core/index.d.ts","./node_modules/@types/express/index.d.ts","./src/monitoring/alerting/alerting.service.ts","./src/monitoring/scheduled-task-monitoring.service.ts","./src/monitoring/monitoring.controller.ts","./src/monitoring/performance/performance-analysis.service.ts","./src/monitoring/optimization/optimization.service.ts","./src/monitoring/monitoring.service.ts","./src/monitoring/monitoring.module.ts","./src/common/interceptors/monitoring.interceptor.ts","./src/monitoring/logging/typeorm-logger.ts","./src/common/database/transaction.service.ts","./src/common/database/transactional.decorator.ts","./src/common/database/transactional.interceptor.ts","./src/common/database/database.module.ts","./node_modules/@nestjs/bull-shared/dist/bull.messages.d.ts","./node_modules/@nestjs/bull-shared/dist/bull.tokens.d.ts","./node_modules/@nestjs/bull-shared/dist/errors/missing-shared-bull-config.error.d.ts","./node_modules/@nestjs/bull-shared/dist/errors/index.d.ts","./node_modules/@nestjs/bull-shared/dist/helpers/create-conditional-dep-holder.helper.d.ts","./node_modules/@nestjs/bull-shared/dist/helpers/index.d.ts","./node_modules/@nestjs/bull-shared/dist/utils/get-queue-token.util.d.ts","./node_modules/@nestjs/bull-shared/dist/utils/index.d.ts","./node_modules/@nestjs/bull-shared/dist/index.d.ts","./node_modules/ioredis/built/types.d.ts","./node_modules/ioredis/built/command.d.ts","./node_modules/ioredis/built/scanstream.d.ts","./node_modules/ioredis/built/utils/rediscommander.d.ts","./node_modules/ioredis/built/transaction.d.ts","./node_modules/ioredis/built/utils/commander.d.ts","./node_modules/ioredis/built/connectors/abstractconnector.d.ts","./node_modules/ioredis/built/connectors/connectorconstructor.d.ts","./node_modules/ioredis/built/connectors/sentinelconnector/types.d.ts","./node_modules/ioredis/built/connectors/sentinelconnector/sentineliterator.d.ts","./node_modules/ioredis/built/connectors/sentinelconnector/index.d.ts","./node_modules/ioredis/built/connectors/standaloneconnector.d.ts","./node_modules/ioredis/built/redis/redisoptions.d.ts","./node_modules/ioredis/built/cluster/util.d.ts","./node_modules/ioredis/built/cluster/clusteroptions.d.ts","./node_modules/ioredis/built/cluster/index.d.ts","./node_modules/denque/index.d.ts","./node_modules/ioredis/built/subscriptionset.d.ts","./node_modules/ioredis/built/datahandler.d.ts","./node_modules/ioredis/built/redis.d.ts","./node_modules/ioredis/built/pipeline.d.ts","./node_modules/ioredis/built/index.d.ts","./node_modules/bull/index.d.ts","./node_modules/@nestjs/bull/dist/interfaces/bull.interfaces.d.ts","./node_modules/@nestjs/bull/dist/bull.types.d.ts","./node_modules/@nestjs/bull/dist/interfaces/bull-module-options.interface.d.ts","./node_modules/@nestjs/bull/dist/interfaces/shared-bull-config.interface.d.ts","./node_modules/@nestjs/bull/dist/interfaces/index.d.ts","./node_modules/@nestjs/bull/dist/bull.module.d.ts","./node_modules/@nestjs/bull/dist/decorators/inject-queue.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/process.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/processor.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/queue-hooks.decorators.d.ts","./node_modules/@nestjs/bull/dist/decorators/index.d.ts","./node_modules/@nestjs/bull/dist/enums/bull-queue-events.enum.d.ts","./node_modules/@nestjs/bull/dist/enums/bull-queue-global-events.enum.d.ts","./node_modules/@nestjs/bull/dist/enums/index.d.ts","./node_modules/@nestjs/bull/dist/utils/get-queue-options-token.util.d.ts","./node_modules/@nestjs/bull/dist/utils/get-shared-config-token.util.d.ts","./node_modules/@nestjs/bull/dist/utils/index.d.ts","./node_modules/@nestjs/bull/dist/index.d.ts","./node_modules/eventemitter2/eventemitter2.d.ts","./node_modules/@nestjs/event-emitter/dist/constants.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/event-emitter-options.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/on-event-options.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/event-payload-host.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/index.d.ts","./node_modules/@nestjs/event-emitter/dist/decorators/on-event.decorator.d.ts","./node_modules/@nestjs/event-emitter/dist/decorators/index.d.ts","./node_modules/@nestjs/event-emitter/dist/event-emitter-readiness.watcher.d.ts","./node_modules/@nestjs/event-emitter/dist/event-emitter.module.d.ts","./node_modules/@nestjs/event-emitter/dist/index.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.constants.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/cache-manager.interface.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/cache-module.interface.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.module-definition.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.module.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/cache-key.decorator.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/cache-ttl.decorator.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/index.d.ts","./node_modules/@nestjs/cache-manager/dist/interceptors/cache.interceptor.d.ts","./node_modules/@nestjs/cache-manager/dist/interceptors/index.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/index.d.ts","./node_modules/@nestjs/cache-manager/dist/index.d.ts","./node_modules/@nestjs/cache-manager/index.d.ts","./node_modules/joi/lib/index.d.ts","./src/config/env.validation.ts","./node_modules/@nestjs/passport/dist/abstract.strategy.d.ts","./node_modules/@nestjs/passport/dist/interfaces/auth-module.options.d.ts","./node_modules/@nestjs/passport/dist/interfaces/type.interface.d.ts","./node_modules/@nestjs/passport/dist/interfaces/index.d.ts","./node_modules/@nestjs/passport/dist/auth.guard.d.ts","./node_modules/@nestjs/passport/dist/passport.module.d.ts","./node_modules/@types/passport/index.d.ts","./node_modules/@nestjs/passport/dist/passport/passport.serializer.d.ts","./node_modules/@nestjs/passport/dist/passport/passport.strategy.d.ts","./node_modules/@nestjs/passport/dist/index.d.ts","./node_modules/@nestjs/passport/index.d.ts","./src/common/constants/auth.constants.ts","./src/auth/guards/jwt-auth.guard.ts","./node_modules/@nestjs/throttler/dist/throttler-storage-record.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler-storage.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.guard.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler-module-options.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.decorator.d.ts","./node_modules/@nestjs/throttler/dist/throttler.exception.d.ts","./node_modules/@nestjs/throttler/dist/throttler.guard.d.ts","./node_modules/@nestjs/throttler/dist/throttler.module.d.ts","./node_modules/@nestjs/throttler/dist/throttler.providers.d.ts","./node_modules/@nestjs/throttler/dist/throttler-storage-options.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.service.d.ts","./node_modules/@nestjs/throttler/dist/utilities.d.ts","./node_modules/@nestjs/throttler/dist/index.d.ts","./node_modules/axios/index.d.cts","./src/health/health.service.ts","./src/common/services/shutdown-state.service.ts","./src/health/health.controller.ts","./src/health/health.module.ts","./src/session/session.constants.ts","./src/session/session.service.ts","./src/session/session.module.ts","./src/common/utils/bull-redis.util.ts","./src/common/guards/throttle.guard.ts","./src/config/feature-flags.config.ts","./src/common/lazy-loading/startup-logger.service.ts","./src/common/timeout/timeout-config.service.ts","./src/common/interceptors/timeout.interceptor.ts","./src/common/interceptors/api-version.interceptor.ts","./src/common/modules/api-versioning.module.ts","./src/common/constants/queue.constants.ts","./src/common/constants/event.constants.ts","./src/sync/conflicts/conflict-resolution.service.ts","./src/sync/consistency/data-consistency.service.ts","./src/sync/cache/cache-invalidation.service.ts","./src/sync/replication/replication.service.ts","./src/sync/sync.service.ts","./src/sync/sync.module.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-body-parser-options.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-body-parser.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/serve-static-options.interface.d.ts","./node_modules/@nestjs/platform-express/adapters/express-adapter.d.ts","./node_modules/@nestjs/platform-express/adapters/index.d.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-application.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/index.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/multer-options.interface.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/any-files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/file-fields.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/file.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/no-files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/index.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/files-upload-module.interface.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/index.d.ts","./node_modules/@nestjs/platform-express/multer/multer.module.d.ts","./node_modules/@nestjs/platform-express/multer/index.d.ts","./node_modules/@nestjs/platform-express/index.d.ts","./src/cdn/entities/content-metadata.entity.ts","./node_modules/@smithy/types/dist-types/abort-handler.d.ts","./node_modules/@smithy/types/dist-types/abort.d.ts","./node_modules/@smithy/types/dist-types/auth/auth.d.ts","./node_modules/@smithy/types/dist-types/auth/httpapikeyauth.d.ts","./node_modules/@smithy/types/dist-types/identity/identity.d.ts","./node_modules/@smithy/types/dist-types/response.d.ts","./node_modules/@smithy/types/dist-types/command.d.ts","./node_modules/@smithy/types/dist-types/endpoint.d.ts","./node_modules/@smithy/types/dist-types/feature-ids.d.ts","./node_modules/@smithy/types/dist-types/logger.d.ts","./node_modules/@smithy/types/dist-types/uri.d.ts","./node_modules/@smithy/types/dist-types/http.d.ts","./node_modules/@smithy/types/dist-types/util.d.ts","./node_modules/@smithy/types/dist-types/middleware.d.ts","./node_modules/@smithy/types/dist-types/auth/httpsigner.d.ts","./node_modules/@smithy/types/dist-types/auth/identityproviderconfig.d.ts","./node_modules/@smithy/types/dist-types/auth/httpauthscheme.d.ts","./node_modules/@smithy/types/dist-types/auth/httpauthschemeprovider.d.ts","./node_modules/@smithy/types/dist-types/auth/index.d.ts","./node_modules/@smithy/types/dist-types/transform/exact.d.ts","./node_modules/@smithy/types/dist-types/externals-check/browser-externals-check.d.ts","./node_modules/@smithy/types/dist-types/blob/blob-payload-input-types.d.ts","./node_modules/@smithy/types/dist-types/crypto.d.ts","./node_modules/@smithy/types/dist-types/checksum.d.ts","./node_modules/@smithy/types/dist-types/client.d.ts","./node_modules/@smithy/types/dist-types/connection/config.d.ts","./node_modules/@smithy/types/dist-types/transfer.d.ts","./node_modules/@smithy/types/dist-types/connection/manager.d.ts","./node_modules/@smithy/types/dist-types/connection/pool.d.ts","./node_modules/@smithy/types/dist-types/connection/index.d.ts","./node_modules/@smithy/types/dist-types/eventstream.d.ts","./node_modules/@smithy/types/dist-types/encode.d.ts","./node_modules/@smithy/types/dist-types/endpoints/shared.d.ts","./node_modules/@smithy/types/dist-types/endpoints/endpointruleobject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/errorruleobject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/treeruleobject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/rulesetobject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/index.d.ts","./node_modules/@smithy/types/dist-types/extensions/checksum.d.ts","./node_modules/@smithy/types/dist-types/extensions/defaultclientconfiguration.d.ts","./node_modules/@smithy/types/dist-types/shapes.d.ts","./node_modules/@smithy/types/dist-types/retry.d.ts","./node_modules/@smithy/types/dist-types/extensions/retry.d.ts","./node_modules/@smithy/types/dist-types/extensions/defaultextensionconfiguration.d.ts","./node_modules/@smithy/types/dist-types/extensions/index.d.ts","./node_modules/@smithy/types/dist-types/http/httphandlerinitialization.d.ts","./node_modules/@smithy/types/dist-types/identity/apikeyidentity.d.ts","./node_modules/@smithy/types/dist-types/identity/awscredentialidentity.d.ts","./node_modules/@smithy/types/dist-types/identity/tokenidentity.d.ts","./node_modules/@smithy/types/dist-types/identity/index.d.ts","./node_modules/@smithy/types/dist-types/pagination.d.ts","./node_modules/@smithy/types/dist-types/profile.d.ts","./node_modules/@smithy/types/dist-types/serde.d.ts","./node_modules/@smithy/types/dist-types/schema/sentinels.d.ts","./node_modules/@smithy/types/dist-types/schema/static-schemas.d.ts","./node_modules/@smithy/types/dist-types/schema/traits.d.ts","./node_modules/@smithy/types/dist-types/schema/schema.d.ts","./node_modules/@smithy/types/dist-types/schema/schema-deprecated.d.ts","./node_modules/@smithy/types/dist-types/signature.d.ts","./node_modules/@smithy/types/dist-types/stream.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-common-types.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-payload-input-types.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-payload-output-types.d.ts","./node_modules/@smithy/types/dist-types/transform/type-transform.d.ts","./node_modules/@smithy/types/dist-types/transform/client-method-transforms.d.ts","./node_modules/@smithy/types/dist-types/transform/client-payload-blob-type-narrow.d.ts","./node_modules/@smithy/types/dist-types/transform/mutable.d.ts","./node_modules/@smithy/types/dist-types/transform/no-undefined.d.ts","./node_modules/@smithy/types/dist-types/waiter.d.ts","./node_modules/@smithy/types/dist-types/index.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromenv.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/gethomedir.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getprofilename.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getssotokenfilepath.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getssotokenfromfile.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/constants.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/loadsharedconfigfiles.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/loadssosessiondata.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/parseknownfiles.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/externaldatainterceptor.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/types.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/readfile.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/index.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromsharedconfigfiles.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromstatic.d.ts","./node_modules/@smithy/node-config-provider/dist-types/configloader.d.ts","./node_modules/@smithy/node-config-provider/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/constants.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/node_request_checksum_calculation_config_options.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/node_response_checksum_validation_config_options.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/resolveflexiblechecksumsconfig.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/configuration.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexiblechecksumsmiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexiblechecksumsinputmiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexiblechecksumsresponsemiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/getflexiblechecksumsplugin.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-host-header/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/check-content-length-header.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/region-redirect-middleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/region-redirect-endpoint-middleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-expires-middleware.d.ts","./node_modules/@aws-sdk/types/dist-types/abort.d.ts","./node_modules/@aws-sdk/types/dist-types/auth.d.ts","./node_modules/@aws-sdk/types/dist-types/blob/blob-types.d.ts","./node_modules/@aws-sdk/types/dist-types/checksum.d.ts","./node_modules/@aws-sdk/types/dist-types/client.d.ts","./node_modules/@aws-sdk/types/dist-types/command.d.ts","./node_modules/@aws-sdk/types/dist-types/connection.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/identity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/anonymousidentity.d.ts","./node_modules/@aws-sdk/types/dist-types/feature-ids.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/awscredentialidentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/loginidentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/tokenidentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/index.d.ts","./node_modules/@aws-sdk/types/dist-types/util.d.ts","./node_modules/@aws-sdk/types/dist-types/credentials.d.ts","./node_modules/@aws-sdk/types/dist-types/crypto.d.ts","./node_modules/@aws-sdk/types/dist-types/dns.d.ts","./node_modules/@aws-sdk/types/dist-types/encode.d.ts","./node_modules/@aws-sdk/types/dist-types/endpoint.d.ts","./node_modules/@aws-sdk/types/dist-types/eventstream.d.ts","./node_modules/@aws-sdk/types/dist-types/extensions/index.d.ts","./node_modules/@aws-sdk/types/dist-types/function.d.ts","./node_modules/@aws-sdk/types/dist-types/http.d.ts","./node_modules/@aws-sdk/types/dist-types/logger.d.ts","./node_modules/@aws-sdk/types/dist-types/middleware.d.ts","./node_modules/@aws-sdk/types/dist-types/pagination.d.ts","./node_modules/@aws-sdk/types/dist-types/profile.d.ts","./node_modules/@aws-sdk/types/dist-types/request.d.ts","./node_modules/@aws-sdk/types/dist-types/response.d.ts","./node_modules/@aws-sdk/types/dist-types/retry.d.ts","./node_modules/@aws-sdk/types/dist-types/serde.d.ts","./node_modules/@aws-sdk/types/dist-types/shapes.d.ts","./node_modules/@aws-sdk/types/dist-types/signature.d.ts","./node_modules/@aws-sdk/types/dist-types/stream.d.ts","./node_modules/@aws-sdk/types/dist-types/token.d.ts","./node_modules/@aws-sdk/types/dist-types/transfer.d.ts","./node_modules/@aws-sdk/types/dist-types/uri.d.ts","./node_modules/@aws-sdk/types/dist-types/waiter.d.ts","./node_modules/@aws-sdk/types/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/interfaces/s3expressidentity.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/s3expressidentitycacheentry.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/s3expressidentitycache.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/interfaces/s3expressidentityprovider.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/s3expressidentityproviderimpl.d.ts","./node_modules/@smithy/signature-v4/dist-types/signaturev4base.d.ts","./node_modules/@smithy/signature-v4/dist-types/signaturev4.d.ts","./node_modules/@smithy/signature-v4/dist-types/constants.d.ts","./node_modules/@smithy/signature-v4/dist-types/getcanonicalheaders.d.ts","./node_modules/@smithy/signature-v4/dist-types/getcanonicalquery.d.ts","./node_modules/@smithy/signature-v4/dist-types/getpayloadhash.d.ts","./node_modules/@smithy/signature-v4/dist-types/moveheaderstoquery.d.ts","./node_modules/@smithy/signature-v4/dist-types/preparerequest.d.ts","./node_modules/@smithy/signature-v4/dist-types/credentialderivation.d.ts","./node_modules/@smithy/signature-v4/dist-types/headerutil.d.ts","./node_modules/@smithy/signature-v4/dist-types/signature-v4a-container.d.ts","./node_modules/@smithy/signature-v4/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/signaturev4s3express.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/constants.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/functions/s3expressmiddleware.d.ts","./node_modules/@smithy/protocol-http/dist-types/httprequest.d.ts","./node_modules/@smithy/protocol-http/dist-types/httpresponse.d.ts","./node_modules/@smithy/protocol-http/dist-types/httphandler.d.ts","./node_modules/@smithy/protocol-http/dist-types/extensions/httpextensionconfiguration.d.ts","./node_modules/@smithy/protocol-http/dist-types/extensions/index.d.ts","./node_modules/@smithy/protocol-http/dist-types/field.d.ts","./node_modules/@smithy/protocol-http/dist-types/fields.d.ts","./node_modules/@smithy/protocol-http/dist-types/isvalidhostname.d.ts","./node_modules/@smithy/protocol-http/dist-types/types.d.ts","./node_modules/@smithy/protocol-http/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/functions/s3expresshttpsigningmiddleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3configuration.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/throw-200-exceptions.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/validate-bucket-name.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/cbor.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/cbor-types.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/parsecborbody.d.ts","./node_modules/@smithy/util-stream/dist-types/blob/uint8arrayblobadapter.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/checksumstream.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/checksumstream.browser.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/createchecksumstream.browser.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/createchecksumstream.d.ts","./node_modules/@smithy/util-stream/dist-types/createbufferedreadable.d.ts","./node_modules/@smithy/util-stream/dist-types/getawschunkedencodingstream.d.ts","./node_modules/@smithy/util-stream/dist-types/headstream.d.ts","./node_modules/@smithy/util-stream/dist-types/sdk-stream-mixin.d.ts","./node_modules/@smithy/util-stream/dist-types/splitstream.d.ts","./node_modules/@smithy/util-stream/dist-types/stream-type-check.d.ts","./node_modules/@smithy/util-stream/dist-types/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/collect-stream-body.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/extended-encode-uri-component.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/deref.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/middleware/schema-middleware-types.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/middleware/getschemaserdeplugin.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/schema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/listschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/mapschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/operationschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/operation.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/structureschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/errorschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/normalizedschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/simpleschema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/sentinels.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/translatetraits.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/typeregistry.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/event-streams/eventstreamserde.d.ts","./node_modules/@smithy/core/dist-types/submodules/event-streams/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serdecontext.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/httpprotocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/httpbindingprotocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/rpcprotocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/requestbuilder.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/resolve-path.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/fromstringshapedeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/httpinterceptingshapedeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/tostringshapeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/httpinterceptingshapeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/determinetimestampformat.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/cborcodec.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/smithyrpcv2cborprotocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/cbor/awssmithyrpcv2cborprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/coercing-serializers.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/configurableserdecontext.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/jsonshapedeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/jsonshapeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/jsoncodec.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsjsonrpcprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsjson1_0protocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsjson1_1protocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsrestjsonprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsexpectunion.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/parsejsonbody.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/xmlshapeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/xmlcodec.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/xmlshapedeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/queryserializersettings.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/queryshapeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/awsqueryprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/awsec2queryprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/awsrestxmlprotocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/parsexmlbody.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/protocol/s3restxmlprotocol.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/configurations.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/user-agent-middleware.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsconfig/nodeusedualstackendpointconfigoptions.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsconfig/nodeusefipsendpointconfigoptions.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsconfig/resolveendpointsconfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsconfig/resolvecustomendpointsconfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsconfig/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionconfig/config.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionconfig/resolveregionconfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionconfig/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/endpointvarianttag.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/endpointvariant.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/partitionhash.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/regionhash.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/getregioninfo.d.ts","./node_modules/@smithy/config-resolver/dist-types/regioninfo/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/index.d.ts","./node_modules/@smithy/eventstream-serde-config-resolver/dist-types/eventstreamserdeconfig.d.ts","./node_modules/@smithy/eventstream-serde-config-resolver/dist-types/index.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/resolveendpointconfig.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/types.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/getendpointfrominstructions.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/toendpointv1.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/index.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/endpointmiddleware.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/getendpointplugin.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/resolveendpointrequiredconfig.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/index.d.ts","./node_modules/@smithy/util-retry/dist-types/standardretrystrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/types.d.ts","./node_modules/@smithy/util-retry/dist-types/adaptiveretrystrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/configuredretrystrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/defaultratelimiter.d.ts","./node_modules/@smithy/util-retry/dist-types/config.d.ts","./node_modules/@smithy/util-retry/dist-types/constants.d.ts","./node_modules/@smithy/util-retry/dist-types/retries-2026-config.d.ts","./node_modules/@smithy/util-retry/dist-types/index.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/types.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/standardretrystrategy.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/adaptiveretrystrategy.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/delaydecider.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/retrydecider.d.ts","./node_modules/@smithy/middleware-retry/dist-types/configurations.d.ts","./node_modules/@smithy/middleware-retry/dist-types/omitretryheadersmiddleware.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retrymiddleware.d.ts","./node_modules/@smithy/middleware-retry/dist-types/parseretryafterheader.d.ts","./node_modules/@smithy/middleware-retry/dist-types/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/client.d.ts","./node_modules/@smithy/smithy-client/dist-types/collect-stream-body.d.ts","./node_modules/@smithy/smithy-client/dist-types/command.d.ts","./node_modules/@smithy/smithy-client/dist-types/constants.d.ts","./node_modules/@smithy/smithy-client/dist-types/create-aggregated-client.d.ts","./node_modules/@smithy/smithy-client/dist-types/default-error-handler.d.ts","./node_modules/@smithy/smithy-client/dist-types/defaults-mode.d.ts","./node_modules/@smithy/smithy-client/dist-types/emitwarningifunsupportedversion.d.ts","./node_modules/@smithy/smithy-client/dist-types/exceptions.d.ts","./node_modules/@smithy/smithy-client/dist-types/extended-encode-uri-component.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/checksum.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/retry.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/defaultextensionconfiguration.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/get-array-if-single-item.d.ts","./node_modules/@smithy/smithy-client/dist-types/get-value-from-text-node.d.ts","./node_modules/@smithy/smithy-client/dist-types/is-serializable-header-value.d.ts","./node_modules/@smithy/smithy-client/dist-types/nooplogger.d.ts","./node_modules/@smithy/smithy-client/dist-types/object-mapping.d.ts","./node_modules/@smithy/smithy-client/dist-types/resolve-path.d.ts","./node_modules/@smithy/smithy-client/dist-types/ser-utils.d.ts","./node_modules/@smithy/smithy-client/dist-types/serde-json.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/copydocumentwithtransform.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/date-utils.d.ts","./node_modules/@smithy/uuid/dist-types/v4.d.ts","./node_modules/@smithy/uuid/dist-types/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/generateidempotencytoken.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/lazy-json.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/parse-utils.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/quote-header.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/schema-serde-lib/schema-date-utils.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/split-every.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/split-header.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/value/numericvalue.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/resolveawssdksigv4aconfig.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/awssdksigv4signer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/awssdksigv4asigner.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/node_auth_scheme_preference_options.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/resolveawssdksigv4config.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/aws_sdk/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/utils/getbearertokenenvkey.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpauthschemes/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/endpoint/endpointparameters.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/auth/httpauthschemeprovider.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/abortmultipartuploadcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/completemultipartuploadcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/copyobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/createbucketcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/createbucketmetadataconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/createbucketmetadatatableconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/createmultipartuploadcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/createsessioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketanalyticsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketcorscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketencryptioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketintelligenttieringconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketinventoryconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketlifecyclecommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketmetadataconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketmetadatatableconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketmetricsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketownershipcontrolscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketpolicycommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketreplicationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebuckettaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletebucketwebsitecommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deleteobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deleteobjectscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deleteobjecttaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/deletepublicaccessblockcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketabaccommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketaccelerateconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketaclcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketanalyticsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketcorscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketencryptioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketintelligenttieringconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketinventoryconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketlifecycleconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketlocationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketloggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketmetadataconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketmetadatatableconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketmetricsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketnotificationconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketownershipcontrolscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketpolicycommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketpolicystatuscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketreplicationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketrequestpaymentcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbuckettaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketversioningcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getbucketwebsitecommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectaclcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectattributescommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectlegalholdcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectlockconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjectretentioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjecttaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getobjecttorrentcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/getpublicaccessblockcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/headbucketcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/headobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listbucketanalyticsconfigurationscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listbucketintelligenttieringconfigurationscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listbucketinventoryconfigurationscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listbucketmetricsconfigurationscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listbucketscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listdirectorybucketscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listmultipartuploadscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listobjectscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listobjectsv2command.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listobjectversionscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/listpartscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketabaccommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketaccelerateconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketaclcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketanalyticsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketcorscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketencryptioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketintelligenttieringconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketinventoryconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketlifecycleconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketloggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketmetricsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketnotificationconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketownershipcontrolscommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketpolicycommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketreplicationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketrequestpaymentcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbuckettaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketversioningcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putbucketwebsitecommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjectaclcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjectlegalholdcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjectlockconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjectretentioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putobjecttaggingcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/putpublicaccessblockcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/renameobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/models_1.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/restoreobjectcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/selectobjectcontentcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/updatebucketmetadatainventorytableconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/updatebucketmetadatajournaltableconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/updateobjectencryptioncommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/uploadpartcommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/uploadpartcopycommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/writegetobjectresponsecommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/auth/httpauthextensionconfiguration.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/extensionconfiguration.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/runtimeextensions.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/s3client.d.ts","./node_modules/@smithy/util-waiter/dist-types/waiter.d.ts","./node_modules/@smithy/util-waiter/dist-types/createwaiter.d.ts","./node_modules/@smithy/util-waiter/dist-types/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/s3.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/interfaces.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/listbucketspaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/listdirectorybucketspaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/listobjectsv2paginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/listpartspaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitforbucketexists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitforbucketnotexists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitforobjectexists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitforobjectnotexists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/s3serviceexception.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/index.d.ts","./src/common/types/file.types.ts","./src/media/storage/file-storage.service.ts","./src/media/processing/video-processing.service.ts","./node_modules/sharp/lib/index.d.ts","./src/media/validation/file-validation.constants.ts","./src/media/validation/file-validation.service.ts","./src/media/validation/malware-scanning.service.ts","./src/media/processing/image-processing.service.ts","./src/media/validation/upload-progress.service.ts","./node_modules/uuid/dist/max.d.ts","./node_modules/uuid/dist/nil.d.ts","./node_modules/uuid/dist/types.d.ts","./node_modules/uuid/dist/parse.d.ts","./node_modules/uuid/dist/stringify.d.ts","./node_modules/uuid/dist/v1.d.ts","./node_modules/uuid/dist/v1tov6.d.ts","./node_modules/uuid/dist/v35.d.ts","./node_modules/uuid/dist/v3.d.ts","./node_modules/uuid/dist/v4.d.ts","./node_modules/uuid/dist/v5.d.ts","./node_modules/uuid/dist/v6.d.ts","./node_modules/uuid/dist/v6tov1.d.ts","./node_modules/uuid/dist/v7.d.ts","./node_modules/uuid/dist/validate.d.ts","./node_modules/uuid/dist/version.d.ts","./node_modules/uuid/dist/index.d.ts","./src/media/media.service.ts","./src/media/validation/upload-validation.util.ts","./src/media/media.controller.ts","./src/media/processing/document-processing.service.ts","./node_modules/@types/fluent-ffmpeg/index.d.ts","./src/media/processing/video.processor.ts","./src/media/media.module.ts","./src/backup/enums/backup-status.enum.ts","./src/backup/enums/backup-type.enum.ts","./src/backup/enums/region.enum.ts","./src/backup/entities/backup-record.entity.ts","./src/backup/enums/recovery-test-status.enum.ts","./src/backup/entities/recovery-test.entity.ts","./node_modules/class-validator/types/validation/validationerror.d.ts","./node_modules/class-validator/types/validation/validatoroptions.d.ts","./node_modules/class-validator/types/validation-schema/validationschema.d.ts","./node_modules/class-validator/types/container.d.ts","./node_modules/class-validator/types/validation/validationarguments.d.ts","./node_modules/class-validator/types/decorator/validationoptions.d.ts","./node_modules/class-validator/types/decorator/common/allow.d.ts","./node_modules/class-validator/types/decorator/common/isdefined.d.ts","./node_modules/class-validator/types/decorator/common/isoptional.d.ts","./node_modules/class-validator/types/decorator/common/validate.d.ts","./node_modules/class-validator/types/validation/validatorconstraintinterface.d.ts","./node_modules/class-validator/types/decorator/common/validateby.d.ts","./node_modules/class-validator/types/decorator/common/validateif.d.ts","./node_modules/class-validator/types/decorator/common/validatenested.d.ts","./node_modules/class-validator/types/decorator/common/validatepromise.d.ts","./node_modules/class-validator/types/decorator/common/islatlong.d.ts","./node_modules/class-validator/types/decorator/common/islatitude.d.ts","./node_modules/class-validator/types/decorator/common/islongitude.d.ts","./node_modules/class-validator/types/decorator/common/equals.d.ts","./node_modules/class-validator/types/decorator/common/notequals.d.ts","./node_modules/class-validator/types/decorator/common/isempty.d.ts","./node_modules/class-validator/types/decorator/common/isnotempty.d.ts","./node_modules/class-validator/types/decorator/common/isin.d.ts","./node_modules/class-validator/types/decorator/common/isnotin.d.ts","./node_modules/class-validator/types/decorator/number/isdivisibleby.d.ts","./node_modules/class-validator/types/decorator/number/ispositive.d.ts","./node_modules/class-validator/types/decorator/number/isnegative.d.ts","./node_modules/class-validator/types/decorator/number/max.d.ts","./node_modules/class-validator/types/decorator/number/min.d.ts","./node_modules/class-validator/types/decorator/date/mindate.d.ts","./node_modules/class-validator/types/decorator/date/maxdate.d.ts","./node_modules/class-validator/types/decorator/string/contains.d.ts","./node_modules/class-validator/types/decorator/string/notcontains.d.ts","./node_modules/@types/validator/lib/isboolean.d.ts","./node_modules/@types/validator/lib/isemail.d.ts","./node_modules/@types/validator/lib/isfqdn.d.ts","./node_modules/@types/validator/lib/isiban.d.ts","./node_modules/@types/validator/lib/isiso31661alpha2.d.ts","./node_modules/@types/validator/lib/isiso4217.d.ts","./node_modules/@types/validator/lib/isiso6391.d.ts","./node_modules/@types/validator/lib/istaxid.d.ts","./node_modules/@types/validator/lib/isurl.d.ts","./node_modules/@types/validator/index.d.ts","./node_modules/class-validator/types/decorator/string/isalpha.d.ts","./node_modules/class-validator/types/decorator/string/isalphanumeric.d.ts","./node_modules/class-validator/types/decorator/string/isdecimal.d.ts","./node_modules/class-validator/types/decorator/string/isascii.d.ts","./node_modules/class-validator/types/decorator/string/isbase64.d.ts","./node_modules/class-validator/types/decorator/string/isbytelength.d.ts","./node_modules/class-validator/types/decorator/string/iscreditcard.d.ts","./node_modules/class-validator/types/decorator/string/iscurrency.d.ts","./node_modules/class-validator/types/decorator/string/isemail.d.ts","./node_modules/class-validator/types/decorator/string/isfqdn.d.ts","./node_modules/class-validator/types/decorator/string/isfullwidth.d.ts","./node_modules/class-validator/types/decorator/string/ishalfwidth.d.ts","./node_modules/class-validator/types/decorator/string/isvariablewidth.d.ts","./node_modules/class-validator/types/decorator/string/ishexcolor.d.ts","./node_modules/class-validator/types/decorator/string/ishexadecimal.d.ts","./node_modules/class-validator/types/decorator/string/ismacaddress.d.ts","./node_modules/class-validator/types/decorator/string/isip.d.ts","./node_modules/class-validator/types/decorator/string/isport.d.ts","./node_modules/class-validator/types/decorator/string/isisbn.d.ts","./node_modules/class-validator/types/decorator/string/isisin.d.ts","./node_modules/class-validator/types/decorator/string/isiso8601.d.ts","./node_modules/class-validator/types/decorator/string/isjson.d.ts","./node_modules/class-validator/types/decorator/string/isjwt.d.ts","./node_modules/class-validator/types/decorator/string/islowercase.d.ts","./node_modules/class-validator/types/decorator/string/ismobilephone.d.ts","./node_modules/class-validator/types/decorator/string/isiso31661alpha2.d.ts","./node_modules/class-validator/types/decorator/string/isiso31661alpha3.d.ts","./node_modules/class-validator/types/decorator/string/ismongoid.d.ts","./node_modules/class-validator/types/decorator/string/ismultibyte.d.ts","./node_modules/class-validator/types/decorator/string/issurrogatepair.d.ts","./node_modules/class-validator/types/decorator/string/isurl.d.ts","./node_modules/class-validator/types/decorator/string/isuuid.d.ts","./node_modules/class-validator/types/decorator/string/isfirebasepushid.d.ts","./node_modules/class-validator/types/decorator/string/isuppercase.d.ts","./node_modules/class-validator/types/decorator/string/length.d.ts","./node_modules/class-validator/types/decorator/string/maxlength.d.ts","./node_modules/class-validator/types/decorator/string/minlength.d.ts","./node_modules/class-validator/types/decorator/string/matches.d.ts","./node_modules/libphonenumber-js/types.d.cts","./node_modules/libphonenumber-js/max/index.d.cts","./node_modules/class-validator/types/decorator/string/isphonenumber.d.ts","./node_modules/class-validator/types/decorator/string/ismilitarytime.d.ts","./node_modules/class-validator/types/decorator/string/ishash.d.ts","./node_modules/class-validator/types/decorator/string/isissn.d.ts","./node_modules/class-validator/types/decorator/string/isdatestring.d.ts","./node_modules/class-validator/types/decorator/string/isbooleanstring.d.ts","./node_modules/class-validator/types/decorator/string/isnumberstring.d.ts","./node_modules/class-validator/types/decorator/string/isbase32.d.ts","./node_modules/class-validator/types/decorator/string/isbic.d.ts","./node_modules/class-validator/types/decorator/string/isbtcaddress.d.ts","./node_modules/class-validator/types/decorator/string/isdatauri.d.ts","./node_modules/class-validator/types/decorator/string/isean.d.ts","./node_modules/class-validator/types/decorator/string/isethereumaddress.d.ts","./node_modules/class-validator/types/decorator/string/ishsl.d.ts","./node_modules/class-validator/types/decorator/string/isiban.d.ts","./node_modules/class-validator/types/decorator/string/isidentitycard.d.ts","./node_modules/class-validator/types/decorator/string/isisrc.d.ts","./node_modules/class-validator/types/decorator/string/islocale.d.ts","./node_modules/class-validator/types/decorator/string/ismagneturi.d.ts","./node_modules/class-validator/types/decorator/string/ismimetype.d.ts","./node_modules/class-validator/types/decorator/string/isoctal.d.ts","./node_modules/class-validator/types/decorator/string/ispassportnumber.d.ts","./node_modules/class-validator/types/decorator/string/ispostalcode.d.ts","./node_modules/class-validator/types/decorator/string/isrfc3339.d.ts","./node_modules/class-validator/types/decorator/string/isrgbcolor.d.ts","./node_modules/class-validator/types/decorator/string/issemver.d.ts","./node_modules/class-validator/types/decorator/string/isstrongpassword.d.ts","./node_modules/class-validator/types/decorator/string/istimezone.d.ts","./node_modules/class-validator/types/decorator/string/isbase58.d.ts","./node_modules/class-validator/types/decorator/string/is-tax-id.d.ts","./node_modules/class-validator/types/decorator/string/is-iso4217-currency-code.d.ts","./node_modules/class-validator/types/decorator/typechecker/isboolean.d.ts","./node_modules/class-validator/types/decorator/typechecker/isdate.d.ts","./node_modules/class-validator/types/decorator/typechecker/isnumber.d.ts","./node_modules/class-validator/types/decorator/typechecker/isenum.d.ts","./node_modules/class-validator/types/decorator/typechecker/isint.d.ts","./node_modules/class-validator/types/decorator/typechecker/isstring.d.ts","./node_modules/class-validator/types/decorator/typechecker/isarray.d.ts","./node_modules/class-validator/types/decorator/typechecker/isobject.d.ts","./node_modules/class-validator/types/decorator/array/arraycontains.d.ts","./node_modules/class-validator/types/decorator/array/arraynotcontains.d.ts","./node_modules/class-validator/types/decorator/array/arraynotempty.d.ts","./node_modules/class-validator/types/decorator/array/arrayminsize.d.ts","./node_modules/class-validator/types/decorator/array/arraymaxsize.d.ts","./node_modules/class-validator/types/decorator/array/arrayunique.d.ts","./node_modules/class-validator/types/decorator/object/isnotemptyobject.d.ts","./node_modules/class-validator/types/decorator/object/isinstance.d.ts","./node_modules/class-validator/types/decorator/decorators.d.ts","./node_modules/class-validator/types/validation/validationtypes.d.ts","./node_modules/class-validator/types/validation/validator.d.ts","./node_modules/class-validator/types/register-decorator.d.ts","./node_modules/class-validator/types/metadata/validationmetadataargs.d.ts","./node_modules/class-validator/types/metadata/validationmetadata.d.ts","./node_modules/class-validator/types/metadata/constraintmetadata.d.ts","./node_modules/class-validator/types/metadata/metadatastorage.d.ts","./node_modules/class-validator/types/index.d.ts","./src/backup/dto/backup-response.dto.ts","./src/backup/interfaces/backup.interfaces.ts","./src/backup/backup.service.ts","./node_modules/@aws-sdk/client-kms/dist-types/auth/httpauthschemeprovider.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/cancelkeydeletioncommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/connectcustomkeystorecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/createaliascommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/createcustomkeystorecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/creategrantcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/createkeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/decryptcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/deletealiascommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/deletecustomkeystorecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/deleteimportedkeymaterialcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/derivesharedsecretcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/describecustomkeystorescommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/describekeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/disablekeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/disablekeyrotationcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/disconnectcustomkeystorecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/enablekeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/enablekeyrotationcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/encryptcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generatedatakeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generatedatakeypaircommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generatedatakeypairwithoutplaintextcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generatedatakeywithoutplaintextcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generatemaccommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/generaterandomcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/getkeypolicycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/getkeyrotationstatuscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/getparametersforimportcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/getpublickeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/importkeymaterialcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listaliasescommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listgrantscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listkeypoliciescommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listkeyrotationscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listkeyscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listresourcetagscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/listretirablegrantscommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/putkeypolicycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/reencryptcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/replicatekeycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/retiregrantcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/revokegrantcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/rotatekeyondemandcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/schedulekeydeletioncommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/signcommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/tagresourcecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/untagresourcecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/updatealiascommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/updatecustomkeystorecommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/updatekeydescriptioncommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/updateprimaryregioncommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/verifycommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/verifymaccommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/endpoint/endpointparameters.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/auth/httpauthextensionconfiguration.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/extensionconfiguration.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/runtimeextensions.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/kmsclient.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/kms.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/interfaces.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/describecustomkeystorespaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listaliasespaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listgrantspaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listkeypoliciespaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listkeyrotationspaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listkeyspaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listresourcetagspaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/listretirablegrantspaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/kmsserviceexception.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/index.d.ts","./src/backup/disaster-recovery/disaster-recovery.service.ts","./src/backup/integrity/data-integrity.service.ts","./src/backup/dto/recovery-test-response.dto.ts","./src/backup/testing/recovery-testing.service.ts","./src/backup/monitoring/backup-monitoring.service.ts","./src/backup/dto/restore-backup.dto.ts","./src/backup/dto/trigger-recovery-test.dto.ts","./src/backup/backup.controller.ts","./src/backup/processing/backup-queue.processor.ts","./src/backup/backup.module.ts","./src/collaboration/documents/shared-document.service.ts","./src/collaboration/whiteboard/whiteboard.service.ts","./src/collaboration/versioning/version-control.service.ts","./src/collaboration/permissions/collaboration-permissions.service.ts","./src/collaboration/collaboration.service.ts","./node_modules/@nestjs/websockets/adapters/ws-adapter.d.ts","./node_modules/@nestjs/websockets/adapters/index.d.ts","./node_modules/@nestjs/websockets/decorators/connected-socket.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/gateway-server.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/message-body.decorator.d.ts","./node_modules/@nestjs/websockets/interfaces/gateway-metadata.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-connection.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-disconnect.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-init.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/index.d.ts","./node_modules/@nestjs/websockets/interfaces/server-and-event-streams-host.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/web-socket-server.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/ws-response.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/index.d.ts","./node_modules/@nestjs/websockets/decorators/socket-gateway.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/subscribe-message.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/index.d.ts","./node_modules/@nestjs/websockets/errors/ws-exception.d.ts","./node_modules/@nestjs/websockets/errors/index.d.ts","./node_modules/@nestjs/websockets/exceptions/base-ws-exception-filter.d.ts","./node_modules/@nestjs/websockets/exceptions/index.d.ts","./node_modules/@nestjs/websockets/interfaces/nest-gateway.interface.d.ts","./node_modules/@nestjs/websockets/gateway-metadata-explorer.d.ts","./node_modules/@nestjs/websockets/index.d.ts","./src/collaboration/constants/collaboration-events.constants.ts","./node_modules/engine.io-parser/build/cjs/commons.d.ts","./node_modules/engine.io-parser/build/cjs/encodepacket.d.ts","./node_modules/engine.io-parser/build/cjs/decodepacket.d.ts","./node_modules/engine.io-parser/build/cjs/index.d.ts","./node_modules/engine.io/build/parser-v3/index.d.ts","./node_modules/@types/ws/index.d.ts","./node_modules/engine.io/build/transport.d.ts","./node_modules/engine.io/build/socket.d.ts","./node_modules/@types/cors/index.d.ts","./node_modules/engine.io/build/contrib/types.cookie.d.ts","./node_modules/engine.io/build/server.d.ts","./node_modules/engine.io/build/transports/polling.d.ts","./node_modules/engine.io/build/transports/websocket.d.ts","./node_modules/engine.io/build/transports/webtransport.d.ts","./node_modules/engine.io/build/transports/index.d.ts","./node_modules/engine.io/build/userver.d.ts","./node_modules/engine.io/build/engine.io.d.ts","./node_modules/@socket.io/component-emitter/lib/cjs/index.d.ts","./node_modules/socket.io-parser/build/cjs/index.d.ts","./node_modules/socket.io/dist/typed-events.d.ts","./node_modules/socket.io/dist/client.d.ts","./node_modules/socket.io-adapter/dist/in-memory-adapter.d.ts","./node_modules/socket.io-adapter/dist/cluster-adapter.d.ts","./node_modules/socket.io-adapter/dist/index.d.ts","./node_modules/socket.io/dist/socket-types.d.ts","./node_modules/socket.io/dist/broadcast-operator.d.ts","./node_modules/socket.io/dist/socket.d.ts","./node_modules/socket.io/dist/namespace.d.ts","./node_modules/socket.io/dist/index.d.ts","./src/common/guards/ws-throttler.guard.ts","./src/common/utils/websocket.utils.ts","./src/collaboration/gateway/collaboration.gateway.ts","./src/courses/entities/lesson.entity.ts","./src/courses/entities/course-module.entity.ts","./src/courses/entities/enrollment.entity.ts","./src/courses/entities/course.entity.ts","./src/users/entities/user.entity.ts","./src/auth/decorators/roles.decorator.ts","./src/auth/guards/roles.guard.ts","./src/collaboration/collaboration.controller.ts","./src/collaboration/collaboration.module.ts","./src/data-warehouse/etl/etl-pipeline.service.ts","./src/data-warehouse/modeling/dimensional-modeling.service.ts","./src/data-warehouse/quality/data-quality.service.ts","./src/data-warehouse/lineage/data-lineage.service.ts","./src/data-warehouse/loading/incremental-loader.service.ts","./src/data-warehouse/data-warehouse.controller.ts","./src/data-warehouse/data-warehouse.module.ts","./src/queues/enums/job-priority.enum.ts","./src/queues/interfaces/queue.interfaces.ts","./src/queues/queue.service.ts","./src/common/decorators/roles.decorator.ts","./src/common/guards/roles.guard.ts","./src/queues/prioritization/prioritization.service.ts","./src/queues/scheduler/job-scheduler.service.ts","./src/queues/monitoring/queue-monitoring.service.ts","./node_modules/class-transformer/types/interfaces/decorator-options/expose-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/exclude-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/transform-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/type-discriminator-descriptor.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/type-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/exclude-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/expose-metadata.interface.d.ts","./node_modules/class-transformer/types/enums/transformation-type.enum.d.ts","./node_modules/class-transformer/types/enums/index.d.ts","./node_modules/class-transformer/types/interfaces/target-map.interface.d.ts","./node_modules/class-transformer/types/interfaces/class-transformer-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/transform-fn-params.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/transform-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/type-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/class-constructor.type.d.ts","./node_modules/class-transformer/types/interfaces/type-help-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/index.d.ts","./node_modules/class-transformer/types/classtransformer.d.ts","./node_modules/class-transformer/types/decorators/exclude.decorator.d.ts","./node_modules/class-transformer/types/decorators/expose.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-instance-to-instance.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-instance-to-plain.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-plain-to-instance.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform.decorator.d.ts","./node_modules/class-transformer/types/decorators/type.decorator.d.ts","./node_modules/class-transformer/types/decorators/index.d.ts","./node_modules/class-transformer/types/index.d.ts","./src/queues/dto/queue.dto.ts","./src/queues/queue.controller.ts","./src/queues/retry/retry-logic.service.ts","./src/common/utils/pii-sanitizer.utils.ts","./src/queues/processors/default-queue.processor.ts","./src/queues/queue.module.ts","./node_modules/@nestjs/graphql/dist/decorators/args-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/base-type-options.interface.d.ts","./node_modules/graphql/version.d.ts","./node_modules/graphql/jsutils/maybe.d.ts","./node_modules/graphql/language/source.d.ts","./node_modules/graphql/jsutils/objmap.d.ts","./node_modules/graphql/jsutils/path.d.ts","./node_modules/graphql/jsutils/promiseorvalue.d.ts","./node_modules/graphql/language/kinds.d.ts","./node_modules/graphql/language/tokenkind.d.ts","./node_modules/graphql/language/ast.d.ts","./node_modules/graphql/language/location.d.ts","./node_modules/graphql/error/graphqlerror.d.ts","./node_modules/graphql/language/directivelocation.d.ts","./node_modules/graphql/type/directives.d.ts","./node_modules/graphql/type/schema.d.ts","./node_modules/graphql/type/definition.d.ts","./node_modules/graphql/execution/execute.d.ts","./node_modules/graphql/graphql.d.ts","./node_modules/graphql/type/scalars.d.ts","./node_modules/graphql/type/introspection.d.ts","./node_modules/graphql/type/validate.d.ts","./node_modules/graphql/type/assertname.d.ts","./node_modules/graphql/type/index.d.ts","./node_modules/graphql/language/printlocation.d.ts","./node_modules/graphql/language/lexer.d.ts","./node_modules/graphql/language/parser.d.ts","./node_modules/graphql/language/printer.d.ts","./node_modules/graphql/language/visitor.d.ts","./node_modules/graphql/language/predicates.d.ts","./node_modules/graphql/language/index.d.ts","./node_modules/graphql/execution/subscribe.d.ts","./node_modules/graphql/execution/values.d.ts","./node_modules/graphql/execution/index.d.ts","./node_modules/graphql/subscription/index.d.ts","./node_modules/graphql/utilities/typeinfo.d.ts","./node_modules/graphql/validation/validationcontext.d.ts","./node_modules/graphql/validation/validate.d.ts","./node_modules/graphql/validation/rules/maxintrospectiondepthrule.d.ts","./node_modules/graphql/validation/specifiedrules.d.ts","./node_modules/graphql/validation/rules/executabledefinitionsrule.d.ts","./node_modules/graphql/validation/rules/fieldsoncorrecttyperule.d.ts","./node_modules/graphql/validation/rules/fragmentsoncompositetypesrule.d.ts","./node_modules/graphql/validation/rules/knownargumentnamesrule.d.ts","./node_modules/graphql/validation/rules/knowndirectivesrule.d.ts","./node_modules/graphql/validation/rules/knownfragmentnamesrule.d.ts","./node_modules/graphql/validation/rules/knowntypenamesrule.d.ts","./node_modules/graphql/validation/rules/loneanonymousoperationrule.d.ts","./node_modules/graphql/validation/rules/nofragmentcyclesrule.d.ts","./node_modules/graphql/validation/rules/noundefinedvariablesrule.d.ts","./node_modules/graphql/validation/rules/nounusedfragmentsrule.d.ts","./node_modules/graphql/validation/rules/nounusedvariablesrule.d.ts","./node_modules/graphql/validation/rules/overlappingfieldscanbemergedrule.d.ts","./node_modules/graphql/validation/rules/possiblefragmentspreadsrule.d.ts","./node_modules/graphql/validation/rules/providedrequiredargumentsrule.d.ts","./node_modules/graphql/validation/rules/scalarleafsrule.d.ts","./node_modules/graphql/validation/rules/singlefieldsubscriptionsrule.d.ts","./node_modules/graphql/validation/rules/uniqueargumentnamesrule.d.ts","./node_modules/graphql/validation/rules/uniquedirectivesperlocationrule.d.ts","./node_modules/graphql/validation/rules/uniquefragmentnamesrule.d.ts","./node_modules/graphql/validation/rules/uniqueinputfieldnamesrule.d.ts","./node_modules/graphql/validation/rules/uniqueoperationnamesrule.d.ts","./node_modules/graphql/validation/rules/uniquevariablenamesrule.d.ts","./node_modules/graphql/validation/rules/valuesofcorrecttyperule.d.ts","./node_modules/graphql/validation/rules/variablesareinputtypesrule.d.ts","./node_modules/graphql/validation/rules/variablesinallowedpositionrule.d.ts","./node_modules/graphql/validation/rules/loneschemadefinitionrule.d.ts","./node_modules/graphql/validation/rules/uniqueoperationtypesrule.d.ts","./node_modules/graphql/validation/rules/uniquetypenamesrule.d.ts","./node_modules/graphql/validation/rules/uniqueenumvaluenamesrule.d.ts","./node_modules/graphql/validation/rules/uniquefielddefinitionnamesrule.d.ts","./node_modules/graphql/validation/rules/uniqueargumentdefinitionnamesrule.d.ts","./node_modules/graphql/validation/rules/uniquedirectivenamesrule.d.ts","./node_modules/graphql/validation/rules/possibletypeextensionsrule.d.ts","./node_modules/graphql/validation/rules/custom/nodeprecatedcustomrule.d.ts","./node_modules/graphql/validation/rules/custom/noschemaintrospectioncustomrule.d.ts","./node_modules/graphql/validation/index.d.ts","./node_modules/graphql/error/syntaxerror.d.ts","./node_modules/graphql/error/locatederror.d.ts","./node_modules/graphql/error/index.d.ts","./node_modules/graphql/utilities/getintrospectionquery.d.ts","./node_modules/graphql/utilities/getoperationast.d.ts","./node_modules/graphql/utilities/getoperationroottype.d.ts","./node_modules/graphql/utilities/introspectionfromschema.d.ts","./node_modules/graphql/utilities/buildclientschema.d.ts","./node_modules/graphql/utilities/buildastschema.d.ts","./node_modules/graphql/utilities/extendschema.d.ts","./node_modules/graphql/utilities/lexicographicsortschema.d.ts","./node_modules/graphql/utilities/printschema.d.ts","./node_modules/graphql/utilities/typefromast.d.ts","./node_modules/graphql/utilities/valuefromast.d.ts","./node_modules/graphql/utilities/valuefromastuntyped.d.ts","./node_modules/graphql/utilities/astfromvalue.d.ts","./node_modules/graphql/utilities/coerceinputvalue.d.ts","./node_modules/graphql/utilities/concatast.d.ts","./node_modules/graphql/utilities/separateoperations.d.ts","./node_modules/graphql/utilities/stripignoredcharacters.d.ts","./node_modules/graphql/utilities/typecomparators.d.ts","./node_modules/graphql/utilities/assertvalidname.d.ts","./node_modules/graphql/utilities/findbreakingchanges.d.ts","./node_modules/graphql/utilities/typedquerydocumentnode.d.ts","./node_modules/graphql/utilities/resolveschemacoordinate.d.ts","./node_modules/graphql/utilities/index.d.ts","./node_modules/graphql/index.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/field-middleware.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/build-schema-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/complexity.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/custom-scalar.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/gql-exception-filter.interface.d.ts","./node_modules/@graphql-typed-document-node/core/typings/index.d.ts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/interfaces.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/loaders.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/helpers.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getdirectiveextensions.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/types.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-implementing-types.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/validate-documents.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/filterschema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/heal.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getresolversfromschema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/foreachfield.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/foreachdefaultvalue.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mapschema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/addtypes.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/rewire.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/prune.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mergedeep.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/stub.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/selectionsets.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getresponsekeyfrominfo.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fields.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/renametype.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/transforminputvalue.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/executor.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mapasynciterator.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/updateargument.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/implementsabstracttype.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/errors.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/observabletoasynciterable.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/visitresult.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getargumentvalues.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/valuematchescriteria.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/isasynciterable.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/isdocumentnode.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/astfromvalueuntyped.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/withcancel.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/roottypes.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/comments.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/collectfields.d.cts","./node_modules/cross-inspect/typings/index.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/memoize.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fixschemaast.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getoperationastfromrequest.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/extractextensionsfromschema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/path.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/jsutils.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mergeincrementalresult.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/debugtimer.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/map-maybe-promise.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fakepromise.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/createdeferred.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/index.d.cts","./node_modules/@nestjs/graphql/dist/graphql-ast.explorer.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/schema-file-config.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/gql-module-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/graphql-driver.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/resolve-type-fn.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/return-type-func.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/build-federated-schema-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/index.d.ts","./node_modules/@nestjs/graphql/dist/decorators/args.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/context.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/directive.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/extensions.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/hide-field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/info.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/input-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/interface-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/mutation.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/object-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/parent.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/query.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-property.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-reference.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolver.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/root.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/scalar.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/subscription.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/index.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/type-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/directive.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/param.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/resolver.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/orphaned-reference.registry.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/property.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/class.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/enum.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/extensions.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/union.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/type-mapper.service.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/enum-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/type-fields.accessor.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/ast-definition-node.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/input-type-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/interface.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/output-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/resolve-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/interface-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/object-type.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/object-type-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/union-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/type-definitions.storage.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/input-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/args.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/root-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/mutation-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/orphaned-types.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/query-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/subscription-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/type-definitions.generator.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/graphql-schema.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/helpers/file-system.helper.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/resolver-metadata.interface.d.ts","./node_modules/@nestjs/graphql/dist/services/base-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-arguments-host.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-execution-context.d.ts","./node_modules/graphql-ws/lib/common.d.ts","./node_modules/graphql-ws/lib/client.d.ts","./node_modules/graphql-ws/lib/server.d.ts","./node_modules/graphql-ws/lib/index.d.ts","./node_modules/subscriptions-transport-ws/node_modules/eventemitter3/index.d.ts","./node_modules/subscriptions-transport-ws/dist/client.d.ts","./node_modules/subscriptions-transport-ws/dist/server.d.ts","./node_modules/subscriptions-transport-ws/dist/message-types.d.ts","./node_modules/subscriptions-transport-ws/dist/protocol.d.ts","./node_modules/subscriptions-transport-ws/dist/index.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-subscription.service.d.ts","./node_modules/@nestjs/graphql/dist/services/resolvers-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/scalars-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-schema.builder.d.ts","./node_modules/@nestjs/graphql/dist/graphql.factory.d.ts","./node_modules/@nestjs/graphql/dist/drivers/abstract-graphql.driver.d.ts","./node_modules/@nestjs/graphql/dist/drivers/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-types.loader.d.ts","./node_modules/@nestjs/graphql/dist/graphql-definitions.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/graphql-federation-definitions.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/type-defs-decorator.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/graphql-federation.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-schema.host.d.ts","./node_modules/@nestjs/graphql/dist/graphql.constants.d.ts","./node_modules/@nestjs/graphql/dist/graphql.module.d.ts","./node_modules/@nestjs/graphql/dist/scalars/iso-date.scalar.d.ts","./node_modules/@nestjs/graphql/dist/scalars/timestamp.scalar.d.ts","./node_modules/@nestjs/graphql/dist/scalars/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/type-metadata.storage.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/schema-builder.module.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/index.d.ts","./node_modules/@nestjs/graphql/dist/tokens.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/create-union-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/register-enum-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/index.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/field-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/class-decorator-factory.interface.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/intersection-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/omit-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/partial-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/pick-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/index.d.ts","./node_modules/@nestjs/graphql/dist/utils/extend.util.d.ts","./node_modules/@nestjs/graphql/dist/utils/transform-schema.util.d.ts","./node_modules/@nestjs/graphql/dist/index.d.ts","./node_modules/@nestjs/apollo/dist/decorators/plugin.decorator.d.ts","./node_modules/@nestjs/apollo/dist/decorators/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/keyvaluecache.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/prefixingkeyvaluecache.d.ts","./node_modules/lru-cache/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/inmemorylrucache.d.ts","./node_modules/@apollo/utils.logger/dist/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/errorsaremissescache.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/index.d.ts","./node_modules/@apollo/protobufjs/index.d.ts","./node_modules/@apollo/usage-reporting-protobuf/generated/cjs/protobuf.d.ts","./node_modules/@apollo/utils.fetcher/dist/index.d.ts","./node_modules/@apollo/server-gateway-interface/dist/cjs/index.d.ts","./node_modules/@apollo/utils.withrequired/dist/index.d.ts","./node_modules/@apollo/server/dist/cjs/utils/resolvable.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/context.d.ts","./node_modules/@apollo/server/dist/cjs/utils/headermap.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/http.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/incrementaldeliverypolyfill.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/graphql.d.ts","./node_modules/@graphql-tools/utils/typings/interfaces.d.cts","./node_modules/@graphql-tools/utils/typings/loaders.d.cts","./node_modules/@graphql-tools/utils/typings/helpers.d.cts","./node_modules/@graphql-tools/utils/typings/get-directives.d.cts","./node_modules/@graphql-tools/utils/typings/types.d.cts","./node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/get-implementing-types.d.cts","./node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/validate-documents.d.cts","./node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.cts","./node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.cts","./node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.cts","./node_modules/@graphql-tools/utils/typings/filterschema.d.cts","./node_modules/@graphql-tools/utils/typings/heal.d.cts","./node_modules/@graphql-tools/utils/typings/getresolversfromschema.d.cts","./node_modules/@graphql-tools/utils/typings/foreachfield.d.cts","./node_modules/@graphql-tools/utils/typings/foreachdefaultvalue.d.cts","./node_modules/@graphql-tools/utils/typings/mapschema.d.cts","./node_modules/@graphql-tools/utils/typings/addtypes.d.cts","./node_modules/@graphql-tools/utils/typings/rewire.d.cts","./node_modules/@graphql-tools/utils/typings/prune.d.cts","./node_modules/@graphql-tools/utils/typings/mergedeep.d.cts","./node_modules/@graphql-tools/utils/typings/stub.d.cts","./node_modules/@graphql-tools/utils/typings/selectionsets.d.cts","./node_modules/@graphql-tools/utils/typings/getresponsekeyfrominfo.d.cts","./node_modules/@graphql-tools/utils/typings/fields.d.cts","./node_modules/@graphql-tools/utils/typings/renametype.d.cts","./node_modules/@graphql-tools/utils/typings/transforminputvalue.d.cts","./node_modules/@graphql-tools/utils/typings/mapasynciterator.d.cts","./node_modules/@graphql-tools/utils/typings/updateargument.d.cts","./node_modules/@graphql-tools/utils/typings/implementsabstracttype.d.cts","./node_modules/@graphql-tools/utils/typings/errors.d.cts","./node_modules/@graphql-tools/utils/typings/observabletoasynciterable.d.cts","./node_modules/@graphql-tools/utils/typings/visitresult.d.cts","./node_modules/@graphql-tools/utils/typings/getargumentvalues.d.cts","./node_modules/@graphql-tools/utils/typings/valuematchescriteria.d.cts","./node_modules/@graphql-tools/utils/typings/isasynciterable.d.cts","./node_modules/@graphql-tools/utils/typings/isdocumentnode.d.cts","./node_modules/@graphql-tools/utils/typings/astfromvalueuntyped.d.cts","./node_modules/@graphql-tools/utils/typings/executor.d.cts","./node_modules/@graphql-tools/utils/typings/withcancel.d.cts","./node_modules/@graphql-tools/utils/typings/aggregateerror.d.cts","./node_modules/@graphql-tools/utils/typings/roottypes.d.cts","./node_modules/@graphql-tools/utils/typings/comments.d.cts","./node_modules/@graphql-tools/utils/typings/collectfields.d.cts","./node_modules/@graphql-tools/utils/typings/inspect.d.cts","./node_modules/@graphql-tools/utils/typings/memoize.d.cts","./node_modules/@graphql-tools/utils/typings/fixschemaast.d.cts","./node_modules/@graphql-tools/utils/typings/getoperationastfromrequest.d.cts","./node_modules/@graphql-tools/utils/typings/extractextensionsfromschema.d.cts","./node_modules/@graphql-tools/utils/typings/path.d.cts","./node_modules/@graphql-tools/utils/typings/jsutils.d.cts","./node_modules/@graphql-tools/utils/typings/directives.d.cts","./node_modules/@graphql-tools/utils/typings/index.d.cts","./node_modules/@graphql-tools/schema/typings/assertresolverspresent.d.cts","./node_modules/@graphql-tools/schema/typings/chainresolvers.d.cts","./node_modules/@graphql-tools/schema/typings/addresolverstoschema.d.cts","./node_modules/@graphql-tools/schema/typings/checkforresolvetyperesolver.d.cts","./node_modules/@graphql-tools/schema/typings/extendresolversfrominterfaces.d.cts","./node_modules/@graphql-tools/schema/typings/types.d.cts","./node_modules/@graphql-tools/schema/typings/makeexecutableschema.d.cts","./node_modules/@graphql-tools/schema/typings/merge-schemas.d.cts","./node_modules/@graphql-tools/schema/typings/index.d.cts","./node_modules/@apollo/server/dist/cjs/incrementaldeliverypolyfill.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/constructor.d.ts","./node_modules/@apollo/cache-control-types/dist/cjs/index.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/requestpipeline.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/plugins.d.ts","./node_modules/@apollo/server/dist/cjs/externaltypes/index.d.ts","./node_modules/@apollo/server/dist/cjs/utils/schemamanager.d.ts","./node_modules/@apollo/server/dist/cjs/apolloserver.d.ts","./node_modules/@apollo/server/dist/cjs/index.d.ts","./node_modules/@apollographql/graphql-playground-html/dist/render-playground-page.d.ts","./node_modules/@apollographql/graphql-playground-html/dist/index.d.ts","./node_modules/@apollo/usage-reporting-protobuf/generated/esm/protobuf.d.ts","./node_modules/@apollo/server-gateway-interface/dist/esm/index.d.ts","./node_modules/@apollo/server/dist/esm/utils/resolvable.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/context.d.ts","./node_modules/@apollo/server/dist/esm/utils/headermap.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/http.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/incrementaldeliverypolyfill.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/graphql.d.ts","./node_modules/@graphql-tools/utils/typings/interfaces.d.ts","./node_modules/@graphql-tools/utils/typings/loaders.d.ts","./node_modules/@graphql-tools/utils/typings/helpers.d.ts","./node_modules/@graphql-tools/utils/typings/get-directives.d.ts","./node_modules/@graphql-tools/utils/typings/types.d.ts","./node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/get-implementing-types.d.ts","./node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/validate-documents.d.ts","./node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.ts","./node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.ts","./node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.ts","./node_modules/@graphql-tools/utils/typings/filterschema.d.ts","./node_modules/@graphql-tools/utils/typings/heal.d.ts","./node_modules/@graphql-tools/utils/typings/getresolversfromschema.d.ts","./node_modules/@graphql-tools/utils/typings/foreachfield.d.ts","./node_modules/@graphql-tools/utils/typings/foreachdefaultvalue.d.ts","./node_modules/@graphql-tools/utils/typings/mapschema.d.ts","./node_modules/@graphql-tools/utils/typings/addtypes.d.ts","./node_modules/@graphql-tools/utils/typings/rewire.d.ts","./node_modules/@graphql-tools/utils/typings/prune.d.ts","./node_modules/@graphql-tools/utils/typings/mergedeep.d.ts","./node_modules/@graphql-tools/utils/typings/stub.d.ts","./node_modules/@graphql-tools/utils/typings/selectionsets.d.ts","./node_modules/@graphql-tools/utils/typings/getresponsekeyfrominfo.d.ts","./node_modules/@graphql-tools/utils/typings/fields.d.ts","./node_modules/@graphql-tools/utils/typings/renametype.d.ts","./node_modules/@graphql-tools/utils/typings/transforminputvalue.d.ts","./node_modules/@graphql-tools/utils/typings/mapasynciterator.d.ts","./node_modules/@graphql-tools/utils/typings/updateargument.d.ts","./node_modules/@graphql-tools/utils/typings/implementsabstracttype.d.ts","./node_modules/@graphql-tools/utils/typings/errors.d.ts","./node_modules/@graphql-tools/utils/typings/observabletoasynciterable.d.ts","./node_modules/@graphql-tools/utils/typings/visitresult.d.ts","./node_modules/@graphql-tools/utils/typings/getargumentvalues.d.ts","./node_modules/@graphql-tools/utils/typings/valuematchescriteria.d.ts","./node_modules/@graphql-tools/utils/typings/isasynciterable.d.ts","./node_modules/@graphql-tools/utils/typings/isdocumentnode.d.ts","./node_modules/@graphql-tools/utils/typings/astfromvalueuntyped.d.ts","./node_modules/@graphql-tools/utils/typings/executor.d.ts","./node_modules/@graphql-tools/utils/typings/withcancel.d.ts","./node_modules/@graphql-tools/utils/typings/aggregateerror.d.ts","./node_modules/@graphql-tools/utils/typings/roottypes.d.ts","./node_modules/@graphql-tools/utils/typings/comments.d.ts","./node_modules/@graphql-tools/utils/typings/collectfields.d.ts","./node_modules/@graphql-tools/utils/typings/inspect.d.ts","./node_modules/@graphql-tools/utils/typings/memoize.d.ts","./node_modules/@graphql-tools/utils/typings/fixschemaast.d.ts","./node_modules/@graphql-tools/utils/typings/getoperationastfromrequest.d.ts","./node_modules/@graphql-tools/utils/typings/extractextensionsfromschema.d.ts","./node_modules/@graphql-tools/utils/typings/path.d.ts","./node_modules/@graphql-tools/utils/typings/jsutils.d.ts","./node_modules/@graphql-tools/utils/typings/directives.d.ts","./node_modules/@graphql-tools/utils/typings/index.d.ts","./node_modules/@graphql-tools/schema/typings/assertresolverspresent.d.ts","./node_modules/@graphql-tools/schema/typings/chainresolvers.d.ts","./node_modules/@graphql-tools/schema/typings/addresolverstoschema.d.ts","./node_modules/@graphql-tools/schema/typings/checkforresolvetyperesolver.d.ts","./node_modules/@graphql-tools/schema/typings/extendresolversfrominterfaces.d.ts","./node_modules/@graphql-tools/schema/typings/types.d.ts","./node_modules/@graphql-tools/schema/typings/makeexecutableschema.d.ts","./node_modules/@graphql-tools/schema/typings/merge-schemas.d.ts","./node_modules/@graphql-tools/schema/typings/index.d.ts","./node_modules/@apollo/server/dist/esm/incrementaldeliverypolyfill.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/constructor.d.ts","./node_modules/@apollo/cache-control-types/dist/esm/index.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/requestpipeline.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/plugins.d.ts","./node_modules/@apollo/server/dist/esm/externaltypes/index.d.ts","./node_modules/@apollo/server/dist/esm/utils/schemamanager.d.ts","./node_modules/@apollo/server/dist/esm/apolloserver.d.ts","./node_modules/@apollo/server/dist/esm/index.d.ts","./node_modules/@apollo/server-plugin-landing-page-graphql-playground/dist/esm/index.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-federation-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-gateway-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/index.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-base.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-federation.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-gateway.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/index.d.ts","./node_modules/@nestjs/apollo/dist/errors/authentication.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/forbidden.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/user-input.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/validation.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/index.d.ts","./node_modules/@nestjs/apollo/dist/utils/get-apollo-server.d.ts","./node_modules/@nestjs/apollo/dist/utils/index.d.ts","./node_modules/@nestjs/apollo/dist/services/plugins-explorer.service.d.ts","./node_modules/@nestjs/apollo/dist/services/index.d.ts","./node_modules/@nestjs/apollo/dist/index.d.ts","./node_modules/graphql-subscriptions/dist/pubsub-async-iterable-iterator.d.ts","./node_modules/graphql-subscriptions/dist/pubsub-engine.d.ts","./node_modules/graphql-subscriptions/dist/pubsub.d.ts","./node_modules/graphql-subscriptions/dist/with-filter.d.ts","./node_modules/graphql-subscriptions/dist/index.d.ts","./src/common/validators/password.validator.ts","./src/users/dto/create-user.dto.ts","./src/users/dto/update-user.dto.ts","./node_modules/bcryptjs/umd/types.d.ts","./node_modules/bcryptjs/umd/index.d.ts","./src/common/utils/user.utils.ts","./src/common/constants/app.constants.ts","./src/common/dto/pagination.dto.ts","./src/common/utils/pagination.util.ts","./src/common/utils/sanitization.utils.ts","./src/users/dto/get-users.dto.ts","./src/caching/caching.constants.ts","./src/caching/caching.service.ts","./src/users/users.service.ts","./src/auth/decorators/current-user.decorator.ts","./src/common/export/export.service.ts","./src/users/users.controller.ts","./src/users/users.module.ts","./src/courses/dto/create-course.dto.ts","./src/courses/dto/update-course.dto.ts","./src/courses/dto/course-search.dto.ts","./src/caching/strategies/cache-strategies.service.ts","./src/caching/invalidation/invalidation.service.ts","./src/courses/courses.service.ts","./src/courses/dto/create-module.dto.ts","./src/courses/modules/modules.service.ts","./src/courses/dto/create-lesson.dto.ts","./src/courses/lessons/lessons.service.ts","./src/courses/enrollments/enrollments.service.ts","./src/courses/courses.controller.ts","./src/courses/courses.module.ts","./src/assessment/enums/assessment-status.enum.ts","./src/assessment/enums/question-type.enum.ts","./src/assessment/entities/question.entity.ts","./src/assessment/entities/assessment.entity.ts","./src/assessment/entities/answer.entity.ts","./src/assessment/entities/assessment-attempt.entity.ts","./src/assessment/feedback/feedback-generation.service.ts","./src/assessment/scoring/score-calculation.service.ts","./src/assessment/assessments.service.ts","./src/assessment/assessment.controller.ts","./src/assessment/questions/question-bank.service.ts","./src/assessment/assessment.module.ts","./node_modules/@types/ms/index.d.ts","./node_modules/@types/jsonwebtoken/index.d.ts","./node_modules/@nestjs/jwt/dist/interfaces/jwt-module-options.interface.d.ts","./node_modules/@nestjs/jwt/dist/interfaces/index.d.ts","./node_modules/@nestjs/jwt/dist/jwt.errors.d.ts","./node_modules/@nestjs/jwt/dist/jwt.module.d.ts","./node_modules/@nestjs/jwt/dist/jwt.service.d.ts","./node_modules/@nestjs/jwt/dist/index.d.ts","./node_modules/@nestjs/jwt/index.d.ts","./src/auth/dto/auth.dto.ts","./src/notifications/entities/notification.entity.ts","./src/notifications/entities/notification-preferences.entity.ts","./src/notifications/dto/notification.dto.ts","./src/auth/guards/ws-jwt-auth.guard.ts","./src/notifications/notifications.gateway.ts","./src/notifications/notification-templates.service.ts","./src/notifications/preferences/preferences.service.ts","./node_modules/@types/nodemailer/lib/dkim/index.d.ts","./node_modules/@types/nodemailer/lib/mailer/mail-message.d.ts","./node_modules/@types/nodemailer/lib/xoauth2/index.d.ts","./node_modules/@types/nodemailer/lib/mailer/index.d.ts","./node_modules/@types/nodemailer/lib/mime-node/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-connection/index.d.ts","./node_modules/@types/nodemailer/lib/shared/index.d.ts","./node_modules/@types/nodemailer/lib/json-transport/index.d.ts","./node_modules/@types/nodemailer/lib/sendmail-transport/index.d.ts","./node_modules/@types/nodemailer/lib/ses-transport/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-pool/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-transport/index.d.ts","./node_modules/@types/nodemailer/lib/stream-transport/index.d.ts","./node_modules/@types/nodemailer/index.d.ts","./node_modules/handlebars/types/index.d.ts","./src/notifications/email/email.service.ts","./src/notifications/notifications.service.ts","./src/audit-log/enums/audit-action.enum.ts","./src/audit-log/audit-log.entity.ts","./src/audit-log/audit-log.service.ts","./src/auth/auth.service.ts","./src/auth/auth.controller.ts","./node_modules/@types/passport-strategy/index.d.ts","./node_modules/@types/passport-jwt/index.d.ts","./src/auth/strategies/jwt.strategy.ts","./src/notifications/notifications.controller.ts","./src/notifications/email/email.processor.ts","./src/notifications/notifications.module.ts","./src/audit-log/audit-log.controller.ts","./src/audit-log/interceptors/audit-log.interceptor.ts","./src/audit-log/tasks/audit-retention.task.ts","./src/audit-log/audit-log.module.ts","./src/auth/auth.module.ts","./src/graphql/types/course.type.ts","./src/graphql/types/user.type.ts","./src/graphql/types/assessment.type.ts","./src/graphql/inputs/user.input.ts","./src/graphql/inputs/course.input.ts","./src/graphql/resolvers/query.resolver.ts","./src/graphql/inputs/assessment.input.ts","./src/graphql/resolvers/mutation.resolver.ts","./src/graphql/resolvers/subscription.resolver.ts","./src/graphql/resolvers/user.resolver.ts","./src/graphql/resolvers/course.resolver.ts","./src/graphql/resolvers/assessment.resolver.ts","./node_modules/dataloader/index.d.ts","./src/graphql/services/dataloader.service.ts","./src/graphql/services/query-complexity.service.ts","./src/graphql/graphql.module.ts","./src/migrations/entities/migration.entity.ts","./src/migrations/conflicts/conflict-resolution.service.ts","./src/migrations/validation/schema-validation.service.ts","./src/migrations/samples/001-create-users-table.migration.ts","./src/migrations/samples/002-create-courses-table.migration.ts","./src/migrations/samples/003-create-course-modules-table.migration.ts","./src/migrations/samples/004-create-lessons-table.migration.ts","./src/migrations/samples/005-create-enrollments-table.migration.ts","./src/migrations/samples/006-create-migrations-tracking-table.migration.ts","./src/migrations/migration.registry.ts","./src/migrations/rollback/rollback.service.ts","./src/migrations/environments/environment-sync.service.ts","./src/migrations/migration.service.ts","./src/migrations/migration.controller.ts","./src/migrations/migration-runner.service.ts","./src/migrations/migration.module.ts","./src/ab-testing/entities/variant-metric.entity.ts","./src/ab-testing/entities/experiment-variant.entity.ts","./src/ab-testing/entities/experiment-metric.entity.ts","./src/ab-testing/entities/experiment.entity.ts","./src/ab-testing/ab-testing.service.ts","./src/ab-testing/experiments/experiment.service.ts","./src/ab-testing/analysis/statistical-analysis.service.ts","./src/ab-testing/automation/automated-decision.service.ts","./src/ab-testing/reporting/ab-testing-reports.service.ts","./src/ab-testing/ab-testing.controller.ts","./src/ab-testing/ab-testing.module.ts","./src/observability/interfaces/observability.interfaces.ts","./src/observability/logging/structured-logger.service.ts","./src/observability/logging/log-aggregation.service.ts","./node_modules/@opentelemetry/api/build/src/baggage/internal/symbol.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/types.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/utils.d.ts","./node_modules/@opentelemetry/api/build/src/common/exception.d.ts","./node_modules/@opentelemetry/api/build/src/common/time.d.ts","./node_modules/@opentelemetry/api/build/src/common/attributes.d.ts","./node_modules/@opentelemetry/api/build/src/context/types.d.ts","./node_modules/@opentelemetry/api/build/src/context/context.d.ts","./node_modules/@opentelemetry/api/build/src/api/context.d.ts","./node_modules/@opentelemetry/api/build/src/diag/types.d.ts","./node_modules/@opentelemetry/api/build/src/diag/consolelogger.d.ts","./node_modules/@opentelemetry/api/build/src/api/diag.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/metric.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/meter.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/noopmeter.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/meterprovider.d.ts","./node_modules/@opentelemetry/api/build/src/api/metrics.d.ts","./node_modules/@opentelemetry/api/build/src/propagation/textmappropagator.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/context-helpers.d.ts","./node_modules/@opentelemetry/api/build/src/api/propagation.d.ts","./node_modules/@opentelemetry/api/build/src/trace/attributes.d.ts","./node_modules/@opentelemetry/api/build/src/trace/trace_state.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span_context.d.ts","./node_modules/@opentelemetry/api/build/src/trace/link.d.ts","./node_modules/@opentelemetry/api/build/src/trace/status.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span_kind.d.ts","./node_modules/@opentelemetry/api/build/src/trace/spanoptions.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer_options.d.ts","./node_modules/@opentelemetry/api/build/src/trace/proxytracer.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer_provider.d.ts","./node_modules/@opentelemetry/api/build/src/trace/proxytracerprovider.d.ts","./node_modules/@opentelemetry/api/build/src/trace/samplingresult.d.ts","./node_modules/@opentelemetry/api/build/src/trace/sampler.d.ts","./node_modules/@opentelemetry/api/build/src/trace/trace_flags.d.ts","./node_modules/@opentelemetry/api/build/src/trace/internal/utils.d.ts","./node_modules/@opentelemetry/api/build/src/trace/spancontext-utils.d.ts","./node_modules/@opentelemetry/api/build/src/trace/invalid-span-constants.d.ts","./node_modules/@opentelemetry/api/build/src/trace/context-utils.d.ts","./node_modules/@opentelemetry/api/build/src/api/trace.d.ts","./node_modules/@opentelemetry/api/build/src/context-api.d.ts","./node_modules/@opentelemetry/api/build/src/diag-api.d.ts","./node_modules/@opentelemetry/api/build/src/metrics-api.d.ts","./node_modules/@opentelemetry/api/build/src/propagation-api.d.ts","./node_modules/@opentelemetry/api/build/src/trace-api.d.ts","./node_modules/@opentelemetry/api/build/src/index.d.ts","./src/observability/tracing/distributed-tracing.service.ts","./src/observability/metrics/metrics-analysis.service.ts","./src/observability/anomaly/anomaly-detection.service.ts","./src/observability/observability.service.ts","./src/observability/observability.controller.ts","./src/observability/observability.module.ts","./src/rate-limiting/services/distrubutes.service.ts","./src/rate-limiting/services/quota.service.ts","./src/rate-limiting/services/adaptive-rate-limiting.service.ts","./src/rate-limiting/services/throttling.service.ts","./src/rate-limiting/dto/create-rate-limiting.dto.ts","./src/rate-limiting/dto/update-rate-limiting.dto.ts","./src/rate-limiting/rate-limiting.service.ts","./src/rate-limiting/services/rate-limiting.module.ts","./src/caching/warming/cache-warming.service.ts","./src/caching/analytics/cache-analytics.service.ts","./src/caching/cache-management.controller.ts","./src/caching/caching.module.ts","./src/feature-flags/interfaces/index.ts","./src/feature-flags/analytics/flag-analytics.service.ts","./src/feature-flags/rollout/rollout.service.ts","./src/feature-flags/experimentation/experimentation.service.ts","./src/feature-flags/targeting/targeting.service.ts","./src/feature-flags/evaluation/flag-evaluation.service.ts","./src/feature-flags/feature-flags.module.ts","./node_modules/@elastic/transport/lib/symbols.d.ts","./node_modules/@elastic/transport/lib/connection/baseconnection.d.ts","./node_modules/hpagent/index.d.ts","./node_modules/@elastic/transport/lib/connection/httpconnection.d.ts","./node_modules/undici/types/header.d.ts","./node_modules/undici/types/readable.d.ts","./node_modules/undici/types/file.d.ts","./node_modules/undici/types/fetch.d.ts","./node_modules/undici/types/formdata.d.ts","./node_modules/undici/types/connector.d.ts","./node_modules/undici/types/client.d.ts","./node_modules/undici/types/errors.d.ts","./node_modules/undici/types/dispatcher.d.ts","./node_modules/undici/types/global-dispatcher.d.ts","./node_modules/undici/types/global-origin.d.ts","./node_modules/undici/types/pool-stats.d.ts","./node_modules/undici/types/pool.d.ts","./node_modules/undici/types/handlers.d.ts","./node_modules/undici/types/balanced-pool.d.ts","./node_modules/undici/types/agent.d.ts","./node_modules/undici/types/mock-interceptor.d.ts","./node_modules/undici/types/mock-agent.d.ts","./node_modules/undici/types/mock-client.d.ts","./node_modules/undici/types/mock-pool.d.ts","./node_modules/undici/types/mock-errors.d.ts","./node_modules/undici/types/proxy-agent.d.ts","./node_modules/undici/types/env-http-proxy-agent.d.ts","./node_modules/undici/types/retry-handler.d.ts","./node_modules/undici/types/retry-agent.d.ts","./node_modules/undici/types/api.d.ts","./node_modules/undici/types/interceptors.d.ts","./node_modules/undici/types/util.d.ts","./node_modules/undici/types/cookies.d.ts","./node_modules/undici/types/patch.d.ts","./node_modules/undici/types/websocket.d.ts","./node_modules/undici/types/eventsource.d.ts","./node_modules/undici/types/filereader.d.ts","./node_modules/undici/types/diagnostics-channel.d.ts","./node_modules/undici/types/content-type.d.ts","./node_modules/undici/types/cache.d.ts","./node_modules/undici/types/index.d.ts","./node_modules/undici/index.d.ts","./node_modules/@elastic/transport/lib/connection/undiciconnection.d.ts","./node_modules/@elastic/transport/lib/connection/index.d.ts","./node_modules/@elastic/transport/lib/serializer.d.ts","./node_modules/@elastic/transport/lib/pool/baseconnectionpool.d.ts","./node_modules/@elastic/transport/lib/pool/weightedconnectionpool.d.ts","./node_modules/@elastic/transport/lib/pool/clusterconnectionpool.d.ts","./node_modules/@elastic/transport/lib/pool/cloudconnectionpool.d.ts","./node_modules/@elastic/transport/lib/pool/index.d.ts","./node_modules/@elastic/transport/lib/transport.d.ts","./node_modules/@elastic/transport/lib/types.d.ts","./node_modules/@elastic/transport/lib/errors.d.ts","./node_modules/@elastic/transport/lib/diagnostic.d.ts","./node_modules/@elastic/transport/index.d.ts","./node_modules/@elastic/elasticsearch/lib/sniffingtransport.d.ts","./node_modules/flatbuffers/js/constants.d.ts","./node_modules/flatbuffers/js/encoding.d.ts","./node_modules/flatbuffers/js/byte-buffer.d.ts","./node_modules/flatbuffers/js/builder.d.ts","./node_modules/flatbuffers/js/types.d.ts","./node_modules/flatbuffers/js/utils.d.ts","./node_modules/flatbuffers/js/flatbuffers.d.ts","./node_modules/apache-arrow/fb/body-compression-method.d.ts","./node_modules/apache-arrow/fb/compression-type.d.ts","./node_modules/apache-arrow/fb/body-compression.d.ts","./node_modules/apache-arrow/fb/buffer.d.ts","./node_modules/apache-arrow/fb/field-node.d.ts","./node_modules/apache-arrow/fb/record-batch.d.ts","./node_modules/apache-arrow/fb/dictionary-batch.d.ts","./node_modules/apache-arrow/fb/endianness.d.ts","./node_modules/apache-arrow/fb/dictionary-kind.d.ts","./node_modules/apache-arrow/fb/int.d.ts","./node_modules/apache-arrow/fb/dictionary-encoding.d.ts","./node_modules/apache-arrow/fb/key-value.d.ts","./node_modules/apache-arrow/fb/binary.d.ts","./node_modules/apache-arrow/fb/bool.d.ts","./node_modules/apache-arrow/fb/date-unit.d.ts","./node_modules/apache-arrow/fb/date.d.ts","./node_modules/apache-arrow/fb/decimal.d.ts","./node_modules/apache-arrow/fb/time-unit.d.ts","./node_modules/apache-arrow/fb/duration.d.ts","./node_modules/apache-arrow/fb/fixed-size-binary.d.ts","./node_modules/apache-arrow/fb/fixed-size-list.d.ts","./node_modules/apache-arrow/fb/precision.d.ts","./node_modules/apache-arrow/fb/floating-point.d.ts","./node_modules/apache-arrow/fb/interval-unit.d.ts","./node_modules/apache-arrow/fb/interval.d.ts","./node_modules/apache-arrow/fb/large-binary.d.ts","./node_modules/apache-arrow/fb/large-list.d.ts","./node_modules/apache-arrow/fb/large-utf8.d.ts","./node_modules/apache-arrow/fb/list.d.ts","./node_modules/apache-arrow/fb/map.d.ts","./node_modules/apache-arrow/fb/null.d.ts","./node_modules/apache-arrow/fb/run-end-encoded.d.ts","./node_modules/apache-arrow/fb/struct-.d.ts","./node_modules/apache-arrow/fb/time.d.ts","./node_modules/apache-arrow/fb/timestamp.d.ts","./node_modules/apache-arrow/fb/union-mode.d.ts","./node_modules/apache-arrow/fb/union.d.ts","./node_modules/apache-arrow/fb/utf8.d.ts","./node_modules/apache-arrow/fb/type.d.ts","./node_modules/apache-arrow/fb/field.d.ts","./node_modules/apache-arrow/fb/schema.d.ts","./node_modules/apache-arrow/fb/sparse-matrix-compressed-axis.d.ts","./node_modules/apache-arrow/fb/sparse-matrix-index-csx.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index-coo.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index-csf.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index.d.ts","./node_modules/apache-arrow/fb/tensor-dim.d.ts","./node_modules/apache-arrow/fb/sparse-tensor.d.ts","./node_modules/apache-arrow/fb/tensor.d.ts","./node_modules/apache-arrow/fb/message-header.d.ts","./node_modules/apache-arrow/fb/metadata-version.d.ts","./node_modules/apache-arrow/enum.d.ts","./node_modules/apache-arrow/schema.d.ts","./node_modules/apache-arrow/row/map.d.ts","./node_modules/apache-arrow/row/struct.d.ts","./node_modules/apache-arrow/builder/buffer.d.ts","./node_modules/apache-arrow/io/node/builder.d.ts","./node_modules/apache-arrow/io/whatwg/builder.d.ts","./node_modules/apache-arrow/builder.d.ts","./node_modules/apache-arrow/builder/bool.d.ts","./node_modules/apache-arrow/builder/null.d.ts","./node_modules/apache-arrow/builder/date.d.ts","./node_modules/apache-arrow/builder/decimal.d.ts","./node_modules/apache-arrow/builder/int.d.ts","./node_modules/apache-arrow/builder/dictionary.d.ts","./node_modules/apache-arrow/builder/fixedsizebinary.d.ts","./node_modules/apache-arrow/builder/float.d.ts","./node_modules/apache-arrow/builder/time.d.ts","./node_modules/apache-arrow/builder/timestamp.d.ts","./node_modules/apache-arrow/builder/interval.d.ts","./node_modules/apache-arrow/builder/duration.d.ts","./node_modules/apache-arrow/builder/utf8.d.ts","./node_modules/apache-arrow/builder/largeutf8.d.ts","./node_modules/apache-arrow/builder/binary.d.ts","./node_modules/apache-arrow/builder/largebinary.d.ts","./node_modules/apache-arrow/builder/list.d.ts","./node_modules/apache-arrow/builder/fixedsizelist.d.ts","./node_modules/apache-arrow/builder/map.d.ts","./node_modules/apache-arrow/builder/struct.d.ts","./node_modules/apache-arrow/builder/union.d.ts","./node_modules/apache-arrow/interfaces.d.ts","./node_modules/apache-arrow/type.d.ts","./node_modules/apache-arrow/vector.d.ts","./node_modules/apache-arrow/data.d.ts","./node_modules/apache-arrow/recordbatch.d.ts","./node_modules/apache-arrow/table.d.ts","./node_modules/apache-arrow/visitor.d.ts","./node_modules/apache-arrow/factories.d.ts","./node_modules/apache-arrow/io/interfaces.d.ts","./node_modules/apache-arrow/util/buffer.d.ts","./node_modules/apache-arrow/io/stream.d.ts","./node_modules/apache-arrow/fb/block.d.ts","./node_modules/apache-arrow/ipc/metadata/file.d.ts","./node_modules/apache-arrow/ipc/metadata/json.d.ts","./node_modules/apache-arrow/ipc/metadata/message.d.ts","./node_modules/apache-arrow/io/file.d.ts","./node_modules/apache-arrow/ipc/message.d.ts","./node_modules/apache-arrow/ipc/reader.d.ts","./node_modules/apache-arrow/ipc/writer.d.ts","./node_modules/apache-arrow/ipc/serialization.d.ts","./node_modules/apache-arrow/ipc/compression/registry.d.ts","./node_modules/apache-arrow/util/bn.d.ts","./node_modules/apache-arrow/util/int.d.ts","./node_modules/apache-arrow/util/bit.d.ts","./node_modules/apache-arrow/util/interval.d.ts","./node_modules/apache-arrow/visitor/typecomparator.d.ts","./node_modules/apache-arrow/arrow.d.ts","./node_modules/apache-arrow/arrow.dom.d.ts","./node_modules/apache-arrow/arrow.node.d.ts","./node_modules/@elastic/elasticsearch/lib/api/types.d.ts","./node_modules/@elastic/elasticsearch/lib/helpers.d.ts","./node_modules/@elastic/elasticsearch/lib/api/typeswithbodykey.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/async_search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/autoscaling.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/bulk.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/capabilities.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/cat.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ccr.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/clear_scroll.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/close_point_in_time.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/cluster.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/connector.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/count.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/create.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/dangling_indices.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_by_query.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_by_query_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/enrich.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/eql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/esql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/exists.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/exists_source.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/explain.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/features.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/field_caps.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/fleet.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script_context.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script_languages.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_source.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/graph.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/health_report.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ilm.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/index.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/indices.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/inference.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/info.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ingest.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/knn_search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/license.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/logstash.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/mget.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/migration.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ml.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/monitoring.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/msearch.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/msearch_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/mtermvectors.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/nodes.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/open_point_in_time.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ping.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/profiling.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/put_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/query_rules.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/rank_eval.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/reindex.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/reindex_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/render_search_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/rollup.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/scripts_painless_execute.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/scroll.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_application.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_mvt.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_shards.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/searchable_snapshots.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/security.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/shutdown.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/simulate.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/slm.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/snapshot.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/sql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ssl.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/streams.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/synonyms.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/tasks.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/terms_enum.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/termvectors.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/text_structure.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/transform.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update_by_query.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update_by_query_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/watcher.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/xpack.d.ts","./node_modules/@elastic/elasticsearch/lib/api/index.d.ts","./node_modules/@elastic/elasticsearch/lib/client.d.ts","./node_modules/@elastic/elasticsearch/index.d.ts","./node_modules/@nestjs/elasticsearch/dist/interfaces/elasticsearch-module-options.interface.d.ts","./node_modules/@nestjs/elasticsearch/dist/elasticsearch.module.d.ts","./node_modules/@nestjs/elasticsearch/dist/elasticsearch.service.d.ts","./node_modules/@nestjs/elasticsearch/dist/interfaces/index.d.ts","./node_modules/@nestjs/elasticsearch/dist/index.d.ts","./node_modules/@nestjs/elasticsearch/index.d.ts","./src/search/autocomplete/autocomplete.service.ts","./src/search/filters/search-filters.service.ts","./src/search/search.service.ts","./src/search/search.controller.ts","./src/search/indexing/indexing.service.ts","./src/config/elasticsearch.config.ts","./src/search/search.module.ts","./src/email-marketing/entities/email-template.entity.ts","./src/email-marketing/entities/ab-test-variant.entity.ts","./src/email-marketing/enums/ab-test-status.enum.ts","./src/email-marketing/entities/ab-test.entity.ts","./src/email-marketing/enums/recipient-status.enum.ts","./src/email-marketing/entities/campaign-recipient.entity.ts","./src/email-marketing/enums/campaign-status.enum.ts","./src/email-marketing/entities/campaign.entity.ts","./src/email-marketing/enums/segment-rule-operator.enum.ts","./src/email-marketing/enums/segment-rule-field.enum.ts","./src/email-marketing/entities/segment-rule.entity.ts","./src/email-marketing/entities/segment.entity.ts","./src/email-marketing/dto/create-segment.dto.ts","./src/email-marketing/dto/update-segment.dto.ts","./src/email-marketing/segmentation/segmentation.service.ts","./src/email-marketing/dto/create-template.dto.ts","./src/email-marketing/dto/update-template.dto.ts","./src/email-marketing/templates/template-management.service.ts","./src/email-marketing/enums/email-event-type.enum.ts","./src/email-marketing/entities/email-event.entity.ts","./src/email-marketing/dto/create-ab-test.dto.ts","./src/email-marketing/ab-testing/ab-testing.service.ts","./src/email-marketing/analytics/email-analytics.service.ts","./src/email-marketing/dto/create-campaign.dto.ts","./src/email-marketing/dto/update-campaign.dto.ts","./src/email-marketing/dto/schedule-campaign.dto.ts","./src/email-marketing/email-marketing.service.ts","./src/email-marketing/enums/trigger-type.enum.ts","./src/email-marketing/entities/automation-trigger.entity.ts","./src/email-marketing/enums/action-type.enum.ts","./src/email-marketing/entities/automation-action.entity.ts","./src/email-marketing/enums/workflow-status.enum.ts","./src/email-marketing/entities/automation-workflow.entity.ts","./src/email-marketing/dto/create-automation.dto.ts","./src/email-marketing/dto/update-automation.dto.ts","./src/email-marketing/automation/automation.service.ts","./src/email-marketing/sender/email-sender.service.ts","./src/email-marketing/email-marketing.controller.ts","./src/email-marketing/templates/template.controller.ts","./src/email-marketing/automation/automation.controller.ts","./src/email-marketing/dto/add-segment-members.dto.ts","./src/email-marketing/segmentation/segment.controller.ts","./src/email-marketing/analytics/email-analytics.controller.ts","./src/email-marketing/ab-testing/ab-testing.controller.ts","./src/email-marketing/tracking/tracking.controller.ts","./src/email-marketing/entities/email-subscription.entity.ts","./src/email-marketing/processors/email-queue.processor.ts","./src/email-marketing/email-marketing.module.ts","./src/gamification/entities/point-transaction.entity.ts","./src/gamification/entities/user-progress.entity.ts","./src/gamification/entities/badge.entity.ts","./src/gamification/entities/user-badge.entity.ts","./src/gamification/entities/challenge.entity.ts","./src/gamification/entities/user-challenge.entity.ts","./src/gamification/points/points.service.ts","./src/gamification/badges/badges.service.ts","./src/gamification/leaderboards/leaderboards.service.ts","./src/gamification/challenges/challenges.service.ts","./src/gamification/gamification.service.ts","./src/gamification/gamification.controller.ts","./src/gamification/gamification.module.ts","./src/learning-paths/services/skill-assessment.service.ts","./src/learning-paths/services/path-generation.service.ts","./src/learning-paths/services/milestone-tracking.service.ts","./src/learning-paths/learning-paths.service.ts","./src/learning-paths/learning-paths.controller.ts","./src/learning-paths/learning-paths.module.ts","./node_modules/@huggingface/tasks/dist/commonjs/pipelines.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/audio-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/automatic-speech-recognition/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/chat-completion/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/document-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/feature-extraction/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/fill-mask/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-text/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-segmentation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-text-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-text-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/object-detection/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/depth-estimation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/sentence-similarity/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/summarization/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/table-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-speech/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/token-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/translation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-generation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/video-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/visual-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-image-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-object-detection/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/index.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/widget-example.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tokenizer-data.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-data.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-libraries-downloads.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-libraries.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/library-to-tasks.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/default-widget-inputs.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/gguf.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/common.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/types.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/inputs.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/index.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/hardware.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/local-apps.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/dataset-libraries.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/inference-providers.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/eval.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/types.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/custom/request.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/custom/streamingrequest.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/utils.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/audioclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/audiotoaudio.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/automaticspeechrecognition.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/texttospeech.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/utils.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagesegmentation.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagetoimage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagetotext.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagetovideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagetexttoimage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imagetexttovideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/objectdetection.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/texttoimage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/texttovideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/zeroshotimageclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/chatcompletion.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/chatcompletionstream.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/featureextraction.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/fillmask.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/questionanswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/sentencesimilarity.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/summarization.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/tablequestionanswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textgeneration.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textgenerationstream.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/tokenclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/translation.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/zeroshotclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/multimodal/documentquestionanswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/multimodal/visualquestionanswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/tabular/tabularclassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/tabular/tabularregression.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/inferenceclient.d.ts","./node_modules/@huggingface/inference/dist/commonjs/vendor/type-fest/basic.d.ts","./node_modules/@huggingface/inference/dist/commonjs/errors.d.ts","./node_modules/@huggingface/inference/dist/commonjs/snippets/getinferencesnippets.d.ts","./node_modules/@huggingface/inference/dist/commonjs/snippets/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/providers/providerhelper.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/getproviderhelper.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/makerequestoptions.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/logger.d.ts","./node_modules/@huggingface/inference/dist/commonjs/index.d.ts","./src/moderation/auto/auto-moderation.service.ts","./src/moderation/manual/review-item.entity.ts","./src/moderation/manual/manual-review.service.ts","./src/moderation/safety/content-safety.service.ts","./src/moderation/analytics/moderation-event.entity.ts","./src/moderation/analytics/moderation-analytics.service.ts","./src/moderation/moderation.service.ts","./src/moderation/moderation.module.ts","./node_modules/@nestjs/axios/dist/interfaces/http-module.interface.d.ts","./node_modules/@nestjs/axios/dist/interfaces/index.d.ts","./node_modules/@nestjs/axios/dist/http.module.d.ts","./node_modules/@nestjs/axios/dist/http.service.d.ts","./node_modules/@nestjs/axios/dist/index.d.ts","./node_modules/@nestjs/axios/index.d.ts","./src/orchestration/discovery/service-discovery.service.ts","./src/common/utils/correlation.utils.ts","./src/orchestration/service-mesh/service-mesh.service.ts","./src/orchestration/workflow/workflow-engine.service.ts","./src/orchestration/locks/distributed-lock.service.ts","./src/orchestration/health/health-checker.service.ts","./src/orchestration/orchestration.module.ts","./src/security/security.service.ts","./src/security/encryption/encryption.service.ts","./src/security/threats/threat-detection.service.ts","./src/security/compliance/compliance.service.ts","./src/security/audit/audit-logging.service.ts","./src/security/security.module.ts","./src/tenancy/entities/tenant.entity.ts","./src/tenancy/entities/tenant-config.entity.ts","./src/tenancy/entities/tenant-billing.entity.ts","./src/tenancy/entities/tenant-customization.entity.ts","./src/tenancy/dto/tenant.dto.ts","./src/tenancy/billing/tenant-billing.service.ts","./src/tenancy/customization/customization.service.ts","./src/tenancy/tenancy.service.ts","./src/tenancy/admin/tenant-admin.service.ts","./src/tenancy/tenancy.controller.ts","./src/tenancy/isolation/isolation.service.ts","./src/tenancy/decorators/requires-tenant.decorator.ts","./src/tenancy/guards/tenant.guard.ts","./src/tenancy/tenancy.module.ts","./src/cdn/optimization/asset-optimization.service.ts","./src/cdn/providers/cloudflare.service.ts","./src/cdn/caching/edge-caching.service.ts","./src/cdn/geo/geo-location.service.ts","./src/cdn/cdn.service.ts","./src/cdn/dto/upload-content.dto.ts","./src/cdn/cdn.controller.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/auth/httpauthschemeprovider.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/associatealiascommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/associatedistributiontenantwebaclcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/associatedistributionwebaclcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/copydistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createanycastiplistcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createcachepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createcloudfrontoriginaccessidentitycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createconnectiongroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createcontinuousdeploymentpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createdistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createdistributionwithtagscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createfieldlevelencryptionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createfieldlevelencryptionprofilecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createinvalidationcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createinvalidationfordistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createkeygroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createkeyvaluestorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createmonitoringsubscriptioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createoriginaccesscontrolcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createoriginrequestpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createpublickeycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createrealtimelogconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createresponseheaderspolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createstreamingdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createstreamingdistributionwithtagscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createtruststorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/createvpcorigincommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteanycastiplistcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletecachepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletecloudfrontoriginaccessidentitycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteconnectiongroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletecontinuousdeploymentpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletedistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletedistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletefieldlevelencryptionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletefieldlevelencryptionprofilecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletefunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletekeygroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletekeyvaluestorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletemonitoringsubscriptioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteoriginaccesscontrolcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteoriginrequestpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletepublickeycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleterealtimelogconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteresourcepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deleteresponseheaderspolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletestreamingdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletetruststorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/deletevpcorigincommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/describeconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/describefunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/describekeyvaluestorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/disassociatedistributiontenantwebaclcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/disassociatedistributionwebaclcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getanycastiplistcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcachepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcachepolicyconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcloudfrontoriginaccessidentitycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcloudfrontoriginaccessidentityconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getconnectiongroupbyroutingendpointcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getconnectiongroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcontinuousdeploymentpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getcontinuousdeploymentpolicyconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getdistributionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getdistributiontenantbydomaincommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getdistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getfieldlevelencryptioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getfieldlevelencryptionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getfieldlevelencryptionprofilecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getfieldlevelencryptionprofileconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getinvalidationcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getinvalidationfordistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getkeygroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getkeygroupconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/models_1.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getmanagedcertificatedetailscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getmonitoringsubscriptioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getoriginaccesscontrolcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getoriginaccesscontrolconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getoriginrequestpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getoriginrequestpolicyconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getpublickeycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getpublickeyconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getrealtimelogconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getresourcepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getresponseheaderspolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getresponseheaderspolicyconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getstreamingdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getstreamingdistributionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/gettruststorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/getvpcorigincommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listanycastiplistscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listcachepoliciescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listcloudfrontoriginaccessidentitiescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listconflictingaliasescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listconnectionfunctionscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listconnectiongroupscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listcontinuousdeploymentpoliciescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyanycastiplistidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbycachepolicyidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyconnectionmodecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbykeygroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyoriginrequestpolicyidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyownedresourcecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyrealtimelogconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyresponseheaderspolicyidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbytruststorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbyvpcoriginidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionsbywebaclidcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributionscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributiontenantsbycustomizationcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdistributiontenantscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listdomainconflictscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listfieldlevelencryptionconfigscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listfieldlevelencryptionprofilescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listfunctionscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listinvalidationscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listinvalidationsfordistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listkeygroupscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listkeyvaluestorescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listoriginaccesscontrolscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listoriginrequestpoliciescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listpublickeyscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listrealtimelogconfigscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listresponseheaderspoliciescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/liststreamingdistributionscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listtagsforresourcecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listtruststorescommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/listvpcoriginscommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/publishconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/publishfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/putresourcepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/tagresourcecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/testconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/testfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/untagresourcecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateanycastiplistcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatecachepolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatecloudfrontoriginaccessidentitycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateconnectionfunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateconnectiongroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatecontinuousdeploymentpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatedistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatedistributiontenantcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatedistributionwithstagingconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatedomainassociationcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatefieldlevelencryptionconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatefieldlevelencryptionprofilecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatefunctioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatekeygroupcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatekeyvaluestorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateoriginaccesscontrolcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateoriginrequestpolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatepublickeycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updaterealtimelogconfigcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updateresponseheaderspolicycommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatestreamingdistributioncommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatetruststorecommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/updatevpcorigincommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/verifydnsconfigurationcommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/endpoint/endpointparameters.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/auth/httpauthextensionconfiguration.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/extensionconfiguration.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/runtimeextensions.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/cloudfrontclient.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/cloudfront.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/interfaces.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listcloudfrontoriginaccessidentitiespaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listconnectionfunctionspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listconnectiongroupspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributionspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributionsbyconnectionfunctionpaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributionsbyconnectionmodepaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributionsbytruststorepaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributiontenantspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdistributiontenantsbycustomizationpaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listdomainconflictspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listinvalidationspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listinvalidationsfordistributiontenantpaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listkeyvaluestorespaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listoriginaccesscontrolspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listpublickeyspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/liststreamingdistributionspaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/listtruststorespaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitfordistributiondeployed.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitforinvalidationcompleted.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitforinvalidationfordistributiontenantcompleted.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitforstreamingdistributiondeployed.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/cloudfrontserviceexception.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/index.d.ts","./src/cdn/providers/aws-cloudfront.service.ts","./src/cdn/cdn.module.ts","./src/payments/entities/payment.entity.ts","./src/payments/entities/subscription.entity.ts","./src/payments/dto/create-payment.dto.ts","./src/payments/entities/refund.entity.ts","./src/payments/entities/invoice.entity.ts","./src/payments/dto/refund.dto.ts","./src/payments/dto/create-subscription.dto.ts","./node_modules/stripe/types/lib.d.ts","./node_modules/stripe/types/crypto/crypto.d.ts","./node_modules/stripe/types/net/net.d.ts","./node_modules/stripe/types/shared.d.ts","./node_modules/stripe/types/errors.d.ts","./node_modules/stripe/types/oauth.d.ts","./node_modules/stripe/types/webhooks.d.ts","./node_modules/stripe/types/eventtypes.d.ts","./node_modules/stripe/types/upcominginvoices.d.ts","./node_modules/stripe/types/thinevent.d.ts","./node_modules/stripe/types/accountlinksresource.d.ts","./node_modules/stripe/types/accountsessionsresource.d.ts","./node_modules/stripe/types/accountsresource.d.ts","./node_modules/stripe/types/applepaydomainsresource.d.ts","./node_modules/stripe/types/applicationfeesresource.d.ts","./node_modules/stripe/types/apps/secretsresource.d.ts","./node_modules/stripe/types/balanceresource.d.ts","./node_modules/stripe/types/balancetransactionsresource.d.ts","./node_modules/stripe/types/billing/alertsresource.d.ts","./node_modules/stripe/types/billing/creditbalancesummaryresource.d.ts","./node_modules/stripe/types/billing/creditbalancetransactionsresource.d.ts","./node_modules/stripe/types/billing/creditgrantsresource.d.ts","./node_modules/stripe/types/billing/metereventadjustmentsresource.d.ts","./node_modules/stripe/types/billing/metereventsresource.d.ts","./node_modules/stripe/types/billing/metersresource.d.ts","./node_modules/stripe/types/billingportal/configurationsresource.d.ts","./node_modules/stripe/types/billingportal/sessionsresource.d.ts","./node_modules/stripe/types/chargesresource.d.ts","./node_modules/stripe/types/checkout/sessionsresource.d.ts","./node_modules/stripe/types/climate/ordersresource.d.ts","./node_modules/stripe/types/climate/productsresource.d.ts","./node_modules/stripe/types/climate/suppliersresource.d.ts","./node_modules/stripe/types/confirmationtokensresource.d.ts","./node_modules/stripe/types/countryspecsresource.d.ts","./node_modules/stripe/types/couponsresource.d.ts","./node_modules/stripe/types/creditnotesresource.d.ts","./node_modules/stripe/types/customersessionsresource.d.ts","./node_modules/stripe/types/customersresource.d.ts","./node_modules/stripe/types/disputesresource.d.ts","./node_modules/stripe/types/entitlements/activeentitlementsresource.d.ts","./node_modules/stripe/types/entitlements/featuresresource.d.ts","./node_modules/stripe/types/ephemeralkeysresource.d.ts","./node_modules/stripe/types/eventsresource.d.ts","./node_modules/stripe/types/exchangeratesresource.d.ts","./node_modules/stripe/types/filelinksresource.d.ts","./node_modules/stripe/types/filesresource.d.ts","./node_modules/stripe/types/financialconnections/accountsresource.d.ts","./node_modules/stripe/types/financialconnections/sessionsresource.d.ts","./node_modules/stripe/types/financialconnections/transactionsresource.d.ts","./node_modules/stripe/types/forwarding/requestsresource.d.ts","./node_modules/stripe/types/identity/verificationreportsresource.d.ts","./node_modules/stripe/types/identity/verificationsessionsresource.d.ts","./node_modules/stripe/types/invoiceitemsresource.d.ts","./node_modules/stripe/types/invoicepaymentsresource.d.ts","./node_modules/stripe/types/invoicerenderingtemplatesresource.d.ts","./node_modules/stripe/types/invoicesresource.d.ts","./node_modules/stripe/types/issuing/authorizationsresource.d.ts","./node_modules/stripe/types/issuing/cardholdersresource.d.ts","./node_modules/stripe/types/issuing/cardsresource.d.ts","./node_modules/stripe/types/issuing/disputesresource.d.ts","./node_modules/stripe/types/issuing/personalizationdesignsresource.d.ts","./node_modules/stripe/types/issuing/physicalbundlesresource.d.ts","./node_modules/stripe/types/issuing/tokensresource.d.ts","./node_modules/stripe/types/issuing/transactionsresource.d.ts","./node_modules/stripe/types/mandatesresource.d.ts","./node_modules/stripe/types/paymentintentsresource.d.ts","./node_modules/stripe/types/paymentlinksresource.d.ts","./node_modules/stripe/types/paymentmethodconfigurationsresource.d.ts","./node_modules/stripe/types/paymentmethoddomainsresource.d.ts","./node_modules/stripe/types/paymentmethodsresource.d.ts","./node_modules/stripe/types/payoutsresource.d.ts","./node_modules/stripe/types/plansresource.d.ts","./node_modules/stripe/types/pricesresource.d.ts","./node_modules/stripe/types/productsresource.d.ts","./node_modules/stripe/types/promotioncodesresource.d.ts","./node_modules/stripe/types/quotesresource.d.ts","./node_modules/stripe/types/radar/earlyfraudwarningsresource.d.ts","./node_modules/stripe/types/radar/valuelistitemsresource.d.ts","./node_modules/stripe/types/radar/valuelistsresource.d.ts","./node_modules/stripe/types/refundsresource.d.ts","./node_modules/stripe/types/reporting/reportrunsresource.d.ts","./node_modules/stripe/types/reporting/reporttypesresource.d.ts","./node_modules/stripe/types/reviewsresource.d.ts","./node_modules/stripe/types/setupattemptsresource.d.ts","./node_modules/stripe/types/setupintentsresource.d.ts","./node_modules/stripe/types/shippingratesresource.d.ts","./node_modules/stripe/types/sigma/scheduledqueryrunsresource.d.ts","./node_modules/stripe/types/sourcesresource.d.ts","./node_modules/stripe/types/subscriptionitemsresource.d.ts","./node_modules/stripe/types/subscriptionschedulesresource.d.ts","./node_modules/stripe/types/subscriptionsresource.d.ts","./node_modules/stripe/types/tax/calculationsresource.d.ts","./node_modules/stripe/types/tax/registrationsresource.d.ts","./node_modules/stripe/types/tax/settingsresource.d.ts","./node_modules/stripe/types/tax/transactionsresource.d.ts","./node_modules/stripe/types/taxcodesresource.d.ts","./node_modules/stripe/types/taxidsresource.d.ts","./node_modules/stripe/types/taxratesresource.d.ts","./node_modules/stripe/types/terminal/configurationsresource.d.ts","./node_modules/stripe/types/terminal/connectiontokensresource.d.ts","./node_modules/stripe/types/terminal/locationsresource.d.ts","./node_modules/stripe/types/terminal/readersresource.d.ts","./node_modules/stripe/types/testhelpers/confirmationtokensresource.d.ts","./node_modules/stripe/types/testhelpers/customersresource.d.ts","./node_modules/stripe/types/testhelpers/issuing/authorizationsresource.d.ts","./node_modules/stripe/types/testhelpers/issuing/cardsresource.d.ts","./node_modules/stripe/types/testhelpers/issuing/personalizationdesignsresource.d.ts","./node_modules/stripe/types/testhelpers/issuing/transactionsresource.d.ts","./node_modules/stripe/types/testhelpers/refundsresource.d.ts","./node_modules/stripe/types/testhelpers/terminal/readersresource.d.ts","./node_modules/stripe/types/testhelpers/testclocksresource.d.ts","./node_modules/stripe/types/testhelpers/treasury/inboundtransfersresource.d.ts","./node_modules/stripe/types/testhelpers/treasury/outboundpaymentsresource.d.ts","./node_modules/stripe/types/testhelpers/treasury/outboundtransfersresource.d.ts","./node_modules/stripe/types/testhelpers/treasury/receivedcreditsresource.d.ts","./node_modules/stripe/types/testhelpers/treasury/receiveddebitsresource.d.ts","./node_modules/stripe/types/tokensresource.d.ts","./node_modules/stripe/types/topupsresource.d.ts","./node_modules/stripe/types/transfersresource.d.ts","./node_modules/stripe/types/treasury/creditreversalsresource.d.ts","./node_modules/stripe/types/treasury/debitreversalsresource.d.ts","./node_modules/stripe/types/treasury/financialaccountsresource.d.ts","./node_modules/stripe/types/treasury/inboundtransfersresource.d.ts","./node_modules/stripe/types/treasury/outboundpaymentsresource.d.ts","./node_modules/stripe/types/treasury/outboundtransfersresource.d.ts","./node_modules/stripe/types/treasury/receivedcreditsresource.d.ts","./node_modules/stripe/types/treasury/receiveddebitsresource.d.ts","./node_modules/stripe/types/treasury/transactionentriesresource.d.ts","./node_modules/stripe/types/treasury/transactionsresource.d.ts","./node_modules/stripe/types/v2/billing/metereventadjustmentsresource.d.ts","./node_modules/stripe/types/v2/billing/metereventsessionresource.d.ts","./node_modules/stripe/types/v2/billing/metereventstreamresource.d.ts","./node_modules/stripe/types/v2/billing/metereventsresource.d.ts","./node_modules/stripe/types/v2/core/eventdestinationsresource.d.ts","./node_modules/stripe/types/v2/eventtypes.d.ts","./node_modules/stripe/types/v2/core/eventsresource.d.ts","./node_modules/stripe/types/webhookendpointsresource.d.ts","./node_modules/stripe/types/accountlinks.d.ts","./node_modules/stripe/types/accountsessions.d.ts","./node_modules/stripe/types/accounts.d.ts","./node_modules/stripe/types/applepaydomains.d.ts","./node_modules/stripe/types/applicationfees.d.ts","./node_modules/stripe/types/applications.d.ts","./node_modules/stripe/types/apps/secrets.d.ts","./node_modules/stripe/types/balance.d.ts","./node_modules/stripe/types/balancetransactionsources.d.ts","./node_modules/stripe/types/balancetransactions.d.ts","./node_modules/stripe/types/bankaccounts.d.ts","./node_modules/stripe/types/billing/alerttriggereds.d.ts","./node_modules/stripe/types/billing/alerts.d.ts","./node_modules/stripe/types/billing/creditbalancesummary.d.ts","./node_modules/stripe/types/billing/creditbalancetransactions.d.ts","./node_modules/stripe/types/billing/creditgrants.d.ts","./node_modules/stripe/types/billing/metereventadjustments.d.ts","./node_modules/stripe/types/billing/metereventsummaries.d.ts","./node_modules/stripe/types/billing/meterevents.d.ts","./node_modules/stripe/types/billing/meters.d.ts","./node_modules/stripe/types/billingportal/configurations.d.ts","./node_modules/stripe/types/billingportal/sessions.d.ts","./node_modules/stripe/types/capabilities.d.ts","./node_modules/stripe/types/cards.d.ts","./node_modules/stripe/types/cashbalances.d.ts","./node_modules/stripe/types/charges.d.ts","./node_modules/stripe/types/checkout/sessions.d.ts","./node_modules/stripe/types/climate/orders.d.ts","./node_modules/stripe/types/climate/products.d.ts","./node_modules/stripe/types/climate/suppliers.d.ts","./node_modules/stripe/types/confirmationtokens.d.ts","./node_modules/stripe/types/connectcollectiontransfers.d.ts","./node_modules/stripe/types/countryspecs.d.ts","./node_modules/stripe/types/coupons.d.ts","./node_modules/stripe/types/creditnotelineitems.d.ts","./node_modules/stripe/types/creditnotes.d.ts","./node_modules/stripe/types/customerbalancetransactions.d.ts","./node_modules/stripe/types/customercashbalancetransactions.d.ts","./node_modules/stripe/types/customersessions.d.ts","./node_modules/stripe/types/customersources.d.ts","./node_modules/stripe/types/customers.d.ts","./node_modules/stripe/types/discounts.d.ts","./node_modules/stripe/types/disputes.d.ts","./node_modules/stripe/types/entitlements/activeentitlementsummaries.d.ts","./node_modules/stripe/types/entitlements/activeentitlements.d.ts","./node_modules/stripe/types/entitlements/features.d.ts","./node_modules/stripe/types/ephemeralkeys.d.ts","./node_modules/stripe/types/events.d.ts","./node_modules/stripe/types/exchangerates.d.ts","./node_modules/stripe/types/externalaccounts.d.ts","./node_modules/stripe/types/feerefunds.d.ts","./node_modules/stripe/types/filelinks.d.ts","./node_modules/stripe/types/files.d.ts","./node_modules/stripe/types/financialconnections/accountowners.d.ts","./node_modules/stripe/types/financialconnections/accountownerships.d.ts","./node_modules/stripe/types/financialconnections/accounts.d.ts","./node_modules/stripe/types/financialconnections/sessions.d.ts","./node_modules/stripe/types/financialconnections/transactions.d.ts","./node_modules/stripe/types/forwarding/requests.d.ts","./node_modules/stripe/types/fundinginstructions.d.ts","./node_modules/stripe/types/identity/verificationreports.d.ts","./node_modules/stripe/types/identity/verificationsessions.d.ts","./node_modules/stripe/types/invoiceitems.d.ts","./node_modules/stripe/types/invoicelineitems.d.ts","./node_modules/stripe/types/invoicepayments.d.ts","./node_modules/stripe/types/invoicerenderingtemplates.d.ts","./node_modules/stripe/types/invoices.d.ts","./node_modules/stripe/types/issuing/authorizations.d.ts","./node_modules/stripe/types/issuing/cardholders.d.ts","./node_modules/stripe/types/issuing/cards.d.ts","./node_modules/stripe/types/issuing/disputes.d.ts","./node_modules/stripe/types/issuing/personalizationdesigns.d.ts","./node_modules/stripe/types/issuing/physicalbundles.d.ts","./node_modules/stripe/types/issuing/tokens.d.ts","./node_modules/stripe/types/issuing/transactions.d.ts","./node_modules/stripe/types/lineitems.d.ts","./node_modules/stripe/types/loginlinks.d.ts","./node_modules/stripe/types/mandates.d.ts","./node_modules/stripe/types/paymentintents.d.ts","./node_modules/stripe/types/paymentlinks.d.ts","./node_modules/stripe/types/paymentmethodconfigurations.d.ts","./node_modules/stripe/types/paymentmethoddomains.d.ts","./node_modules/stripe/types/paymentmethods.d.ts","./node_modules/stripe/types/payouts.d.ts","./node_modules/stripe/types/persons.d.ts","./node_modules/stripe/types/plans.d.ts","./node_modules/stripe/types/prices.d.ts","./node_modules/stripe/types/productfeatures.d.ts","./node_modules/stripe/types/products.d.ts","./node_modules/stripe/types/promotioncodes.d.ts","./node_modules/stripe/types/quotes.d.ts","./node_modules/stripe/types/radar/earlyfraudwarnings.d.ts","./node_modules/stripe/types/radar/valuelistitems.d.ts","./node_modules/stripe/types/radar/valuelists.d.ts","./node_modules/stripe/types/refunds.d.ts","./node_modules/stripe/types/reporting/reportruns.d.ts","./node_modules/stripe/types/reporting/reporttypes.d.ts","./node_modules/stripe/types/reservetransactions.d.ts","./node_modules/stripe/types/reviews.d.ts","./node_modules/stripe/types/setupattempts.d.ts","./node_modules/stripe/types/setupintents.d.ts","./node_modules/stripe/types/shippingrates.d.ts","./node_modules/stripe/types/sigma/scheduledqueryruns.d.ts","./node_modules/stripe/types/sourcemandatenotifications.d.ts","./node_modules/stripe/types/sourcetransactions.d.ts","./node_modules/stripe/types/sources.d.ts","./node_modules/stripe/types/subscriptionitems.d.ts","./node_modules/stripe/types/subscriptionschedules.d.ts","./node_modules/stripe/types/subscriptions.d.ts","./node_modules/stripe/types/tax/calculationlineitems.d.ts","./node_modules/stripe/types/tax/calculations.d.ts","./node_modules/stripe/types/tax/registrations.d.ts","./node_modules/stripe/types/tax/settings.d.ts","./node_modules/stripe/types/tax/transactionlineitems.d.ts","./node_modules/stripe/types/tax/transactions.d.ts","./node_modules/stripe/types/taxcodes.d.ts","./node_modules/stripe/types/taxdeductedatsources.d.ts","./node_modules/stripe/types/taxids.d.ts","./node_modules/stripe/types/taxrates.d.ts","./node_modules/stripe/types/terminal/configurations.d.ts","./node_modules/stripe/types/terminal/connectiontokens.d.ts","./node_modules/stripe/types/terminal/locations.d.ts","./node_modules/stripe/types/terminal/readers.d.ts","./node_modules/stripe/types/testhelpers/testclocks.d.ts","./node_modules/stripe/types/tokens.d.ts","./node_modules/stripe/types/topups.d.ts","./node_modules/stripe/types/transferreversals.d.ts","./node_modules/stripe/types/transfers.d.ts","./node_modules/stripe/types/treasury/creditreversals.d.ts","./node_modules/stripe/types/treasury/debitreversals.d.ts","./node_modules/stripe/types/treasury/financialaccountfeatures.d.ts","./node_modules/stripe/types/treasury/financialaccounts.d.ts","./node_modules/stripe/types/treasury/inboundtransfers.d.ts","./node_modules/stripe/types/treasury/outboundpayments.d.ts","./node_modules/stripe/types/treasury/outboundtransfers.d.ts","./node_modules/stripe/types/treasury/receivedcredits.d.ts","./node_modules/stripe/types/treasury/receiveddebits.d.ts","./node_modules/stripe/types/treasury/transactionentries.d.ts","./node_modules/stripe/types/treasury/transactions.d.ts","./node_modules/stripe/types/v2/billing/metereventadjustments.d.ts","./node_modules/stripe/types/v2/billing/metereventsessions.d.ts","./node_modules/stripe/types/v2/billing/meterevents.d.ts","./node_modules/stripe/types/v2/eventdestinations.d.ts","./node_modules/stripe/types/v2/events.d.ts","./node_modules/stripe/types/webhookendpoints.d.ts","./node_modules/stripe/types/index.d.ts","./src/payments/providers/stripe.service.ts","./src/payments/providers/provider-factory.service.ts","./src/payments/interfaces/payment-provider.interface.ts","./src/payments/payments.service.ts","./src/payments/payments.controller.ts","./src/payments/webhooks/entities/webhook-retry.entity.ts","./src/payments/webhooks/webhook-queue.service.ts","./src/payments/webhooks/webhook.service.ts","./src/payments/webhooks/webhook.controller.ts","./src/payments/webhooks/webhook-management.controller.ts","./src/payments/webhooks/webhook-retry.processor.ts","./src/payments/subscriptions/subscriptions.service.ts","./src/payments/subscriptions/subscription-job.processor.ts","./src/common/database/transaction-helper.service.ts","./src/payments/payments.module.ts","./src/localization/entities/translation.entity.ts","./src/localization/language-detection.service.ts","./src/common/types/request-with-locale.ts","./src/localization/language.middleware.ts","./src/localization/dto/bundle-query.dto.ts","./src/localization/dto/create-translation.dto.ts","./src/localization/dto/export-query.dto.ts","./src/localization/dto/import-translations.dto.ts","./src/localization/dto/list-translations-query.dto.ts","./src/localization/dto/update-translation.dto.ts","./src/localization/localization.constants.ts","./src/localization/localization.service.ts","./src/localization/localization.controller.ts","./src/localization/localization.module.ts","./src/common/csrf/csrf.service.ts","./src/common/middleware/csrf.middleware.ts","./src/common/csrf/csrf.controller.ts","./src/common/csrf/csrf.module.ts","./src/common/timeout/timeout.controller.ts","./src/common/examples/timeout-example.controller.ts","./src/common/timeout/timeout.module.ts","./src/app.module.ts","./node_modules/@types/express-session/index.d.ts","./node_modules/@redis/client/dist/lib/commands/generic-transformers.d.ts","./node_modules/@redis/client/dist/lib/client/parser.d.ts","./node_modules/@redis/client/dist/lib/errors.d.ts","./node_modules/@redis/client/dist/lib/lua-script.d.ts","./node_modules/@redis/client/dist/lib/resp/decoder.d.ts","./node_modules/@redis/client/dist/lib/resp/verbatim-string.d.ts","./node_modules/@redis/client/dist/lib/resp/types.d.ts","./node_modules/@redis/client/dist/lib/utils/digest.d.ts","./node_modules/@redis/client/dist/lib/commands/client_kill.d.ts","./node_modules/@redis/client/dist/lib/commands/cluster_failover.d.ts","./node_modules/@redis/client/dist/lib/commands/cluster_setslot.d.ts","./node_modules/@redis/client/dist/lib/commands/command_list.d.ts","./node_modules/@redis/client/dist/lib/commands/flushall.d.ts","./node_modules/@redis/client/dist/lib/commands/set.d.ts","./node_modules/@redis/client/dist/lib/commands/acl_log.d.ts","./node_modules/@redis/client/dist/lib/commands/auth.d.ts","./node_modules/@redis/client/dist/lib/commands/bgsave.d.ts","./node_modules/@redis/client/dist/lib/commands/bitcount.d.ts","./node_modules/@redis/client/dist/lib/commands/bitfield.d.ts","./node_modules/@redis/client/dist/lib/commands/bitfield_ro.d.ts","./node_modules/@redis/client/dist/lib/commands/bitop.d.ts","./node_modules/@redis/client/dist/lib/commands/lmpop.d.ts","./node_modules/@redis/client/dist/lib/commands/zmpop.d.ts","./node_modules/@redis/client/dist/lib/commands/client_info.d.ts","./node_modules/@redis/client/dist/lib/commands/client_list.d.ts","./node_modules/@redis/client/dist/lib/commands/client_tracking.d.ts","./node_modules/@redis/client/dist/lib/commands/cluster_reset.d.ts","./node_modules/@redis/client/dist/lib/commands/copy.d.ts","./node_modules/@redis/client/dist/lib/commands/eval.d.ts","./node_modules/@redis/client/dist/lib/commands/function_list.d.ts","./node_modules/@redis/client/dist/lib/commands/function_list_withcode.d.ts","./node_modules/@redis/client/dist/lib/commands/function_load.d.ts","./node_modules/@redis/client/dist/lib/commands/function_restore.d.ts","./node_modules/@redis/client/dist/lib/commands/geosearch.d.ts","./node_modules/@redis/client/dist/lib/commands/geoadd.d.ts","./node_modules/@redis/client/dist/lib/commands/geosearch_with.d.ts","./node_modules/@redis/client/dist/lib/commands/georadius_store.d.ts","./node_modules/@redis/client/dist/lib/commands/georadiusbymember_store.d.ts","./node_modules/@redis/client/dist/lib/commands/geosearchstore.d.ts","./node_modules/@redis/client/dist/lib/commands/getex.d.ts","./node_modules/@redis/client/dist/lib/commands/hello.d.ts","./node_modules/@redis/client/dist/lib/commands/hexpire.d.ts","./node_modules/@redis/client/dist/lib/commands/hgetex.d.ts","./node_modules/@redis/client/dist/lib/commands/hrandfield_count_withvalues.d.ts","./node_modules/@redis/client/dist/lib/commands/scan.d.ts","./node_modules/@redis/client/dist/lib/commands/hset.d.ts","./node_modules/@redis/client/dist/lib/commands/hsetex.d.ts","./node_modules/@redis/client/dist/lib/commands/hotkeys_get.d.ts","./node_modules/@redis/client/dist/lib/commands/hotkeys_start.d.ts","./node_modules/@redis/client/dist/lib/commands/latency_graph.d.ts","./node_modules/@redis/client/dist/lib/commands/latency_history.d.ts","./node_modules/@redis/client/dist/lib/commands/lcs_idx.d.ts","./node_modules/@redis/client/dist/lib/commands/lcs_idx_withmatchlen.d.ts","./node_modules/@redis/client/dist/lib/commands/lpos.d.ts","./node_modules/@redis/client/dist/lib/commands/memory_stats.d.ts","./node_modules/@redis/client/dist/lib/commands/memory_usage.d.ts","./node_modules/@redis/client/dist/lib/commands/migrate.d.ts","./node_modules/@redis/client/dist/lib/commands/module_list.d.ts","./node_modules/@redis/client/dist/lib/commands/mset.d.ts","./node_modules/@redis/client/dist/lib/commands/restore.d.ts","./node_modules/@redis/client/dist/lib/commands/sintercard.d.ts","./node_modules/@redis/client/dist/lib/commands/sort.d.ts","./node_modules/@redis/client/dist/lib/commands/common-stream.types.d.ts","./node_modules/@redis/client/dist/lib/commands/xadd.d.ts","./node_modules/@redis/client/dist/lib/commands/xautoclaim.d.ts","./node_modules/@redis/client/dist/lib/commands/xclaim.d.ts","./node_modules/@redis/client/dist/lib/commands/xcfgset.d.ts","./node_modules/@redis/client/dist/lib/commands/xgroup_create.d.ts","./node_modules/@redis/client/dist/lib/commands/xgroup_setid.d.ts","./node_modules/@redis/client/dist/lib/commands/xinfo_consumers.d.ts","./node_modules/@redis/client/dist/lib/commands/xinfo_groups.d.ts","./node_modules/@redis/client/dist/lib/commands/xinfo_stream.d.ts","./node_modules/@redis/client/dist/lib/commands/xpending_range.d.ts","./node_modules/@redis/client/dist/lib/commands/xrange.d.ts","./node_modules/@redis/client/dist/lib/commands/xread.d.ts","./node_modules/@redis/client/dist/lib/commands/xreadgroup.d.ts","./node_modules/@redis/client/dist/lib/commands/xsetid.d.ts","./node_modules/@redis/client/dist/lib/commands/xtrim.d.ts","./node_modules/@redis/client/dist/lib/commands/zadd_incr.d.ts","./node_modules/@redis/client/dist/lib/commands/zadd.d.ts","./node_modules/@redis/client/dist/lib/commands/zinter.d.ts","./node_modules/@redis/client/dist/lib/commands/zintercard.d.ts","./node_modules/@redis/client/dist/lib/commands/zrange.d.ts","./node_modules/@redis/client/dist/lib/commands/zrangebylex.d.ts","./node_modules/@redis/client/dist/lib/commands/zrangebyscore.d.ts","./node_modules/@redis/client/dist/lib/commands/zrangestore.d.ts","./node_modules/@redis/client/dist/lib/commands/zunion.d.ts","./node_modules/@redis/client/dist/lib/commands/zunionstore.d.ts","./node_modules/@redis/client/dist/lib/commands/vadd.d.ts","./node_modules/@redis/client/dist/lib/commands/vinfo.d.ts","./node_modules/@redis/client/dist/lib/commands/vsim.d.ts","./node_modules/@redis/client/dist/lib/commands/index.d.ts","./node_modules/@redis/client/dist/lib/client/socket.d.ts","./node_modules/@redis/client/dist/lib/authx/identity-provider.d.ts","./node_modules/@redis/client/dist/lib/authx/token.d.ts","./node_modules/@redis/client/dist/lib/authx/disposable.d.ts","./node_modules/@redis/client/dist/lib/authx/token-manager.d.ts","./node_modules/@redis/client/dist/lib/authx/credentials-provider.d.ts","./node_modules/@redis/client/dist/lib/authx/index.d.ts","./node_modules/@redis/client/dist/lib/client/pub-sub.d.ts","./node_modules/@redis/client/dist/lib/client/commands-queue.d.ts","./node_modules/@redis/client/dist/lib/multi-command.d.ts","./node_modules/@redis/client/dist/lib/client/multi-command.d.ts","./node_modules/@redis/client/dist/lib/client/legacy-mode.d.ts","./node_modules/@redis/client/dist/lib/client/cache.d.ts","./node_modules/@redis/client/dist/lib/cluster/cluster-slots.d.ts","./node_modules/@redis/client/dist/lib/cluster/multi-command.d.ts","./node_modules/@redis/client/dist/lib/cluster/index.d.ts","./node_modules/@redis/client/dist/lib/client/identity.d.ts","./node_modules/@redis/client/dist/lib/client/pool.d.ts","./node_modules/@redis/client/dist/lib/client/enterprise-maintenance-manager.d.ts","./node_modules/@redis/client/dist/lib/client/index.d.ts","./node_modules/@redis/client/dist/lib/sentinel/types.d.ts","./node_modules/@redis/client/dist/lib/sentinel/multi-commands.d.ts","./node_modules/@redis/client/dist/lib/sentinel/index.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/types.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/metrics.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/client-registry.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/index.d.ts","./node_modules/@redis/client/dist/lib/client/tracing.d.ts","./node_modules/@redis/client/dist/index.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/info.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/insert.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/reserve.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/incrby.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/info.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/merge.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/info.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/insert.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/reserve.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/create.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/info.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/merge.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/incrby.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/info.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/reserve.d.ts","./node_modules/@redis/bloom/dist/lib/commands/index.d.ts","./node_modules/@redis/bloom/dist/lib/index.d.ts","./node_modules/@redis/json/dist/lib/commands/arrindex.d.ts","./node_modules/@redis/json/dist/lib/commands/arrlen.d.ts","./node_modules/@redis/json/dist/lib/commands/arrpop.d.ts","./node_modules/@redis/json/dist/lib/commands/clear.d.ts","./node_modules/@redis/json/dist/lib/commands/debug_memory.d.ts","./node_modules/@redis/json/dist/lib/commands/del.d.ts","./node_modules/@redis/json/dist/lib/commands/forget.d.ts","./node_modules/@redis/json/dist/lib/commands/get.d.ts","./node_modules/@redis/json/dist/lib/commands/mset.d.ts","./node_modules/@redis/json/dist/lib/commands/objkeys.d.ts","./node_modules/@redis/json/dist/lib/commands/objlen.d.ts","./node_modules/@redis/json/dist/lib/commands/set.d.ts","./node_modules/@redis/json/dist/lib/commands/strappend.d.ts","./node_modules/@redis/json/dist/lib/commands/strlen.d.ts","./node_modules/@redis/json/dist/lib/commands/type.d.ts","./node_modules/@redis/json/dist/lib/commands/index.d.ts","./node_modules/@redis/json/dist/lib/index.d.ts","./node_modules/@redis/search/dist/lib/commands/create.d.ts","./node_modules/@redis/search/dist/lib/commands/search.d.ts","./node_modules/@redis/search/dist/lib/commands/aggregate.d.ts","./node_modules/@redis/search/dist/lib/commands/aggregate_withcursor.d.ts","./node_modules/@redis/search/dist/lib/commands/cursor_read.d.ts","./node_modules/@redis/search/dist/lib/commands/dropindex.d.ts","./node_modules/@redis/search/dist/lib/commands/explain.d.ts","./node_modules/@redis/search/dist/lib/commands/explaincli.d.ts","./node_modules/@redis/search/dist/lib/commands/hybrid.d.ts","./node_modules/@redis/search/dist/lib/commands/info.d.ts","./node_modules/@redis/search/dist/lib/commands/profile_search.d.ts","./node_modules/@redis/search/dist/lib/commands/search_nocontent.d.ts","./node_modules/@redis/search/dist/lib/commands/spellcheck.d.ts","./node_modules/@redis/search/dist/lib/commands/sugadd.d.ts","./node_modules/@redis/search/dist/lib/commands/sugget.d.ts","./node_modules/@redis/search/dist/lib/commands/synupdate.d.ts","./node_modules/@redis/search/dist/lib/commands/index.d.ts","./node_modules/@redis/search/dist/lib/index.d.ts","./node_modules/@redis/time-series/dist/lib/commands/add.d.ts","./node_modules/@redis/time-series/dist/lib/commands/helpers.d.ts","./node_modules/@redis/time-series/dist/lib/commands/create.d.ts","./node_modules/@redis/time-series/dist/lib/commands/alter.d.ts","./node_modules/@redis/time-series/dist/lib/commands/createrule.d.ts","./node_modules/@redis/time-series/dist/lib/commands/incrby.d.ts","./node_modules/@redis/time-series/dist/lib/commands/get.d.ts","./node_modules/@redis/time-series/dist/lib/commands/info.d.ts","./node_modules/@redis/time-series/dist/lib/commands/info_debug.d.ts","./node_modules/@redis/time-series/dist/lib/commands/madd.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mget.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mget_withlabels.d.ts","./node_modules/@redis/time-series/dist/lib/commands/range.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange_groupby.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange_selected_labels.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange_selected_labels_groupby.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange_withlabels_groupby.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange_withlabels.d.ts","./node_modules/@redis/time-series/dist/lib/commands/mrange.d.ts","./node_modules/@redis/time-series/dist/lib/commands/index.d.ts","./node_modules/@redis/time-series/dist/lib/index.d.ts","./node_modules/redis/dist/index.d.ts","./node_modules/connect-redis/dist/connect-redis.d.cts","./node_modules/@types/multer/index.d.ts","./src/interfaces/api-error.interface.ts","./src/common/interceptors/global-exception.filter.ts","./src/common/interceptors/response-transform.interceptor.ts","./src/common/interceptors/logging.interceptor.ts","./src/config/cache.config.ts","./node_modules/helmet/index.d.cts","./src/config/cors.config.ts","./src/main.ts","./src/assessment/dto/create-assessment.dto.ts","./src/assessment/dto/update-assessment.dto.ts","./src/audit-log/decorators/audit.decorator.ts","./src/audit-log/tests/audit-log.test.ts","./src/auth/jwt.strategy.ts","./src/caching/decorators/cache.decorator.ts","./src/caching/interceptors/cache.interceptor.ts","./src/common/common.module.ts","./src/common/database/examples/booking-transaction.example.ts","./src/common/database/examples/payment-transaction.example.ts","./src/common/database/examples/voting-transaction.example.ts","./src/common/decorators/translate.decorator.ts","./src/common/examples/transaction-management.example.ts","./src/common/interceptors/api-error.interface.ts","./src/common/interceptors/cache.interceptor.ts","./src/common/lazy-loading/lazy-module-loader.service.ts","./src/common/lazy-loading/lazy-loading.module.ts","./src/common/lazy-loading/index.ts","./src/common/naming/naming.service.ts","./src/common/naming/naming.module.ts","./src/common/validators/is-strong-password.validator.ts","./src/config/swagger.config.ts","./src/courses/guards/ws-jwt-auth.guard.ts","./src/email-marketing/dto/index.ts","./src/email-marketing/entities/index.ts","./src/email-marketing/enums/index.ts","./src/gateways/messaging.gateway.ts","./src/gateways/notifications.gateway.ts","./src/graphql/inputs/index.ts","./src/graphql/middleware/dataloader.middleware.ts","./src/graphql/resolvers/index.ts","./src/graphql/types/index.ts","./src/messaging/tracing/tracing.service.ts","./src/messaging/messaging.service.ts","./src/messaging/event-bus/event-bus.service.ts","./src/messaging/discovery/service-discovery.service.ts","./src/messaging/circuit-breaker/circuit-breaker.service.ts","./src/messaging/messaging.module.ts","./src/migrations/samples/sample-user-table.migration.ts","./node_modules/@nestjs/mapped-types/dist/mapped-type.interface.d.ts","./node_modules/@nestjs/mapped-types/dist/types/remove-fields-with-type.type.d.ts","./node_modules/@nestjs/mapped-types/dist/intersection-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/omit-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/partial-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/pick-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/type-helpers.utils.d.ts","./node_modules/@nestjs/mapped-types/dist/index.d.ts","./node_modules/@nestjs/mapped-types/index.d.ts","./src/payments/dto/update-payment.dto.ts","./src/payments/providers/payment-provider.interface.ts","./src/payments/webhooks/migration-helper.ts","./src/payments/webhooks/stripe-webhook.guard.ts","./src/payments/webhooks/dto/webhook-retry.dto.ts","./src/rate-limiting/services/limit-guard/guard.ts","./src/rate-limiting/rate-limiting.controller.ts","./src/rate-limiting/entities/rate-limiting.entity.ts","./src/tenancy/decorators/current-tenant.decorator.ts","./node_modules/@jest/expect-utils/build/index.d.ts","./node_modules/jest-matcher-utils/node_modules/chalk/index.d.ts","./node_modules/@sinclair/typebox/typebox.d.ts","./node_modules/@jest/schemas/build/index.d.ts","./node_modules/pretty-format/build/index.d.ts","./node_modules/jest-diff/build/index.d.ts","./node_modules/jest-matcher-utils/build/index.d.ts","./node_modules/expect/build/index.d.ts","./node_modules/@types/jest/index.d.ts"],"fileIdsList":[[546,593,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2922,2923,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2923,3007,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3006,3087,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2924,2925,2926,2931,2996,3001,3002,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2924,2995,2996,3000,3001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2925,2927,2929,2930,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2928,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2927,2929,2930,2931,2997,2999,3000,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2927,2930,2931,2997,2999,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2922,2924,2925,2927,2931,2998,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2928,3001,3003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2924,3001,3003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2925,3008,3009,3014,3079,3084,3085,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,3008,3078,3079,3083,3084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2925,3010,3012,3013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3011,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3010,3012,3013,3014,3080,3082,3083,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,3010,3013,3014,3080,3082,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2925,3007,3008,3010,3014,3081,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3011,3084,3086,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,3008,3084,3086,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2921,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2918,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2915,2917,2919,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2916,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2920,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3005,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2222,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1858,1922,1932,2014,2029,2040,2059,2095,3817,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,3988,3991,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,3819,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1952,2040,2095,3819,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,3901,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,3989,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,3819,3901,3988,3990,3991,3992,3993,3994,3995,4014,4019,4020,4021,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,4020,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,3819,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3996,3997,3998,3999,4000,4001,4002,4003,4004,4005,4006,4007,4008,4009,4010,4011,4012,4013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3920,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3922,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3923,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3927,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3928,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3934,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3937,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3938,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3939,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3940,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3945,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3944,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3947,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3948,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3950,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3953,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3955,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3990,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4015,4016,4017,4018,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3888,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3897,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3898,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3914,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2421,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2423,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,2478,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2422,2423,2477,2479,2480,2481,2482,2483,2484,2494,2495,2496,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1858,1922,1932,2014,2029,2040,2059,2095,2421,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,2477,2480,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,2495,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2422,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2435,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2485,2486,2487,2488,2489,2490,2491,2492,2493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2454,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2455,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2456,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2457,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2458,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2459,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2460,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2479,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2105,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,2104,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2107,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2107,2207,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2207,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,2216,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2104,2106,2107,2207,2217,2218,2219,2223,2224,2225,2231,2236,2237,2238,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,2106,2237,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2106,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2106,2107,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2226,2227,2228,2229,2230,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2173,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2174,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2177,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2179,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1830,1857,1858,1902,1922,1932,2011,2014,2029,2031,2040,2059,2095,2104,2105,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,2218,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2232,2233,2234,2235,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2167,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2168,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2097,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2096,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2096,2097,2098,2099,2100,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1847,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1847,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1919,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2101,2102,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1987,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1988,1989,1991,1992,1993,1994,1995,1996,1997,1998,1999,2000,2001,2002,2003,2004,2005,2006,2007,2008,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1970,1993,1994,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,1991,1992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1990,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1970,2005,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,2002,2004,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,2001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2000,2002,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1848,1851,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1852,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1852,1853,1854,1855,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1848,1849,1850,1851,1853,1856,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1847,1848,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1848,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1859,1860,1861,1862,1934,1935,1936,1937,2010,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2009,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1860,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1904,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,1903,1905,1906,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1903,1906,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1903,1904,1905,1906,1907,1920,1921,1922,1933,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,1903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1934,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1935,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2012,2013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,2012,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1876,1877,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1870,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1872,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1870,1871,1873,1874,1875,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1863,1864,1865,1866,1867,1868,1869,1872,1876,1877,1878,1879,1880,1881,1882,1883,1884,1885,1886,1887,1888,1889,1890,1891,1892,1893,1894,1895,1896,1897,1898,1899,1900,1901,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1876,1877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,3379,3496,3497,3498,3587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,3496,3498,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3499,3500,3501,3502,3503,3504,3505,3506,3507,3508,3509,3510,3511,3512,3513,3514,3515,3516,3517,3518,3519,3520,3521,3522,3523,3524,3525,3526,3527,3528,3529,3530,3531,3532,3533,3534,3535,3536,3537,3538,3539,3540,3541,3542,3543,3544,3545,3546,3547,3548,3549,3550,3551,3552,3553,3554,3555,3556,3557,3558,3559,3560,3561,3562,3563,3564,3565,3566,3567,3568,3569,3570,3571,3572,3573,3574,3575,3576,3577,3578,3579,3580,3581,3582,3583,3584,3585,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3325,3374,3375,3378,3379,3497,3586,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,3378,3495,3496,3587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3367,3368,3373,3374,3375,3376,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,631,634,635,3324,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,3325,3326,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3325,3327,3366,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3325,3365,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,3367,3373,3375,3376,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,3374,3375,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3324,3367,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3367,3369,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3369,3370,3371,3372,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3324,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,3298,3324,3367,3368,3373,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,634,3367,3374,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2986,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3069,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2986,2987,2988,2989,2990,2991,2992,2993,2994,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3069,3070,3071,3072,3073,3074,3075,3076,3077,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3075,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3015,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3015,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2932,2933,2934,2935,2936,2937,2938,2939,2940,2941,2942,2943,2944,2945,2946,2947,2948,2949,2950,2951,2952,2953,2954,2955,2956,2957,2958,2959,2960,2961,2962,2963,2964,2965,2966,2967,2968,2969,2970,2971,2972,2973,2974,2975,2976,2977,2978,2979,2980,2981,2982,2983,2984,2985,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3015,3016,3017,3018,3019,3020,3021,3022,3023,3024,3025,3026,3027,3028,3029,3030,3031,3032,3033,3034,3035,3036,3037,3038,3039,3040,3041,3042,3043,3044,3045,3046,3047,3048,3049,3050,3051,3052,3053,3054,3055,3056,3057,3058,3059,3060,3061,3062,3063,3064,3065,3066,3067,3068,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2736,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2972,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3055,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2933,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3016,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3760,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3758,3759,3761,3763,3765,3766,3767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3764,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3765,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3725,3726,3730,3731,3732,3733,3734,3735,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3762,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3721,3722,3724,3725,3726,3727,3729,3730,3731,3732,3733,3734,3735,3736,3737,3738,3739,3740,3741,3742,3743,3744,3745,3746,3747,3748,3749,3750,3751,3752,3753,3754,3755,3756,3757,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3701,3702,3703,3704,3706,3707,3708,3709,3713,3714,3715,3716,3717,3718,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3702,3703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3704,3705,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3710,3711,3712,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3701,3711,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3670,3671,3672,3673,3674,3675,3676,3677,3678,3679,3680,3681,3682,3683,3684,3685,3686,3687,3688,3689,3690,3691,3692,3693,3694,3695,3696,3697,3698,3699,3700,3706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4617],[546,593,2912,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3004,3092,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,2730,2911,3092,3093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,3092,3093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3094,3095,3096,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3098,3099,3100,3101,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2913,3092,3097,3102,3104,3106,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3004,3088,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3089,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2911,3089,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3089,3090,3091,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3105,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[454,460,546,593,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3004,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3103,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3778,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,546,593,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3778,3779,3780,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1615,1616,1618,1620,1622,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1649,1651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1646,1647,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1653,1654,1655,1656,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1658,1659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1623,1648,1651,1652,1657,1660,1663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1647,1649,1650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1649,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1661,1662,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1678,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1048,1678,1679,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1681,1682,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1676,1680,1683,1685,1686,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1684,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1047,1048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1677,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1677,1678,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1687,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[53,304,305,306,307,308,309,310,311,312,313,314,315,316,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,290,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[263,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[253,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[321,322,323,324,325,326,327,328,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[258,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[317,320,329,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[318,319,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[294,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[258,259,260,261,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[331,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[276,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[331,332,333,334,335,336,337,338,339,340,341,342,343,344,345,346,347,348,349,350,351,352,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[359,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[354,355,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[356,358,546,593,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,262,303,330,353,358,360,367,390,395,397,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,248,249,471,476,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[248,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,247,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,369,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[250,371,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,251,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[255,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[268,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[270,271,272,273,274,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,263,278,282,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[276,277,283,284,285,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,55,56,57,58,248,249,250,251,252,253,254,255,256,257,263,268,269,275,282,286,287,288,290,298,299,300,301,302,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[264,265,266,267,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,264,265,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,262,263,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,266,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,294,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[289,291,292,293,294,295,296,297,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[290,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,256,289,293,295,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[265,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[291,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,290,291,292,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,260,280,298,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,279,281,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[252,254,263,269,278,283,299,300,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,252,254,257,299,300,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[261,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,303,361,365,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[365,366,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,361,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,361,362,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[362,363,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[362,363,364,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[382,383,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[382,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[383,384,385,386,387,388,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[381,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[373,383,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[383,384,385,386,387,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,382,383,386,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[368,374,375,376,377,378,379,380,389,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,303,374,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,373,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,373,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[250,256,257,369,370,371,372,373,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,369,370,391,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,369,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[393,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[330,391,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[391,392,394,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,357,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[289,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[396,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,644,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,534,539,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[533,539,546,593,644,645,646,649,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[539,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[540,546,593,642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[534,540,546,593,643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[535,536,537,538,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,647,648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[539,546,593,644,650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,282,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[440,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,460,461,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[442,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,454,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[464,465,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,303,455,460,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,441,467,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,398,468,471,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,460,462,473,475,479,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,477,478,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[468,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,455,460,462,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[481,483,484,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,454,455,460,480,482,485,488,493,494,507,508,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,440,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[467,470,509,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[494,506,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,441,462,463,466,469,501,506,510,513,517,518,519,521,523,529,531,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,448,456,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,452,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,451,452,453,454,459,460,462,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[454,455,458,460,496,505,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,447,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[495,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,455,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,448,455,459,500,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,447,459,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,453,454,458,498,502,503,504,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,448,455,456,457,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,442,455,458,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[459,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[444,445,446,455,459,460,499,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[451,500,511,512,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,442,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,442,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[443,444,445,446,449,451,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[448,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[450,451,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,443,444,445,446,449,450,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[486,487,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,460,462,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[497,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[287,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[268,303,514,515,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[516,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,462,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,462,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,303,398,448,455,456,457,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,280,303,398,441,455,462,500,518,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,282,398,440,520,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[490,491,492,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,489,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[522,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[525,527,528,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[524,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[526,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,454,459,525,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[472,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,455,459,460,462,497,498,500,501,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[530,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3590,3591,3592,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,3588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1671,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1670,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1670,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1665,1666,1672,1673,1674,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1667,1668,1669,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,398,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,2802,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2627,2805,2806,2807,2808,2809,2810,2811,2812,2813,2814,2815,2816,2817,2818,2819,2820,2821,2822,2823,2824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2628,2802,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2802,2818,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,2730,2804,2879,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2880,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2883,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2877,2878,2885,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2884,2886,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2882,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2732,2804,2858,2859,2877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2797,2804,2877,2878,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,532,546,593,2799,2880,2882,2888,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2804,2825,2861,2862,2863,2874,2879,2881,2882,2883,2887,2888,2889,2890,2893,2897,2898,2901,2908,2909,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2731,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,546,593,2730,2732,2796,2797,2798,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,2731,2732,2733,2734,2735,2798,2799,2800,2801,2802,2803,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2891,2892,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2836,2840,2850,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2827,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2836,2839,2840,2849,2850,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2826,2837,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2830,2839,2840,2842,2843,2844,2849,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2852,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2830,2839,2840,2843,2846,2849,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2830,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2829,2830,2840,2843,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2836,2844,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2853,2854,2855,2856,2857,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2858,2859,2895,2896,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2827,2831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2827,2828,2829,2831,2832,2833,2834,2835,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2826,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2731,2804,2826,2827,2828,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2804,2826,2827,2828,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2730,2804,2826,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2894,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2838,2841,2845,2847,2848,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2836,2842,2846,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2838,2841,2845,2847,2848,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[459,460,546,593,2860,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,482,546,593,2862,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2867,2873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2861,2862,2863,2874,2875,2876,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[451,459,460,509,532,546,593,2804,2824,2860,2861,2880,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[460,532,546,593,2730,2799,2804,2861,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2899,2900,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2902,2904,2905,2906,2907,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2643,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2737,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2737,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2740,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2737,2738,2739,2740,2741,2742,2743,2744,2745,2746,2747,2748,2749,2750,2751,2752,2753,2754,2755,2756,2757,2758,2759,2760,2761,2762,2763,2764,2765,2766,2767,2768,2769,2770,2771,2772,2773,2774,2775,2776,2777,2778,2779,2780,2781,2782,2783,2784,2785,2786,2787,2788,2789,2790,2791,2792,2793,2794,2795,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2738,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2741,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3159,3160,3161,3162,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3158,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3157,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3158,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3159,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3163,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4597,4599,4600,4601,4602,4603],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4597,4598],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4604],[398,546,593,1692,1694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1691,1694,1695,1696,1698,1699,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1692,1693,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1692,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1700,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,282,303,398,440,546,593,607,609,1741,1742,1743,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1744,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1745,1747,1758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1741,1742,1746,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,607,609,1601,1741,1742,1743,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1754,1756,1757,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1748,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1749,1750,1751,1752,1753,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,1748,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1755,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1755,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1580,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1581,1582,1583,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1562,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1563,1584,1585,1586,1587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1585,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,401,402,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,402,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[415,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,401,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[399,400,403,404,405,406,407,408,409,410,411,412,413,414,416,417,418,419,420,421,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,426,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,422,426,427,428,433,435,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,424,425,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,423,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,426,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[429,430,431,432,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[434,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[436,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1705,1706,1707,1708,1709,1710,1711,1712,1714,1715,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,1705,1706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,1705,1706,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1704,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1705,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1704,1705,1713,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1039,1040,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1038,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,1037,1038,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1041,1043,1044,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1038,1042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1045,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2513,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2515,2516,2517,2527,2528,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2526,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2530,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2532,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,501,546,593,2534,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2514,2526,2529,2531,2533,2535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2519,2520,2521,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2518,2522,2523,2524,2525,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3261,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3265,3267,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3254,3258,3269,3270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3283,3289,3291,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3252,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3263,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,3254,3255,3256,3257,3258,3259,3260,3261,3262,3263,3264,3265,3266,3267,3268,3269,3271,3272,3273,3274,3275,3276,3277,3278,3279,3280,3281,3282,3283,3284,3285,3286,3287,3288,3289,3290,3292,3293,3294,3295,3296,3297,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3268,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3264,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3257,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3264,3265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3271,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3292,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3257,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3274,3277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3273,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3274,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3272,3274,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3277,3279,3280,3281,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3281,3283,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3272,3275,3278,3285,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3272,3273,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3255,3256,3272,3274,3275,3276,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3274,3277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3256,3257,3275,3278,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3277,3279,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3281,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4474,4475,4476,4477,4478,4479,4480,4481,4482,4483,4484,4485,4486,4487,4488],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4489],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4355,4356,4357,4358,4359,4360,4388,4444,4457,4460,4462,4464,4465,4467,4471,4472],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4448],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4446,4447,4448,4449,4450],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4446,4447,4448],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4464],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4357,4359,4452,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4453,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4460],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4397,4444,4445,4451,4452,4453,4454,4455,4456,4457,4461,4462,4463],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4355,4359,4444,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4444,4454],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4453,4454,4455,4457,4461,4464],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4453],[546,593,604,612,631,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,600,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4452,4457,4460,4464],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4445,4452,4453,4457,4458,4459,4461,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4371],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4376],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4382,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4386],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4361,4362,4363,4364,4365,4366,4367,4368,4369,4370,4371,4372,4373,4374,4375,4376,4377,4378,4379,4380,4381,4382,4383,4384,4385,4386,4387,4388,4389,4390,4391,4392,4393,4394,4395,4396,4397,4398,4399,4400,4401,4402,4403,4404,4405,4406,4407,4408,4409,4410,4411,4412,4413,4414,4415,4416,4417,4418,4419,4420,4421,4422,4423,4424,4425,4426,4427,4428,4429,4430,4431,4432,4433,4434,4435,4436,4437,4438,4439,4440,4441,4442,4443,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4404,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4368],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4415],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4427],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4461,4468],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4468,4469,4470],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4468],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4355,4359],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4355,4356,4357,4358],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4452,4453,4454,4457,4461,4464,4465,4466],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4444,4454,4465],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4445,4453,4457,4464,4467],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4491,4492,4493,4494,4495,4496,4497,4498,4499,4500,4501,4502,4503,4504,4505],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4506],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4508,4509],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4510],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4511],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4509],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4510],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4508,4509,4510,4511,4512,4513,4514,4515,4516,4517,4518,4519,4520,4521,4522,4523],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4509,4510],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4508],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4473,4509],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4508,4509,4510,4524],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4527],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4528],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4526,4527],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4526],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4526,4527,4528,4529,4530,4531,4532,4533,4534,4535,4536,4537,4538,4539,4540,4541,4542,4543,4544],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4527,4530],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4533],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4536],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538,4539,4540],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538,4539],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4530,4538,4539,4545],[546,593,2015,2016,2017,2018,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2019,2022,2028,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2020,2021,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2025,2026,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2025,2026,2027,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2024,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1984,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1938,1939,1940,1985,1986,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,1939,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,1985,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1971,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1952,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,1970,1974,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1972,1973,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1953,1954,1973,1974,1975,1976,1977,1978,1979,1980,1981,1982,1983,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1974,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1973,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1981,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1955,1957,1958,1959,1960,1961,1962,1963,1964,1965,1966,1967,1968,1969,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1956,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1963,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1958,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1964,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2085,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2082,2083,2086,2087,2088,2089,2090,2091,2092,2093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2030,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2032,2033,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2034,2035,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2032,2033,2036,2037,2038,2039,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2051,2052,2053,2054,2055,2056,2057,2058,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2049,2051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2050,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2055,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1831,1844,1845,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1843,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1831,1844,1845,1846,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1925,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1926,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1928,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1923,1924,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1923,1924,1925,1927,1928,1929,1930,1931,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1832,1833,1834,1835,1837,1838,1839,1840,1841,1842,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1836,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1837,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1908,1909,1910,1911,1912,1913,1914,1915,1916,1917,1918,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1908,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1984,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2060,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2070,2071,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2072,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2060,2061,2062,2063,2064,2065,2066,2067,2068,2069,2073,2074,2075,2076,2077,2078,2079,2080,2081,2094,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1761,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1774,1775,1776,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1774,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,1764,1775,1776,1777,1778,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1767,1773,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1786,1788,1789,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1786,1787,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1791,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1768,1793,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,1794,1795,1796,1797,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1770,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,1794,1795,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1772,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1783,1784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,1803,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,1800,1803,1804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1773,1802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1780,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,1771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,1770,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1807,1808,1809,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,1766,1767,1768,1769,1770,1771,1772,1773,1774,1779,1782,1783,1784,1785,1787,1790,1791,1792,1798,1801,1802,1805,1806,1810,1811,1812,1813,1814,1815,1816,1817,1818,1819,1820,1821,1822,1823,1824,1826,1827,1828,1829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,1767,1768,1769,1770,1773,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1767,1785,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1768,1774,1813,1815,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1768,1774,1813,1814,1815,1816,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,1773,1787,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,1791,1808,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1773,1783,1784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,1781,1813,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1767,1823,1824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,608,1767,1772,1785,1813,1822,1823,1824,1825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1767,1785,1801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2041,2042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2041,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2041,2042,2043,2044,2045,2046,2047,2048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1830,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1943,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1942,1944,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1941,1942,1945,1946,1947,1948,1949,1950,1951,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2220,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2220,2221,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,1598,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,641,1592,1593,1594,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,604,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1595,1597,1599,1600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4619,4622],[546,593,598,641,3156,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1572,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1566,1568,1569,1573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,1567,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1567,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,1568,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1565,1567,1568,1569,1570,1571,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,599,604,612,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,595,604,612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[541,542,543,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,596,635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,597,598,605,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,623,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,599,601,604,612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,602,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,603,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,606,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,606,619,623,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,604,607,612,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,607,608,612,623,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,623,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[544,545,546,547,548,549,550,589,590,591,592,593,594,595,596,597,598,599,600,601,602,603,604,605,606,607,608,609,610,611,612,613,614,615,616,617,618,619,620,621,622,623,624,625,626,627,628,629,630,631,632,633,634,635,636,637,638,639,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,611,634,639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,604,612,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,614,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,615,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,591,592,593,594,595,596,597,598,599,600,601,602,603,604,605,606,607,608,609,610,611,612,613,614,615,616,617,618,619,620,621,622,623,624,625,626,627,628,629,630,631,632,633,634,635,636,637,638,639,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,619,620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,619,621,635,637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,624,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,625,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,593,623,628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,629,630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,629,630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,612,623,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,632,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,618,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,611,637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,588,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,588,593,604,606,615,623,626,634,637,639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3174,3176,3180,3181,3182,3183,3184,3185,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,623,634,641,3173,3174,3175,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3176,3177,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3176,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3178,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,623,631,641,3175,3177,3179,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3174,3176,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,3174,3175,3176,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,641,3179,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3195,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1601,1697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,605,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,1596,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2312,2313,2314,2315,2316,2317,2318,2319,2320,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,609,612,623,631,634,640,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,3436,3438,3439,3440,3441,3445,3446,3447,3448,3449,3450,3451,3452,3453,3454,3455,3456,3457,3458,3459,3460,3461,3462,3463,3464,3465,3466,3467,3468,3469,3470,3471,3472,3473,3474,3475,3476,3477,3481,3483,3484,3485,3486,3487,3488,3489,3490,3491,3492,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3494,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3442,3443,3444,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3450,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3442,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3468,3469,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3401,3404,3408,3410,3422,3436,3437,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3467,3468,3469,3470,3472,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3387,3388,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3401,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3392,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3395,3396,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3404,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3397,3398,3425,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3408,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3392,3393,3427,3434,3435,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3389,3390,3391,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3394,3398,3426,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3396,3428,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3396,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3429,3430,3431,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3425,3432,3433,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3425,3433,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3396,3399,3400,3402,3403,3405,3406,3407,3409,3411,3412,3413,3414,3415,3416,3417,3418,3419,3420,3421,3423,3424,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3422,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3445,3446,3447,3448,3449,3450,3451,3452,3453,3454,3455,3456,3457,3458,3459,3460,3461,3462,3463,3464,3465,3466,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3475,3476,3477,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,606,623,626,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3475,3476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3445,3468,3469,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3438,3439,3475,3476,3477,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3438,3439,3476,3478,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3387,3388,3389,3390,3391,3392,3393,3426,3427,3438,3439,3468,3476,3480,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3438,3439,3468,3469,3470,3471,3475,3476,3477,3479,3481,3482,3483,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,3468,3472,3484,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3388,3438,3439,3468,3469,3470,3471,3472,3475,3476,3477,3479,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3469,3470,3472,3481,3493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3468,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3467,3468,3469,3470,3471,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3439,3440,3441,3467,3469,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,641,3386,3467,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3468,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3469,3470,3473,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3116,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1048,1560,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1438,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1412,1413,1414,1415,1416,1417,1418,1419,1420,1421,1422,1423,1424,1425,1426,1427,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,1455,1456,1457,1458,1459,1460,1461,1462,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1453,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1463,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1050,1051,1084,1123,1311,1402,1406,1410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1051,1400,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1052,1053,1054,1055,1056,1057,1058,1059,1060,1061,1062,1063,1064,1065,1066,1067,1068,1069,1070,1071,1072,1073,1074,1075,1076,1077,1078,1079,1080,1081,1082,1083,1085,1086,1087,1088,1089,1090,1091,1092,1093,1094,1095,1096,1097,1098,1099,1100,1101,1102,1103,1104,1105,1106,1107,1108,1109,1110,1111,1112,1113,1114,1115,1116,1117,1118,1119,1120,1121,1122,1123,1124,1125,1126,1127,1128,1129,1130,1131,1132,1133,1134,1135,1136,1137,1138,1139,1140,1141,1142,1143,1144,1145,1146,1147,1148,1149,1150,1151,1152,1153,1154,1155,1156,1157,1158,1159,1160,1161,1162,1163,1164,1165,1166,1167,1168,1169,1170,1171,1172,1173,1174,1175,1176,1177,1178,1179,1180,1181,1182,1183,1184,1185,1186,1187,1188,1189,1190,1191,1192,1193,1194,1195,1196,1197,1198,1199,1200,1201,1202,1203,1204,1205,1206,1207,1208,1209,1210,1211,1212,1213,1214,1215,1216,1217,1218,1219,1220,1221,1222,1223,1224,1225,1226,1227,1228,1229,1230,1231,1232,1233,1234,1235,1236,1237,1238,1239,1240,1241,1242,1243,1244,1245,1246,1247,1248,1249,1250,1251,1252,1253,1254,1255,1256,1257,1258,1259,1260,1261,1262,1263,1264,1265,1266,1267,1268,1269,1270,1271,1272,1273,1274,1275,1276,1277,1278,1279,1280,1281,1282,1283,1284,1285,1286,1287,1288,1289,1290,1291,1292,1293,1294,1295,1296,1297,1298,1299,1300,1301,1302,1303,1304,1305,1306,1307,1308,1309,1310,1311,1312,1313,1314,1315,1316,1317,1318,1319,1320,1321,1322,1323,1324,1325,1326,1327,1328,1329,1330,1331,1332,1333,1334,1335,1336,1337,1338,1339,1340,1341,1342,1343,1344,1345,1346,1347,1348,1349,1350,1351,1352,1353,1354,1355,1356,1357,1358,1359,1360,1361,1362,1363,1364,1365,1366,1367,1368,1369,1370,1371,1372,1373,1374,1375,1376,1377,1378,1379,1380,1381,1382,1383,1384,1385,1386,1387,1388,1389,1390,1391,1392,1393,1394,1395,1396,1397,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1049,1051,1084,1165,1250,1398,1399,1400,1401,1403,1404,1405,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1398,1403,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,631,641,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,1051,1400,1406,1410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1176,1177,1178,1179,1180,1181,1182,1183,1184,1185,1186,1187,1188,1189,1190,1191,1192,1193,1194,1195,1196,1197,1198,1199,1200,1201,1202,1203,1204,1205,1206,1207,1208,1209,1210,1211,1212,1213,1214,1215,1216,1217,1218,1219,1220,1221,1222,1223,1224,1225,1226,1227,1228,1229,1230,1231,1232,1233,1234,1235,1236,1237,1238,1239,1240,1241,1242,1243,1244,1245,1246,1247,1248,1249,1250,1251,1252,1253,1254,1255,1256,1257,1258,1259,1260,1261,1262,1263,1264,1265,1266,1267,1268,1269,1270,1271,1272,1273,1274,1275,1276,1277,1278,1279,1280,1281,1282,1283,1284,1285,1286,1287,1288,1289,1290,1291,1292,1293,1294,1295,1296,1297,1298,1299,1300,1301,1302,1303,1304,1305,1306,1307,1308,1309,1310,1311,1312,1313,1314,1315,1316,1317,1318,1319,1320,1321,1322,1323,1324,1325,1326,1327,1328,1329,1330,1331,1332,1333,1334,1335,1336,1337,1338,1339,1340,1341,1342,1343,1344,1345,1346,1347,1348,1349,1350,1351,1352,1353,1354,1355,1356,1357,1358,1359,1360,1361,1362,1363,1364,1365,1366,1367,1368,1369,1370,1371,1372,1373,1374,1375,1376,1377,1378,1379,1380,1381,1382,1383,1384,1385,1386,1387,1388,1389,1390,1391,1392,1393,1394,1395,1396,1397,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1051,1400,1406,1407,1408,1409,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1403,1407,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1178,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1183,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1185,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1074,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1101,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1123,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1211,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1213,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1215,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1221,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1066,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1232,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1247,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1248,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1250,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1049,1050,1406,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1280,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1325,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1339,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1341,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1364,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1368,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1374,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1376,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1378,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1379,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1381,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1395,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1402,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1465,1466,1467,1468,1469,1470,1471,1472,1473,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1471,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1406,1471,1472,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1474,1475,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1477,1478,1479,1480,1481,1482,1483,1484,1485,1486,1487,1488,1489,1490,1491,1492,1493,1494,1495,1496,1497,1498,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1503,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1504,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1501,1502,1503,1504,1505,1506,1507,1508,1509,1510,1511,1512,1513,1514,1515,1516,1517,1518,1519,1520,1521,1522,1523,1524,1525,1526,1527,1528,1529,1530,1531,1532,1533,1534,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1503,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1520,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1520,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1527,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1527,1528,1529,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1503,1520,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1537,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1537,1538,1539,1540,1541,1542,1543,1544,1545,1546,1547,1548,1549,1550,1551,1552,1553,1554,1555,1556,1557,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1546,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1549,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1411,1412,1413,1414,1415,1416,1417,1418,1419,1420,1421,1422,1423,1424,1425,1426,1427,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,1455,1456,1457,1458,1459,1460,1461,1462,1464,1465,1466,1467,1468,1469,1470,1471,1472,1473,1476,1477,1478,1479,1480,1481,1482,1483,1484,1485,1486,1487,1488,1489,1490,1491,1492,1493,1494,1495,1496,1497,1498,1500,1501,1502,1503,1504,1505,1506,1507,1508,1509,1510,1511,1512,1513,1514,1515,1516,1517,1518,1519,1520,1521,1522,1523,1524,1525,1526,1527,1528,1529,1530,1531,1532,1533,1534,1536,1537,1538,1539,1540,1541,1542,1543,1544,1545,1546,1547,1548,1549,1550,1551,1552,1553,1554,1555,1556,1557,1559,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,1047,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2612,2613,2614,2615,2616,2617,2618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2602,2610,2611,2619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2603,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2594,2595,2596,2597,2598,2599,2600,2603,2604,2605,2606,2607,2608,2609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2602,2604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2605,2610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,2284,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2285,2286,2287,2288,2290,2291,2292,2293,2294,2295,2296,2297,2298,2299,2300,2301,2302,2303,2304,2305,2306,2307,2308,2309,2310,2311,2322,2323,2324,2325,2326,2327,2328,2329,2330,2331,2332,2333,2334,2335,2336,2337,2338,2339,2340,2341,2342,2343,2344,2345,2346,2347,2348,2349,2350,2351,2352,2353,2354,2355,2356,2357,2358,2359,2362,2363,2364,2365,2366,2367,2368,2369,2370,2371,2372,2373,2374,2375,2376,2377,2378,2379,2380,2381,2382,2383,2384,2385,2386,2387,2388,2389,2390,2391,2392,2393,2394,2395,2396,2397,2398,2399,2400,2401,2402,2403,2404,2405,2406,2407,2408,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2321,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2361,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2279,2280,2281,2282,2283,2284,2289,2409,2410,2411,2412,2416,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2281,2414,2415,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,2413,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2279,2280,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4352,4547],[546,593,1573,1576,1578,1579,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1573,1578,1579,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1573,1574,1578,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,1573,1575,1576,1577,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2538,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2538,2539,2540,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2541,2544,2545,2548,2552,2553,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,623,2543,2544,2545,2546,2547,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,2541,2544,2548,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,2541,2542,2543,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,2549,2550,2551,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2541,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2541,2543,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,2548,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4615,4621],[546,593,3382,3384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3381,3384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3380,3381,3382,3383,3384,3385,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3382,3383,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3109,3110,3111,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3109,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3108,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3109,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2864,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2864,2865,2866,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2637,2638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2704,2705,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2633,2634,2637,2639,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2633,2644,2658,2659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2642,2643,2644,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2639,2641,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2642,2643,2644,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2629,2645,2650,2657,2660,2661,2703,2706,2729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2635,2636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2635,2636,2637,2638,2640,2651,2652,2653,2654,2655,2656,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2636,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2636,2637,2639,2652,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2635,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2644,2658,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2633,2634,2637,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2640,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2633,2641,2642,2643,2646,2647,2648,2649,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2639,2641,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2642,2653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,2707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2662,2707,2708,2709,2710,2711,2712,2713,2714,2715,2716,2717,2718,2719,2720,2721,2722,2723,2724,2725,2726,2727,2728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2641,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2632,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2641,2642,2643,2655,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2663,2664,2665,2666,2667,2668,2669,2670,2671,2672,2673,2674,2675,2676,2677,2678,2679,2680,2681,2682,2683,2684,2685,2686,2687,2688,2689,2690,2691,2692,2693,2694,2695,2696,2697,2698,2699,2700,2701,2702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2655,2663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2663,2665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,2642,2662,2663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,2641,2642,2643,2655,2662,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,641,1629,1636,1637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1625,1626,1628,1629,1637,1638,1643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,631,641,1624,1630,1632,1633,1638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1632,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,641,1624,1630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1640,1641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,1625,1626,1627,1630,1634,1635,1636,1637,1638,1639,1643,1644,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1625,1629,1639,1643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1625,1626,1628,1629,1636,1639,1640,1642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1629,1631,1634,1635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,1625,1627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4619],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4616,4620],[546,593,2360,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4618],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4474,4475,4476,4477,4478,4479,4480,4481,4482,4483,4484,4485,4486,4487,4488,4490,4491,4492,4493,4494,4495,4496,4497,4498,4499,4500,4501,4502,4503,4504,4505,4506,4507,4508,4509,4510,4511,4512,4513,4514,4515,4516,4517,4518,4519,4520,4521,4522,4523,4525,4526,4528,4529,4531,4532,4533,4534,4535,4536,4537,4538,4539,4540,4541,4542,4543,4544,4545,4546],[59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,75,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,115,116,117,118,119,120,121,122,123,124,125,126,128,129,130,131,132,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,178,179,180,182,191,193,194,195,196,197,198,200,201,203,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,238,239,240,241,242,243,244,245,246,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,60,61,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,63,220,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,62,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,220,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,228,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[72,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[95,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[116,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,111,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,104,122,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,122,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,163,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,181,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,182,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[204,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[188,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[199,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[188,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,181,188,189,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[181,182,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[202,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,188,189,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[61,62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,182,183,184,185,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[104,182,183,184,185,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[182,184,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,183,184,186,187,191,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,62,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,206,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,105,106,107,108,109,110,112,113,114,115,116,117,118,119,120,121,123,124,125,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[192,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2559,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2559,2560,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2555,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2557,2561,2562,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2554,2556,2557,2564,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,608,609,2554,2556,2557,2561,2562,2563,2564,2565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2557,2558,2561,2563,2564,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2554,2556,2557,2558,2561,2562,2563,2565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,4032,4033,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,4032,4033,4034,4035,4036,4037,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2644,2868,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2869,2870,2871,2872,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,2543,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,719,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,825,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,715,716,717,718,720,721,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,673,716,720,723,727,759,775,776,778,780,786,787,788,789,825,826,827,828,831,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,791,793,795,796,806,808,809,810,811,812,813,814,816,818,819,820,821,824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,664,665,695,937,938,939,940,941,942,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,946,947,948,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,955,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,953,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,983,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,971,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,715,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,696,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,970,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,664,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,699,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,700,701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,656,658,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,711,712,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,654,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,991,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,815,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,790,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,654,655,662,668,670,672,686,687,688,691,692,722,723,725,726,831,837,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,672,690,723,725,731,733,747,760,764,768,775,829,835,837,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,612,631,731,732,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,792,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,807,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,794,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,723,822,823,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,689,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,797,798,799,800,801,802,803,804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,805,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,733,735,739,740,741,742,743,770,772,773,774,776,778,779,780,784,785,787,829,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,733,736,740,744,745,769,770,772,773,774,786,829,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,829,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,714,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,696,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,695,697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,693,698,702,703,704,705,706,707,708,709,710,713,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,653,654,655,659,699,700,701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,831,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,686,718,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,789,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,874,875,878,879,880,881,882,883,884,885,886,887,888,889,890,891,892,893,894,895,896,897,898,899,900,901,902,903,904,905,906,907,908,909,910,911,912,913,914,915,916,917,918,919,920,921,922,923,924,925,926,927,928,929,930,931,932,933,934,935,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,727,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,728,735,736,737,738,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,663,668,729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,739,740,777,831,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,733,734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,744,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,775,776,778,786,787,825,826,829,855,868,869,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,656,658,659,662,663,665,668,669,670,671,672,673,693,694,698,699,701,702,703,714,715,716,717,718,721,723,724,725,727,728,729,730,733,734,735,736,737,738,739,740,741,742,743,746,747,749,750,751,752,753,754,755,756,757,758,759,761,764,765,768,770,771,772,773,774,775,776,777,778,781,782,786,787,788,789,825,829,831,834,835,836,837,838,848,849,851,852,853,854,855,869,870,871,872,936,943,944,945,949,950,951,952,954,956,957,958,959,960,961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980,981,982,984,985,986,987,988,989,990,992,993,994,995,996,997,998,999,1000,1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,1011,1012,1013,1014,1015,1016,1017,1018,1019,1020,1021,1022,1024,1025,1026,1027,1028,1029,1030,1031,1032,1034,1036,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,717,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,838,1017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,717,838,1017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,665,666,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,680,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,657,660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,858,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,661,667,676,677,681,683,713,762,766,830,832,856,857,858,859,860,861,862,863,864,865,866,867,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,656,657,660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,703,704,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,762,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,667,668,675,685,829,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,676,678,679,682,684,686,829,831,833,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,680,681,685,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,674,675,678,679,682,684,685,686,703,704,711,712,713,763,767,829,830,833,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,766,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,673,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,685,686,831,832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,855,856,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,677,831,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,653,654,655,657,661,668,674,685,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,686,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,673,683,685,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,788,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,789,829,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,1030,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,672,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,831,876,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,686,877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,623,718,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,774,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,726,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,770,772,783,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,829,831,840,847,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,727,770,782,829,831,838,839,840,846,847,848,849,850,851,852,853,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,703,726,829,831,839,840,841,842,843,844,845,846,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,847,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,829,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,675,703,731,735,736,737,738,740,781,782,831,838,844,845,847,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,703,773,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,703,770,773,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,670,672,690,723,725,731,747,760,764,768,771,780,786,829,835,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,668,778,786,787,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,733,735,739,740,741,742,743,770,772,773,774,784,785,787,855,1023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,733,739,740,744,745,775,787,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,733,735,739,740,741,742,743,770,772,773,774,784,785,786,838,855,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,777,787,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,726,783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,724,746,761,765,834,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,686,690,691,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,690,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,725,727,747,764,768,831,835,836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,761,763,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,765,767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,674,724,727,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,833,834,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,684,746,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,671,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,748,759,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,749,750,751,752,753,754,755,756,757,758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,753,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,672,686,689,715,725,730,734,747,764,768,775,782,826,831,835,837,848,849,850,851,852,853,855,877,1023,1024,1025,1033,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,831,1035,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,564,593,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,593,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,557,560,593,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,557,560,593,612,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,553,556,559,593,604,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,567,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,558,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,581,582,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,556,560,593,626,634,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,581,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,554,555,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,554,555,556,557,558,559,560,561,562,564,565,566,567,568,569,570,571,572,573,574,575,576,577,578,579,580,582,583,584,585,586,587,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,575,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,567,568,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,558,560,568,569,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,559,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,555,560,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,564,568,569,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,564,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,558,560,563,593,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,557,560,567,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,560,581,593,639,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3364,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,634,3336,3340,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,634,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3333,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,634,3333,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,634,3328,3329,3332,3335,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3343,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3334,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3357,3358,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,634,641,3332,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3357,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3330,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3330,3331,3332,3333,3334,3335,3336,3337,3338,3340,3341,3342,3343,3344,3345,3346,3347,3348,3349,3350,3351,3352,3353,3354,3355,3356,3358,3359,3360,3361,3362,3363,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,3336,3351,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3343,3344,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3334,3336,3344,3345,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3335,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3331,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3340,3344,3345,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3340,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,634,3334,3336,3339,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3333,3336,3343,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3351,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,639,641,3331,3336,3357,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2249,2250,2251,2252,2253,2254,2255,2257,2258,2259,2260,2261,2262,2263,2264,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2251,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2251,2256,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2574,2575,2576,3242,3243,3244,3245,3246,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3238,3239,3240,3241,3242,3243,3244,3245,3246,3247,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3238,3239,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,3244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3238,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3239,3240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3239,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3238,3239,3240,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,3244,3245,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,438,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,438,439,532,546,593,651,1046,1561,1591,1608,1609,1610,1614,1664,1675,1688,1690,1716,1719,1721,1724,1725,1726,1727,1728,1730,1732,1740,2272,2507,2578,2585,2626,3155,3200,3205,3221,3237,3248,3304,3312,3316,3323,3601,3649,3662,3668,3776,3789,3795,3809,4024,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4329,4343,4347,4350],[398,546,593,1703,3152,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3146,3147,3148,3149,3150,3151,3152,3153,3154,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3144,3146,3147,3148,3149,3150,3151,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4558],[546,593,1037,3146,3149,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3144,3147,3148,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3145,3147,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3145,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,3190,3191,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1589,3191,3192,3201,3202,3203,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,2624,3190,3191,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1601,3190,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,651,1037,3191,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,1716,3127,3165,3193,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1611,1701,1724,3130,3164,3193,3194,3197,3200,3204,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,651,1611,1723,2574,3117,3118,3126,3164,3165,3189,3190,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2574,3113,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1701,1702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,2574,2575,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,2566,3157,3164,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,1701,2574,3157,3164,3196,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,1701,3157,3164,3196,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,2498,2500,2501,2502,2503,2504,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1589,1608,1664,1733,2272,2276,2278,2420,2498,2499,2501,2502,2505,2506,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,1589,1591,1602,1603,1646,1664,1733,2273,2274,2275,2276,2418,2419,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,635,651,1602,2241,2420,2497,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2273,2274,2275,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2277,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2273,2274,2275,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2276,2277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,605,1037,1046,2241,2276,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2275,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1590,1591,1602,2273,2276,2277,2278,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,614,635,651,1037,1046,1646,1664,1733,2239,2241,2273,2276,2419,2420,2497,2499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,635,651,1037,1046,1602,1646,1664,1733,2241,2277,2278,2419,2420,2497,2500,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3125,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,2589,2590,3125,3134,3135,3313,3314,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,1675,3124,3125,3134,3135,3313,3314,3315,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,3124,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,532,546,593,3124,3125,3314,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4563],[398,546,593,1675,3124,3125,3134,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3124,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3124,3125,3134,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3811,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1759,1760,2240,2244,2245,2246,2247,3814,3815,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1688,1759,1760,2245,2246,2247,3810,3811,3812,3813,3814,3816,4023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1048,1688,1760,2240,3810,3811,3812,3813,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2243,3814,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,2239,4022,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1717,2240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2508,2509,2510,2511,2512,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,2512,2569,2577,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,2512,2536,2537,2566,2567,2568,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4328],[398,437,546,593,1601,1703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344,4345,4346],[398,546,593,651,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1611,1613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,532,546,593,1611,1612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4332],[546,593,2417,2620,3119,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4328],[398,546,593,1037,1046,1646,1664,2572,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,2589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,1716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1716,2536,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1601,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4549,4550],[180,247,398,546,593,1601,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4573,4574],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4573],[398,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344],[398,532,546,593,1601,1731,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4576],[398,437,546,593,1703,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4348,4349],[546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,3119,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3113,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1561,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,651,3594,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1689,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1732,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,3131,3132,3133,3136,3137,3138,3139,3140,3141,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,2570,2571,2572,2573,2574,3136,3138,3140,3141,3142,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1675,2570,2571,2573,3121,3122,3124,3125,3132,3133,3135,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3131,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2572,2573,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2570,2573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2571,2572,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2573,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2571,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2566,3164,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2570,2571,3139,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2570,2571,2573,3137,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2576,2579,2580,2581,2582,2583,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2579,2580,2581,2582,2583,2584,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3605,3622,3623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3603,3604,3605,3620,3621,3622,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3609,3620,3621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3634,3635,3636,3637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1675,1733,1734,3629,3630,3631,3632,3633,3634,3635,3636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,3629,3631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,3610,3611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3614,3615,3617,3618,3622,3625,3626,3627,3635,3636,3642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3614,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3609,3625,3626,3627,3628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1664,1733,3602,3603,3605,3607,3609,3612,3613,3616,3619,3621,3623,3624,3628,3630,3632,3634,3637,3638,3639,3640,3641,3643,3644,3645,3646,3647,3648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1733,3607,3608,3609,3616,3619,3623,3624,3625,3626,3627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3605,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3603,3604,3609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3631,3634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3629,3634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3630,3632,3633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3606,3609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3602,3605,3607,3608,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3602,3603,3605,3607,3609,3612,3613,3621,3630,3632,3634,3647,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3610,3611,3613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3604,3606,3608,3610,3611,3620,3629,3631,3633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1733,3606,3607,3608,3609,3616,3623,3638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3613,3614,3615,3616,3642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3610,3611,3612,3613,3614,3615,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3186,3619,3620,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3187,3602,3617,3618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3602,3617,3618,3619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,3620,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,3318,3319,3320,3321,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,3319,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3318,3319,3320,3321,3322,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3652,3653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3654,3655,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,3652,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,3654,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3656,3657,3658,3659,3660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3650,3651,3652,3653,3654,3655,3656,3657,3658,3659,3660,3661,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3656,3657,3659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3650,3651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2536,2537,2566,2567,2568,3164,3169,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,614,2911,3107,3112,3130,3143,3155,3205,3211,3213,3214,3215,3216,3217,3219,3220,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3209,3210,3212,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,2574,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,3219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3126,3206,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3211,3213,3214,3215,3216,3217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2911,3112,3126,3136,3152,3206,3207,3208,3209,3210,3212,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2911,3121,3126,3136,3152,3206,3207,3208,3209,3210,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2911,3112,3206,3207,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3136,3206,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2573,2574,3126,3136,3147,3152,3218,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3206,3207,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2574,2911,3206,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1037,1601,1645,1703,1716,1718,1719,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1718,1720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,614,1037,1645,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3666,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3663,3664,3665,3666,3667,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3663,3664,3665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4331,4332],[398,437,546,593,1601,1703,2574,2575,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4331,4332,4334,4335,4336,4337,4338,4339,4341],[398,546,593,1046,1703,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4330,4331,4333,4341,4342],[398,546,593,651,1037,1046,1048,1688,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4330,4331,4335,4337,4338,4339,4340],[398,437,532,546,593,595,613,1645,1719,1722,1731,1732,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4351,4352,4548,4551,4552,4553,4554,4555,4556],[398,437,546,593,1703,1716,1759,2240,2266,2267,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,1664,1733,1760,2241,2242,2245,2246,2247,2248,2266,2268,2269,2271,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1760,2240,2241,2242,2245,2246,2247,2248,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,607,609,1037,1046,1760,2241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2243,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,1760,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,607,609,613,614,1037,1046,1646,1664,1733,1760,2240,2241,2270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,623,651,1760,2239,2240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,614,2240,2243,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1717,2240,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1645,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1675,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1664,1675,1725,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590,4591,4592,4593,4594],[398,546,593,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,3298,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,3223,3232,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3222,3223,3224,3232,3233,3234,3235,3236,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3225,3226,3227,3228,3229,3230,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3222,3223,3224,3231,3232,3233,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3222,3231,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3773,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3768,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3770,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3769,3770,3771,3772,3773,3774,3775,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3769,3771,3772,3774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1590,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1591,1601,1603,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1591,1602,1603,1604,1605,1606,1607,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1602,1605,1606,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,613,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1602,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3166,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,3188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,614,651,1646,1664,1733,3186,3187,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3166,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,3127,3167,3168,3189,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2536,2537,2566,2567,2568,3166,3169,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1664,1733,3164,3166,3167,3170,3171,3172,3188,3189,3198,3199,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1675,1734,2624,3166,3167,3168,3170,3171,3172,3188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3167,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3249,3300,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2265,3249,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,3251,3299,3300,3301,3302,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3250,3251,3299,3300,3301,3302,3303,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3250,3251,3299,3300,3301,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,3298,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1717,3783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3782,3783,3785,3786,3787,3788,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,1717,3782,3783,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4025,4026,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4028,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4027,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4605],[546,593,1037,2574,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,1716,2574,2575,2576,4025,4026,4027,4029,4030,4031,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4317,4318],[398,546,593,1046,1611,1664,1733,2574,3130,4025,4026,4028,4029,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4315,4316,4318,4319,4320,4321,4322,4323,4324,4325,4326,4327,4328],[398,546,593,1037,1046,1611,1612,2574,3118,4025,4026,4027,4028,4029,4030,4031,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4315],[398,546,593,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320],[398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320,4321],[398,546,593,1037,1046,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320],[398,546,593,1037,1046,1646,1664,1733,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317,4318,4320],[398,437,546,593,1601,1716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4322],[398,546,593,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317,4318,4320,4321],[437,546,593,2417,2586,2620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2586,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1646,1664,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2586,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,2623,2624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,1716,2587,2588,2589,2590,2591,2592,2593,2621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1664,1733,2588,2591,2592,2593,2622,2623,2625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,2586,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1580,1589,1646,1664,1733,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3309,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,3309,3310,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4611],[398,546,593,3306,3308,3309,3310,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3305,3306,3307,3308,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3305,3306,3307,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3594,3597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1716,3597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3594,3595,3596,3597,3598,3599,3600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3124,3125,3594,3595,3596,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3790,3791,3792,3793,3794,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,1722,1723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,651,1645,1722,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1048,1675,1688,1734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1675,1733,1734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1675,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1664,1733,1735,1736,1737,1738,1739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1675,1734,1735,1736,1737,1738,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3122,3796,3797,3798,3799,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,3798,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3799,3800,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,3806,3807,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,2574,2575,2576,3796,3800,3801,3802,3803,3804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3796,3797,3798,3799,3801,3802,3803,3804,3805,3806,3808,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,3797,3798,3799,3800,3801,3802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3114,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2572,2573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,1716,2574,2575,2576,3114,3115,3126,3127,3128,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,1664,1703,1733,2572,2574,2576,3126,3128,3129,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,1675,2574,3114,3115,3117,3118,3120,3121,3122,3123,3124,3125,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314]],"fileInfos":[{"version":"bcd24271a113971ba9eb71ff8cb01bc6b0f872a85c23fdbe5d93065b375933cd","affectsGlobalScope":true,"impliedFormat":1},{"version":"3f88bedbeb09c6f5a6645cb24c7c55f1aa22d19ae96c8e6959cbd8b85a707bc6","impliedFormat":1},{"version":"7fe93b39b810eadd916be8db880dd7f0f7012a5cc6ffb62de8f62a2117fa6f1f","impliedFormat":1},{"version":"bb0074cc08b84a2374af33d8bf044b80851ccc9e719a5e202eacf40db2c31600","impliedFormat":1},{"version":"1a7daebe4f45fb03d9ec53d60008fbf9ac45a697fdc89e4ce218bc94b94f94d6","impliedFormat":1},{"version":"f94b133a3cb14a288803be545ac2683e0d0ff6661bcd37e31aaaec54fc382aed","impliedFormat":1},{"version":"f59d0650799f8782fd74cf73c19223730c6d1b9198671b1c5b3a38e1188b5953","impliedFormat":1},{"version":"8a15b4607d9a499e2dbeed9ec0d3c0d7372c850b2d5f1fb259e8f6d41d468a84","impliedFormat":1},{"version":"d6b1eba8496bdd0eed6fc8a685768fe01b2da4a0388b5fe7df558290bffcf32f","affectsGlobalScope":true,"impliedFormat":1},{"version":"eadcffda2aa84802c73938e589b9e58248d74c59cb7fcbca6474e3435ac15504","affectsGlobalScope":true,"impliedFormat":1},{"version":"105ba8ff7ba746404fe1a2e189d1d3d2e0eb29a08c18dded791af02f29fb4711","affectsGlobalScope":true,"impliedFormat":1},{"version":"00343ca5b2e3d48fa5df1db6e32ea2a59afab09590274a6cccb1dbae82e60c7c","affectsGlobalScope":true,"impliedFormat":1},{"version":"ebd9f816d4002697cb2864bea1f0b70a103124e18a8cd9645eeccc09bdf80ab4","affectsGlobalScope":true,"impliedFormat":1},{"version":"2c1afac30a01772cd2a9a298a7ce7706b5892e447bb46bdbeef720f7b5da77ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"7b0225f483e4fa685625ebe43dd584bb7973bbd84e66a6ba7bbe175ee1048b4f","affectsGlobalScope":true,"impliedFormat":1},{"version":"c0a4b8ac6ce74679c1da2b3795296f5896e31c38e888469a8e0f99dc3305de60","affectsGlobalScope":true,"impliedFormat":1},{"version":"3084a7b5f569088e0146533a00830e206565de65cae2239509168b11434cd84f","affectsGlobalScope":true,"impliedFormat":1},{"version":"c5079c53f0f141a0698faa903e76cb41cd664e3efb01cc17a5c46ec2eb0bef42","affectsGlobalScope":true,"impliedFormat":1},{"version":"32cafbc484dea6b0ab62cf8473182bbcb23020d70845b406f80b7526f38ae862","affectsGlobalScope":true,"impliedFormat":1},{"version":"fca4cdcb6d6c5ef18a869003d02c9f0fd95df8cfaf6eb431cd3376bc034cad36","affectsGlobalScope":true,"impliedFormat":1},{"version":"b93ec88115de9a9dc1b602291b85baf825c85666bf25985cc5f698073892b467","affectsGlobalScope":true,"impliedFormat":1},{"version":"f5c06dcc3fe849fcb297c247865a161f995cc29de7aa823afdd75aaaddc1419b","affectsGlobalScope":true,"impliedFormat":1},{"version":"b77e16112127a4b169ef0b8c3a4d730edf459c5f25fe52d5e436a6919206c4d7","affectsGlobalScope":true,"impliedFormat":1},{"version":"fbffd9337146eff822c7c00acbb78b01ea7ea23987f6c961eba689349e744f8c","affectsGlobalScope":true,"impliedFormat":1},{"version":"a995c0e49b721312f74fdfb89e4ba29bd9824c770bbb4021d74d2bf560e4c6bd","affectsGlobalScope":true,"impliedFormat":1},{"version":"c7b3542146734342e440a84b213384bfa188835537ddbda50d30766f0593aff9","affectsGlobalScope":true,"impliedFormat":1},{"version":"ce6180fa19b1cccd07ee7f7dbb9a367ac19c0ed160573e4686425060b6df7f57","affectsGlobalScope":true,"impliedFormat":1},{"version":"3f02e2476bccb9dbe21280d6090f0df17d2f66b74711489415a8aa4df73c9675","affectsGlobalScope":true,"impliedFormat":1},{"version":"45e3ab34c1c013c8ab2dc1ba4c80c780744b13b5676800ae2e3be27ae862c40c","affectsGlobalScope":true,"impliedFormat":1},{"version":"805c86f6cca8d7702a62a844856dbaa2a3fd2abef0536e65d48732441dde5b5b","affectsGlobalScope":true,"impliedFormat":1},{"version":"e42e397f1a5a77994f0185fd1466520691456c772d06bf843e5084ceb879a0ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"f4c2b41f90c95b1c532ecc874bd3c111865793b23aebcc1c3cbbabcd5d76ffb0","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab26191cfad5b66afa11b8bf935ef1cd88fabfcb28d30b2dfa6fad877d050332","affectsGlobalScope":true,"impliedFormat":1},{"version":"2088bc26531e38fb05eedac2951480db5309f6be3fa4a08d2221abb0f5b4200d","affectsGlobalScope":true,"impliedFormat":1},{"version":"cb9d366c425fea79716a8fb3af0d78e6b22ebbab3bd64d25063b42dc9f531c1e","affectsGlobalScope":true,"impliedFormat":1},{"version":"500934a8089c26d57ebdb688fc9757389bb6207a3c8f0674d68efa900d2abb34","affectsGlobalScope":true,"impliedFormat":1},{"version":"689da16f46e647cef0d64b0def88910e818a5877ca5379ede156ca3afb780ac3","affectsGlobalScope":true,"impliedFormat":1},{"version":"bc21cc8b6fee4f4c2440d08035b7ea3c06b3511314c8bab6bef7a92de58a2593","affectsGlobalScope":true,"impliedFormat":1},{"version":"7ca53d13d2957003abb47922a71866ba7cb2068f8d154877c596d63c359fed25","affectsGlobalScope":true,"impliedFormat":1},{"version":"54725f8c4df3d900cb4dac84b64689ce29548da0b4e9b7c2de61d41c79293611","affectsGlobalScope":true,"impliedFormat":1},{"version":"e5594bc3076ac29e6c1ebda77939bc4c8833de72f654b6e376862c0473199323","affectsGlobalScope":true,"impliedFormat":1},{"version":"2f3eb332c2d73e729f3364fcc0c2b375e72a121e8157d25a82d67a138c83a95c","affectsGlobalScope":true,"impliedFormat":1},{"version":"6f4427f9642ce8d500970e4e69d1397f64072ab73b97e476b4002a646ac743b1","affectsGlobalScope":true,"impliedFormat":1},{"version":"48915f327cd1dea4d7bd358d9dc7732f58f9e1626a29cc0c05c8c692419d9bb7","affectsGlobalScope":true,"impliedFormat":1},{"version":"b7bf9377723203b5a6a4b920164df22d56a43f593269ba6ae1fdc97774b68855","affectsGlobalScope":true,"impliedFormat":1},{"version":"db9709688f82c9e5f65a119c64d835f906efe5f559d08b11642d56eb85b79357","affectsGlobalScope":true,"impliedFormat":1},{"version":"4b25b8c874acd1a4cf8444c3617e037d444d19080ac9f634b405583fd10ce1f7","affectsGlobalScope":true,"impliedFormat":1},{"version":"37be57d7c90cf1f8112ee2636a068d8fd181289f82b744160ec56a7dc158a9f5","affectsGlobalScope":true,"impliedFormat":1},{"version":"a917a49ac94cd26b754ab84e113369a75d1a47a710661d7cd25e961cc797065f","affectsGlobalScope":true,"impliedFormat":1},{"version":"1ce14b81c5cc821994aa8ec1d42b220dd41b27fcc06373bce3958af7421b77d4","affectsGlobalScope":true,"impliedFormat":1},{"version":"b3a048b3e9302ef9a34ef4ebb9aecfb28b66abb3bce577206a79fee559c230da","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d6d51a5118d000ed3bfe6e1dd1335bebfff3fef23cd2af2f84a24d30f90cc90","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d8dedbec739bc79642c1e96e9bfc0b83b25b104a0486aebf016fc7b85b39f48","impliedFormat":1},{"version":"e89535c3ec439608bcd0f68af555d0e5ddf121c54abe69343549718bd7506b9c","impliedFormat":1},{"version":"622a984b60c294ffb2f9152cf1d4d12e91d2b733d820eec949cf54d63a3c1025","impliedFormat":1},{"version":"81aae92abdeaccd9c1723cef39232c90c1aed9d9cf199e6e2a523b7d8e058a11","impliedFormat":1},{"version":"a63a6c6806a1e519688ef7bd8ca57be912fc0764485119dbd923021eb4e79665","impliedFormat":1},{"version":"75b57b109d774acca1e151df21cf5cb54c7a1df33a273f0457b9aee4ebd36fb9","impliedFormat":1},{"version":"073ca26c96184db9941b5ec0ddea6981c9b816156d9095747809e524fdd90e35","impliedFormat":1},{"version":"e41d17a2ec23306d953cda34e573ed62954ca6ea9b8c8b74e013d07a6886ce47","impliedFormat":1},{"version":"241bd4add06f06f0699dcd58f3b334718d85e3045d9e9d4fa556f11f4d1569c1","impliedFormat":1},{"version":"2ae3787e1498b20aad1b9c2ee9ea517ec30e89b70d242d8e3e52d1e091039695","impliedFormat":1},{"version":"c7c72c4cffb1bc83617eefed71ed68cc89df73cab9e19507ccdecb3e72b4967e","affectsGlobalScope":true,"impliedFormat":1},{"version":"b8bff8a60af0173430b18d9c3e5c443eaa3c515617210c0c7b3d2e1743c19ecb","impliedFormat":1},{"version":"38b38db08e7121828294dec10957a7a9ff263e33e2a904b346516d4a4acca482","impliedFormat":1},{"version":"a76ebdf2579e68e4cfe618269c47e5a12a4e045c2805ed7f7ab37af8daa6b091","impliedFormat":1},{"version":"8a2aaea564939c22be05d665cc955996721bad6d43148f8fa21ae8f64afecd37","impliedFormat":1},{"version":"e59d36b7b6e8ba2dd36d032a5f5c279d2460968c8b4e691ca384f118fb09b52a","impliedFormat":1},{"version":"e96885c0684c9042ec72a9a43ef977f6b4b4a2728f4b9e737edcbaa0c74e5bf6","impliedFormat":1},{"version":"95950a187596e206d32d5d9c7b932901088c65ed8f9040e614aa8e321e0225ef","impliedFormat":1},{"version":"89e061244da3fc21b7330f4bd32f47c1813dd4d7f1dc3d0883d88943f035b993","impliedFormat":1},{"version":"e46558c2e04d06207b080138678020448e7fc201f3d69c2601b0d1456105f29a","impliedFormat":1},{"version":"71549375db52b1163411dba383b5f4618bdf35dc57fa327a1c7d135cf9bf67d1","impliedFormat":1},{"version":"7e6b2d61d6215a4e82ea75bc31a80ebb8ad0c2b37a60c10c70dd671e8d9d6d5d","impliedFormat":1},{"version":"78bea05df2896083cca28ed75784dde46d4b194984e8fc559123b56873580a23","impliedFormat":1},{"version":"5dd04ced37b7ea09f29d277db11f160df7fd73ba8b9dba86cb25552e0653a637","impliedFormat":1},{"version":"f74b81712e06605677ae1f061600201c425430151f95b5ef4d04387ad7617e6a","impliedFormat":1},{"version":"9a72847fcf4ac937e352d40810f7b7aec7422d9178451148296cf1aa19467620","impliedFormat":1},{"version":"3ae18f60e0b96fa1e025059b7d25b3247ba4dcb5f4372f6d6e67ce2adac74eac","impliedFormat":1},{"version":"2b9260f44a2e071450ae82c110f5dc8f330c9e5c3e85567ed97248330f2bf639","impliedFormat":1},{"version":"4f196e13684186bda6f5115fc4677a87cf84a0c9c4fc17b8f51e0984f3697b6d","impliedFormat":1},{"version":"61419f2c5822b28c1ea483258437c1faab87d00c6f84481aa22afb3380d8e9a4","impliedFormat":1},{"version":"64479aee03812264e421c0bf5104a953ca7b02740ba80090aead1330d0effe91","impliedFormat":1},{"version":"0521108c9f8ddb17654a0a54dae6ba9667c99eddccfd6af5748113e022d1c37a","impliedFormat":1},{"version":"c5570e504be103e255d80c60b56c367bf45d502ca52ee35c55dec882f6563b5c","impliedFormat":1},{"version":"ee764e6e9a7f2b987cc1a2c0a9afd7a8f4d5ebc4fdb66ad557a7f14a8c2bd320","impliedFormat":1},{"version":"0520b5093712c10c6ef23b5fea2f833bf5481771977112500045e5ea7e8e2b69","impliedFormat":1},{"version":"5c3cf26654cf762ac4d7fd7b83f09acfe08eef88d2d6983b9a5a423cb4004ca3","impliedFormat":1},{"version":"e60fa19cf7911c1623b891155d7eb6b7e844e9afdf5738e3b46f3b687730a2bd","impliedFormat":1},{"version":"b1fd72ff2bb0ba91bb588f3e5329f8fc884eb859794f1c4657a2bfa122ae54d0","impliedFormat":1},{"version":"6cf42a4f3cfec648545925d43afaa8bb364ac10a839ffed88249da109361b275","impliedFormat":1},{"version":"d7058e75920120b142a9d57be25562a3cd9a936269fd52908505f530105f2ec4","impliedFormat":1},{"version":"6df52b70d7f7702202f672541a5f4a424d478ee5be51a9d37b8ccbe1dbf3c0f2","impliedFormat":1},{"version":"0ca7f997e9a4d8985e842b7c882e521b6f63233c4086e9fe79dd7a9dc4742b5e","impliedFormat":1},{"version":"91046b5c6b55d3b194c81fd4df52f687736fad3095e9d103ead92bb64dc160ee","impliedFormat":1},{"version":"db5704fdad56c74dfc5941283c1182ed471bd17598209d3ac4a49faa72e43cfc","impliedFormat":1},{"version":"758e8e89559b02b81bc0f8fd395b17ad5aff75490c862cbe369bb1a3d1577c40","impliedFormat":1},{"version":"2ee64342c077b1868f1834c063f575063051edd6e2964257d34aad032d6b657c","impliedFormat":1},{"version":"6f6b4b3d670b6a5f0e24ea001c1b3d36453c539195e875687950a178f1730fa7","impliedFormat":1},{"version":"a472a1d3f25ce13a1d44911cd3983956ac040ce2018e155435ea34afb25f864c","impliedFormat":1},{"version":"b48b83a86dd9cfe36f8776b3ff52fcd45b0e043c0538dc4a4b149ba45fe367b9","impliedFormat":1},{"version":"792de5c062444bd2ee0413fb766e57e03cce7cdaebbfc52fc0c7c8e95069c96b","impliedFormat":1},{"version":"a79e3e81094c7a04a885bad9b049c519aace53300fb8a0fe4f26727cb5a746ce","impliedFormat":1},{"version":"93181bac0d90db185bb730c95214f6118ae997fe836a98a49664147fbcaf1988","impliedFormat":1},{"version":"8a4e89564d8ea66ad87ee3762e07540f9f0656a62043c910d819b4746fc429c5","impliedFormat":1},{"version":"b9011d99942889a0f95e120d06b698c628b0b6fdc3e6b7ecb459b97ed7d5bcc6","impliedFormat":1},{"version":"4d639cbbcc2f8f9ce6d55d5d503830d6c2556251df332dc5255d75af53c8a0e7","impliedFormat":1},{"version":"cdb48277f600ab5f429ecf1c5ea046683bc6b9f73f3deab9a100adac4b34969c","impliedFormat":1},{"version":"75be84956a29040a1afbe864c0a7a369dfdb739380072484eff153905ef867ee","impliedFormat":1},{"version":"b06b4adc2ae03331a92abd1b19af8eb91ec2bf8541747ee355887a167d53145e","impliedFormat":1},{"version":"c54166a85bd60f86d1ebb90ce0117c0ecb850b8a33b366691629fdf26f1bbbd8","impliedFormat":1},{"version":"0d417c15c5c635384d5f1819cc253a540fe786cc3fda32f6a2ae266671506a21","impliedFormat":1},{"version":"80f23f1d60fbed356f726b3b26f9d348dddbb34027926d10d59fad961e70a730","impliedFormat":1},{"version":"cb59317243a11379a101eb2f27b9df1022674c3df1df0727360a0a3f963f523b","impliedFormat":1},{"version":"cc20bb2227dd5de0aab0c8d697d1572f8000550e62c7bf5c92f212f657dd88c5","impliedFormat":1},{"version":"06b8a7d46195b6b3980e523ef59746702fd210b71681a83a5cf73799623621f9","impliedFormat":1},{"version":"860e4405959f646c101b8005a191298b2381af8f33716dc5f42097e4620608f8","impliedFormat":1},{"version":"f7e32adf714b8f25d3c1783473abec3f2e82d5724538d8dcf6f51baaaff1ca7a","impliedFormat":1},{"version":"d0da80c845999a16c24d0783033fb5366ada98df17867c98ad433ede05cd87fd","impliedFormat":1},{"version":"bfbf80f9cd4558af2d7b2006065340aaaced15947d590045253ded50aabb9bc5","impliedFormat":1},{"version":"fd9a991b51870325e46ebb0e6e18722d313f60cd8e596e645ec5ac15b96dbf4e","impliedFormat":1},{"version":"c3bd2b94e4298f81743d92945b80e9b56c1cdfb2bef43c149b7106a2491b1fc9","impliedFormat":1},{"version":"a246cce57f558f9ebaffd55c1e5673da44ea603b4da3b2b47eb88915d30a9181","impliedFormat":1},{"version":"d993eacc103c5a065227153c9aae8acea3a4322fe1a169ee7c70b77015bf0bb2","impliedFormat":1},{"version":"fc2b03d0c042aa1627406e753a26a1eaad01b3c496510a78016822ef8d456bb6","impliedFormat":1},{"version":"063c7ebbe756f0155a8b453f410ca6b76ffa1bbc1048735bcaf9c7c81a1ce35f","impliedFormat":1},{"version":"314e402cd481370d08f63051ae8b8c8e6370db5ee3b8820eeeaaf8d722a6dac6","impliedFormat":1},{"version":"9669075ac38ce36b638b290ba468233980d9f38bdc62f0519213b2fd3e2552ec","impliedFormat":1},{"version":"4d123de012c24e2f373925100be73d50517ac490f9ed3578ac82d0168bfbd303","impliedFormat":1},{"version":"656c9af789629aa36b39092bee3757034009620439d9a39912f587538033ce28","impliedFormat":1},{"version":"3ac3f4bdb8c0905d4c3035d6f7fb20118c21e8a17bee46d3735195b0c2a9f39f","impliedFormat":1},{"version":"1f453e6798ed29c86f703e9b41662640d4f2e61337007f27ac1c616f20093f69","impliedFormat":1},{"version":"af43b7871ff21c62bf1a54ec5c488e31a8d3408d5b51ff2e9f8581b6c55f2fc7","impliedFormat":1},{"version":"70550511d25cbb0b6a64dcac7fffc3c1397fd4cbeb6b23ccc7f9b794ab8a6954","impliedFormat":1},{"version":"af0fbf08386603a62f2a78c42d998c90353b1f1d22e05a384545f7accf881e0a","impliedFormat":1},{"version":"cefc20054d20b85b534206dbcedd509bb74f87f3d8bc45c58c7be3a76caa45e1","impliedFormat":1},{"version":"ad6eee4877d0f7e5244d34bc5026fd6e9cf8e66c5c79416b73f9f6ebf132f924","impliedFormat":1},{"version":"4888fd2bcfee9a0ce89d0df860d233e0cee8ee9c479b6bd5a5d5f9aae98342fe","impliedFormat":1},{"version":"f4749c102ced952aa6f40f0b579865429c4869f6d83df91000e98005476bee87","impliedFormat":1},{"version":"56654d2c5923598384e71cb808fac2818ca3f07dd23bb018988a39d5e64f268b","impliedFormat":1},{"version":"8b6719d3b9e65863da5390cb26994602c10a315aa16e7d70778a63fee6c4c079","impliedFormat":1},{"version":"05f56cd4b929977d18df8f3d08a4c929a2592ef5af083e79974b20a063f30940","impliedFormat":1},{"version":"547d3c406a21b30e2b78629ecc0b2ddaf652d9e0bdb2d59ceebce5612906df33","impliedFormat":1},{"version":"b3a4f9385279443c3a5568ec914a9492b59a723386161fd5ef0619d9f8982f97","impliedFormat":1},{"version":"3fe66aba4fbe0c3ba196a4f9ed2a776fe99dc4d1567a558fb11693e9fcc4e6ed","impliedFormat":1},{"version":"140eef237c7db06fc5adcb5df434ee21e81ee3a6fd57e1a75b8b3750aa2df2d8","impliedFormat":1},{"version":"0944ec553e4744efae790c68807a461720cff9f3977d4911ac0d918a17c9dd99","impliedFormat":1},{"version":"cb46b38d5e791acaa243bf342b8b5f8491639847463ac965b93896d4fb0af0d9","impliedFormat":1},{"version":"7c7d9e116fe51100ff766703e6b5e4424f51ad8977fe474ddd8d0959aa6de257","impliedFormat":1},{"version":"af70a2567e586be0083df3938b6a6792e6821363d8ef559ad8d721a33a5bcdaf","impliedFormat":1},{"version":"006cff3a8bcb92d77953f49a94cd7d5272fef4ab488b9052ef82b6a1260d870b","impliedFormat":1},{"version":"7d44bfdc8ee5e9af70738ff652c622ae3ad81815e63ab49bdc593d34cb3a68e5","impliedFormat":1},{"version":"339814517abd4dbc7b5f013dfd3b5e37ef0ea914a8bbe65413ecffd668792bc6","impliedFormat":1},{"version":"34d5bc0a6958967ec237c99f980155b5145b76e6eb927c9ffc57d8680326b5d8","impliedFormat":1},{"version":"9eae79b70c9d8288032cbe1b21d0941f6bd4f315e14786b2c1d10bccc634e897","impliedFormat":1},{"version":"18ce015ed308ea469b13b17f99ce53bbb97975855b2a09b86c052eefa4aa013a","impliedFormat":1},{"version":"5a931bc4106194e474be141e0bc1046629510dc95b9a0e4b02a3783847222965","impliedFormat":1},{"version":"5e5f371bf23d5ced2212a5ff56675aefbd0c9b3f4d4fdda1b6123ac6e28f058c","impliedFormat":1},{"version":"907c17ad5a05eecb29b42b36cc8fec6437be27cc4986bb3a218e4f74f606911c","impliedFormat":1},{"version":"ce60a562cd2a92f37a88f2ddd99a3abfbc5848d7baf38c48fb8d3243701fcb75","impliedFormat":1},{"version":"a726ad2d0a98bfffbe8bc1cd2d90b6d831638c0adc750ce73103a471eb9a891c","impliedFormat":1},{"version":"f44c0c8ce58d3dacac016607a1a90e5342d830ea84c48d2e571408087ae55894","impliedFormat":1},{"version":"75a315a098e630e734d9bc932d9841b64b30f7a349a20cf4717bf93044eff113","impliedFormat":1},{"version":"9131d95e32b3d4611d4046a613e022637348f6cebfe68230d4e81b691e4761a1","impliedFormat":1},{"version":"b03aa292cfdcd4edc3af00a7dbd71136dd067ec70a7536b655b82f4dd444e857","impliedFormat":1},{"version":"b6e2b0448ced813b8c207810d96551a26e7d7bb73255eea4b9701698f78846d6","impliedFormat":1},{"version":"8ae10cd85c1bd94d2f2d17c4cbd25c068a4b2471c70c2d96434239f97040747a","impliedFormat":1},{"version":"9ed5b799c50467b0c9f81ddf544b6bcda3e34d92076d6cab183c84511e45c39f","impliedFormat":1},{"version":"b4fa87cc1833839e51c49f20de71230e259c15b2c9c3e89e4814acc1d1ef10de","impliedFormat":1},{"version":"e90ac9e4ac0326faa1bc39f37af38ace0f9d4a655cd6d147713c653139cf4928","impliedFormat":1},{"version":"ea27110249d12e072956473a86fd1965df8e1be985f3b686b4e277afefdde584","impliedFormat":1},{"version":"8776a368617ce51129b74db7d55c3373dadcce5d0701e61d106e99998922a239","impliedFormat":1},{"version":"5666075052877fe2fdddd5b16de03168076cf0f03fbca5c1d4a3b8f43cba570c","impliedFormat":1},{"version":"9108ab5af05418f599ab48186193b1b07034c79a4a212a7f73535903ba4ca249","impliedFormat":1},{"version":"bb4e2cdcadf9c9e6ee2820af23cee6582d47c9c9c13b0dca1baaffe01fbbcb5f","impliedFormat":1},{"version":"6e30d0b5a1441d831d19fe02300ab3d83726abd5141cbcc0e2993fa0efd33db4","impliedFormat":1},{"version":"423f28126b2fc8d8d6fa558035309000a1297ed24473c595b7dec52e5c7ebae5","impliedFormat":1},{"version":"fb30734f82083d4790775dae393cd004924ebcbfde49849d9430bf0f0229dd16","impliedFormat":1},{"version":"2c92b04a7a4a1cd9501e1be338bf435738964130fb2ad5bd6c339ee41224ac4c","impliedFormat":1},{"version":"c5c5f0157b41833180419dacfbd2bcce78fb1a51c136bd4bcba5249864d8b9b5","impliedFormat":1},{"version":"02ae43d5bae42efcd5a00d3923e764895ce056bca005a9f4e623aa6b4797c8af","impliedFormat":1},{"version":"db6e01f17012a9d7b610ae764f94a1af850f5d98c9c826ad61747dca0fb800bd","impliedFormat":1},{"version":"8a44b424edee7bb17dc35a558cc15f92555f14a0441205613e0e50452ab3a602","impliedFormat":1},{"version":"24a00d0f98b799e6f628373249ece352b328089c3383b5606214357e9107e7d5","impliedFormat":1},{"version":"33637e3bc64edd2075d4071c55d60b32bdb0d243652977c66c964021b6fc8066","impliedFormat":1},{"version":"0f0ad9f14dedfdca37260931fac1edf0f6b951c629e84027255512f06a6ebc4c","impliedFormat":1},{"version":"16ad86c48bf950f5a480dc812b64225ca4a071827d3d18ffc5ec1ae176399e36","impliedFormat":1},{"version":"8cbf55a11ff59fd2b8e39a4aa08e25c5ddce46e3af0ed71fb51610607a13c505","impliedFormat":1},{"version":"d5bc4544938741f5daf8f3a339bfbf0d880da9e89e79f44a6383aaf056fe0159","impliedFormat":1},{"version":"97f9169882d393e6f303f570168ca86b5fe9aab556e9a43672dae7e6bb8e6495","impliedFormat":1},{"version":"7c9adb3fcd7851497818120b7e151465406e711d6a596a71b807f3a17853cb58","impliedFormat":1},{"version":"6752d402f9282dd6f6317c8c048aaaac27295739a166eed27e00391b358fed9a","impliedFormat":1},{"version":"9fd7466b77020847dbc9d2165829796bf7ea00895b2520ff3752ffdcff53564b","impliedFormat":1},{"version":"fbfc12d54a4488c2eb166ed63bab0fb34413e97069af273210cf39da5280c8d6","impliedFormat":1},{"version":"85a84240002b7cf577cec637167f0383409d086e3c4443852ca248fc6e16711e","impliedFormat":1},{"version":"84794e3abd045880e0fadcf062b648faf982aa80cfc56d28d80120e298178626","impliedFormat":1},{"version":"053d8b827286a16a669a36ffc8ccc8acdf8cc154c096610aa12348b8c493c7b8","impliedFormat":1},{"version":"3cce4ce031710970fe12d4f7834375f5fd455aa129af4c11eb787935923ff551","impliedFormat":1},{"version":"8f62cbd3afbd6a07bb8c934294b6bfbe437021b89e53a4da7de2648ecfc7af25","impliedFormat":1},{"version":"62c3621d34fb2567c17a2c4b89914ebefbfbd1b1b875b070391a7d4f722e55dc","impliedFormat":1},{"version":"c05ac811542e0b59cb9c2e8f60e983461f0b0e39cea93e320fad447ff8e474f3","impliedFormat":1},{"version":"8e7a5b8f867b99cc8763c0b024068fb58e09f7da2c4810c12833e1ca6eb11c4f","impliedFormat":1},{"version":"132351cbd8437a463757d3510258d0fa98fd3ebef336f56d6f359cf3e177a3ce","impliedFormat":1},{"version":"df877050b04c29b9f8409aa10278d586825f511f0841d1ec41b6554f8362092b","impliedFormat":1},{"version":"33d1888c3c27d3180b7fd20bac84e97ecad94b49830d5dd306f9e770213027d1","impliedFormat":1},{"version":"ee942c58036a0de88505ffd7c129f86125b783888288c2389330168677d6347f","impliedFormat":1},{"version":"a3f317d500c30ea56d41501632cdcc376dae6d24770563a5e59c039e1c2a08ec","impliedFormat":1},{"version":"eb21ddc3a8136a12e69176531197def71dc28ffaf357b74d4bf83407bd845991","impliedFormat":1},{"version":"0c1651a159995dfa784c57b4ea9944f16bdf8d924ed2d8b3db5c25d25749a343","impliedFormat":1},{"version":"aaa13958e03409d72e179b5d7f6ec5c6cc666b7be14773ae7b6b5ee4921e52db","impliedFormat":1},{"version":"0a86e049843ad02977a94bb9cdfec287a6c5a0a4b6b5391a6648b1a122072c5a","impliedFormat":1},{"version":"40f06693e2e3e58526b713c937895c02e113552dc8ba81ecd49cdd9596567ddb","impliedFormat":1},{"version":"4ed5e1992aedb174fb8f5aa8796aa6d4dcb8bd819b4af1b162a222b680a37fa0","impliedFormat":1},{"version":"d7f4bd46a8b97232ea6f8c28012b8d2b995e55e729d11405f159d3e00c51420a","impliedFormat":1},{"version":"d604d413aff031f4bfbdae1560e54ebf503d374464d76d50a2c6ded4df525712","impliedFormat":1},{"version":"e4f4f9cf1e3ac9fd91ada072e4d428ecbf0aa6dc57138fb797b8a0ca3a1d521c","impliedFormat":1},{"version":"12bfd290936824373edda13f48a4094adee93239b9a73432db603127881a300d","impliedFormat":1},{"version":"340ceb3ea308f8e98264988a663640e567c553b8d6dc7d5e43a8f3b64f780374","impliedFormat":1},{"version":"c5a769564e530fba3ec696d0a5cff1709b9095a0bdf5b0826d940d2fc9786413","impliedFormat":1},{"version":"7124ef724c3fc833a17896f2d994c368230a8d4b235baed39aa8037db31de54f","impliedFormat":1},{"version":"5de1c0759a76e7710f76899dcae601386424eab11fb2efaf190f2b0f09c3d3d3","impliedFormat":1},{"version":"9c5ee8f7e581f045b6be979f062a61bf076d362bf89c7f966b993a23424e8b0d","impliedFormat":1},{"version":"1a11df987948a86aa1ec4867907c59bdf431f13ed2270444bf47f788a5c7f92d","impliedFormat":1},{"version":"8018dd2e95e7ce6e613ddd81672a54532614dc745520a2f9e3860ff7fb1be0ca","impliedFormat":1},{"version":"b756781cd40d465da57d1fc6a442c34ae61fe8c802d752aace24f6a43fedacee","impliedFormat":1},{"version":"0fe76167c87289ea094e01616dcbab795c11b56bad23e1ef8aba9aa37e93432a","impliedFormat":1},{"version":"3a45029dba46b1f091e8dc4d784e7be970e209cd7d4ff02bd15270a98a9ba24b","impliedFormat":1},{"version":"032c1581f921f8874cf42966f27fd04afcabbb7878fa708a8251cac5415a2a06","impliedFormat":1},{"version":"69c68ed9652842ce4b8e495d63d2cd425862104c9fb7661f72e7aa8a9ef836f8","impliedFormat":1},{"version":"0e704ee6e9fd8b6a5a7167886f4d8915f4bc22ed79f19cb7b32bd28458f50643","impliedFormat":1},{"version":"06f62a14599a68bcde148d1efd60c2e52e8fa540cc7dcfa4477af132bb3de271","impliedFormat":1},{"version":"904a96f84b1bcee9a7f0f258d17f8692e6652a0390566515fe6741a5c6db8c1c","impliedFormat":1},{"version":"11f19ce32d21222419cecab448fa335017ebebf4f9e5457c4fa9df42fa2dcca7","impliedFormat":1},{"version":"2e8ee2cbb5e9159764e2189cf5547aebd0e6b0d9a64d479397bb051cd1991744","impliedFormat":1},{"version":"1b0471d75f5adb7f545c1a97c02a0f825851b95fe6e069ac6ecaa461b8bb321d","impliedFormat":1},{"version":"1d157c31a02b1e5cca9bc495b3d8d39f4b42b409da79f863fb953fbe3c7d4884","impliedFormat":1},{"version":"07baaceaec03d88a4b78cb0651b25f1ae0322ac1aa0b555ae3749a79a41cba86","impliedFormat":1},{"version":"619a132f634b4ebe5b4b4179ea5870f62f2cb09916a25957bff17b408de8b56d","impliedFormat":1},{"version":"f60fa446a397eb1aead9c4e568faf2df8068b4d0306ebc075fb4be16ed26b741","impliedFormat":1},{"version":"f3cb784be4d9e91f966a0b5052a098d9b53b0af0d341f690585b0cc05c6ca412","impliedFormat":1},{"version":"350f63439f8fe2e06c97368ddc7fb6d6c676d54f59520966f7dbbe6a4586014e","impliedFormat":1},{"version":"eba613b9b357ac8c50a925fa31dc7e65ff3b95a07efbaa684b624f143d8d34ba","impliedFormat":1},{"version":"45b74185005ed45bec3f07cac6e4d68eaf02ead9ff5a66721679fb28020e5e7c","impliedFormat":1},{"version":"0f6199602df09bdb12b95b5434f5d7474b1490d2cd8cc036364ab3ba6fd24263","impliedFormat":1},{"version":"c8ca7fd9ec7a3ec82185bfc8213e4a7f63ae748fd6fced931741d23ef4ea3c0f","impliedFormat":1},{"version":"5c6a8a3c2a8d059f0592d4eab59b062210a1c871117968b10797dee36d991ef7","impliedFormat":1},{"version":"ad77fd25ece8e09247040826a777dc181f974d28257c9cd5acb4921b51967bd8","impliedFormat":1},{"version":"795a08ae4e193f345073b49f68826ab6a9b280400b440906e4ec5c237ae777e6","impliedFormat":1},{"version":"8153df63cf65122809db17128e5918f59d6bb43a371b5218f4430c4585f64085","impliedFormat":1},{"version":"a8150bc382dd12ce58e00764d2366e1d59a590288ee3123af8a4a2cb4ef7f9df","impliedFormat":1},{"version":"5adfaf2f9f33957264ad199a186456a4676b2724ed700fc313ff945d03372169","impliedFormat":1},{"version":"d5c41a741cd408c34cb91f84468f70e9bda3dfeabf33251a61039b3cdb8b22d8","impliedFormat":1},{"version":"c91d3f9753a311284e76cdcb348cbb50bca98733336ec726b54d77b7361b34de","impliedFormat":1},{"version":"cbaf4a4aa8a8c02aa681c5870d5c69127974de29b7e01df570edec391a417959","impliedFormat":1},{"version":"c7135e329a18b0e712378d5c7bc2faec6f5ab0e955ea0002250f9e232af8b3e4","impliedFormat":1},{"version":"340a45cd77b41d8a6deda248167fa23d3dc67ec798d411bd282f7b3d555b1695","impliedFormat":1},{"version":"fae330f86bc10db6841b310f32367aaa6f553036a3afc426e0389ddc5566cd74","impliedFormat":1},{"version":"cf25d45c02d5fd5d7adb16230a0e1d6715441eef5c0a79a21bfeaa9bbc058939","impliedFormat":1},{"version":"54c3822eaf6436f2eddc92dd6e410750465aba218adbf8ce5d488d773919ec01","impliedFormat":1},{"version":"99d99a765426accf8133737843fb024a154dc6545fc0ffbba968a7c0b848959d","impliedFormat":1},{"version":"c782c5fd5fa5491c827ecade05c3af3351201dd1c7e77e06711c8029b7a9ee4d","impliedFormat":1},{"version":"883d2104e448bb351c49dd9689a7e8117b480b614b2622732655cef03021bf6d","impliedFormat":1},{"version":"d9b00ee2eca9b149663fdba1c1956331841ae296ee03eaaff6c5becbc0ff1ea8","impliedFormat":1},{"version":"09a7e04beb0547c43270b327c067c85a4e2154372417390731dfe092c4350998","impliedFormat":1},{"version":"eee530aaa93e9ec362e3941ee8355e2d073c7b21d88c2af4713e3d701dab8fef","impliedFormat":1},{"version":"28d47319b97dbeee9130b78eae03b2061d46dedbf92b0d9de13ed7ab8399ccd0","impliedFormat":1},{"version":"8b8b92781a6bf150f9ee83f3d8ee278b6cdb98b8308c7ab3413684fc5d9078ef","impliedFormat":1},{"version":"7a0e4cd92545ad03910fd019ae9838718643bd4dde39881c745f236914901dfa","impliedFormat":1},{"version":"c99ebd20316217e349004ee1a0bc74d32d041fb6864093f10f31984c737b8cad","impliedFormat":1},{"version":"6f622e7f054f5ab86258362ac0a64a2d6a27f1e88732d6f5f052f422e08a70e7","impliedFormat":1},{"version":"d62d2ef93ceeb41cf9dfab25989a1e5f9ca5160741aac7f1453c69a6c14c69be","impliedFormat":1},{"version":"1491e80d72873fc586605283f2d9056ee59b166333a769e64378240df130d1c9","impliedFormat":1},{"version":"c32c073d389cfaa3b3e562423e16c2e6d26b8edebbb7d73ccffff4aa66f2171d","impliedFormat":1},{"version":"eca72bf229eecadb63e758613c62fab13815879053539a22477d83a48a21cd73","impliedFormat":1},{"version":"633db46fd1765736409a4767bfc670861468dde60dbb9a501fba4c1b72f8644d","impliedFormat":1},{"version":"689390db63cb282e6d0e5ce9b8f1ec2ec0912d0e2e6dac7235699a15ad17d339","impliedFormat":1},{"version":"f2ee748883723aa9325e5d7f30fce424f6a786706e1b91a5a55237c78ee89c4a","impliedFormat":1},{"version":"d928324d17146fce30b99a28d1d6b48648feac72bbd23641d3ce5ac34aefdfee","impliedFormat":1},{"version":"142f5190d730259339be1433931c0eb31ae7c7806f4e325f8a470bd9221b6533","impliedFormat":1},{"version":"c33a88f2578e8df2fdf36c6a0482bbee615eb3234c8f084ba31a9a96bd306b7f","impliedFormat":1},{"version":"22cca068109eb0e6b4f8acc3fe638d1e6ac277e2044246438763319792b546a1","impliedFormat":1},{"version":"8776e64e6165838ac152fa949456732755b0976d1867ae5534ce248f0ccd7f41","impliedFormat":1},{"version":"66cd33c4151ea27f6e17c6071652eadde9da1b3637dae65fd060212211c695ce","impliedFormat":1},{"version":"5c4c5b49bbb01828402bb04af1d71673b18852c11b7e95bfd5cf4c3d80d352c8","impliedFormat":1},{"version":"7030df3d920343df00324df59dc93a959a33e0f4940af3fefef8c07b7ee329bf","impliedFormat":1},{"version":"a96bc00e0c356e29e620eaec24a56d6dd7f4e304feefcc99066a1141c6fe05a7","impliedFormat":1},{"version":"d12cc0e5b09943c4cd0848f787eb9d07bf78b60798e4588c50582db9d4decc70","impliedFormat":1},{"version":"53b094f1afe442490555eeeb0384fc1ceb487560c83e31f9c64fb934c2dccd94","impliedFormat":1},{"version":"19c3760af3cbc9da99d5b7763b9e33aaf8d018bc2ed843287b7ff4343adf4634","impliedFormat":1},{"version":"9d1e38aeb76084848d2fcd39b458ec88246de028c0f3f448b304b15d764b23d2","impliedFormat":1},{"version":"d406da1eccf18cec56fd29730c24af69758fe3ff49c4f94335e797119cbc0554","impliedFormat":1},{"version":"4898c93890a136da9156c75acd1a80a941a961b3032a0cf14e1fa09a764448b7","impliedFormat":1},{"version":"f5d7a845e3e1c6c27351ea5f358073d0b0681537a2da6201fab254aa434121d3","impliedFormat":1},{"version":"9ddf8e9069327faa75d20135cab675779844f66590249769c3d35dd2a38c2ba9","impliedFormat":1},{"version":"d7c30f0abfe9e197e376b016086cf66b2ffb84015139963f37301ed0da9d3d0d","impliedFormat":1},{"version":"ff75bba0148f07775bcb54bf4823421ed4ebdb751b3bf79cc003bd22e49d7d73","impliedFormat":1},{"version":"d40d20ac633703a7333770bfd60360126fc3302d5392d237bbb76e8c529a4f95","impliedFormat":1},{"version":"35a9867207c488061fb4f6fe4715802fbc164b4400018d2fa0149ad02db9a61c","impliedFormat":1},{"version":"91bf47a209ad0eae090023c3ebc1165a491cf9758799368ffcbee8dbe7448f33","impliedFormat":1},{"version":"0abe2cd72812bbfc509975860277c7cd6f6e0be95d765a9da77fee98264a7e32","impliedFormat":1},{"version":"13286c0c8524606b17a8d68650970bab896fb505f348f71601abf0f2296e8913","impliedFormat":1},{"version":"fc2a131847515b3dff2f0e835633d9a00a9d03ed59e690e27eec85b7b0522f92","impliedFormat":1},{"version":"90433c678bc26751eb7a5d54a2bb0a14be6f5717f69abb5f7a04afc75dce15a4","impliedFormat":1},{"version":"cd0565ace87a2d7802bf4c20ea23a997c54e598b9eb89f9c75e69478c1f7a0b4","impliedFormat":1},{"version":"738020d2c8fc9df92d5dee4b682d35a776eaedfe2166d12bc8f186e1ea57cc52","impliedFormat":1},{"version":"86dd7c5657a0b0bc6bee8002edcfd544458d3d3c60974555746eb9b2583dc35e","impliedFormat":1},{"version":"d97b96b6ecd4ee03f9f1170722c825ef778430a6a0d7aab03b8929012bf773cd","impliedFormat":1},{"version":"f61963dc02ef27c48fb0e0016a413b1e00bcb8b97a3f5d4473cedc7b44c8dc77","impliedFormat":1},{"version":"272dbfe04cfa965d6fff63fdaba415c1b5a515b1881ae265148f8a84ddeb318f","impliedFormat":1},{"version":"2035fb009b5fafa9a4f4e3b3fdb06d9225b89f2cbbf17a5b62413bf72cea721a","impliedFormat":1},{"version":"eefafec7c059f07b885b79b327d381c9a560e82b439793de597441a4e68d774a","impliedFormat":1},{"version":"72636f59b635c378dc9ea5246b9b3517b1214e340e468e54cb80126353053b2e","impliedFormat":1},{"version":"ebb79f267a3bf2de5f8edc1995c5d31777b539935fab8b7d863e8efb06c8e9ea","impliedFormat":1},{"version":"ada033e6a4c7f4e147e6d76bb881069dc66750619f8cc2472d65beeec1100145","impliedFormat":1},{"version":"0c04cc14a807a5dc0e3752d18a3b2655a135fefbf76ddcdabd0c5df037530d41","impliedFormat":1},{"version":"605d29d619180fbec287d1701e8b1f51f2d16747ec308d20aba3e9a0dac43a0f","impliedFormat":1},{"version":"67c19848b442d77c767414084fc571ce118b08301c4ddff904889d318f3a3363","impliedFormat":1},{"version":"c704ff0e0cb86d1b791767a88af21dadfee259180720a14c12baee668d0eb8fb","impliedFormat":1},{"version":"195c50e15d5b3ea034e01fbdca6f8ad4b35ad47463805bb0360bdffd6fce3009","impliedFormat":1},{"version":"da665f00b6877ae4adb39cd548257f487a76e3d99e006a702a4f38b4b39431cb","impliedFormat":1},{"version":"2b82adc9eead34b824a3f4dad315203fbfa56bee0061ccf9b485820606564f70","impliedFormat":1},{"version":"eb47aaa5e1b0a69388bb48422a991b9364a9c206a97983e0227289a9e1fca178","impliedFormat":1},{"version":"d7a4309673b06223537bc9544b1a5fe9425628e1c8ab5605f3c5ebc27ecb8074","impliedFormat":1},{"version":"db2108aea36e7faa83c38f6fe8225b9ad40835c0cba7fa38e969768299b83173","impliedFormat":1},{"version":"3eadfd083d40777b403f4f4eecfa40f93876f2a01779157cc114b2565a7afb51","impliedFormat":1},{"version":"cb6789ce3eba018d5a7996ccbf50e27541d850e9b4ee97fdcb3cbd8c5093691f","impliedFormat":1},{"version":"a3684ea9719122f9477902acd08cd363a6f3cff6d493df89d4dc12fa58204e27","impliedFormat":1},{"version":"2828dabf17a6507d39ebcc58fef847e111dcf2d51b8e4ff0d32732c72be032b3","impliedFormat":1},{"version":"c0c46113b4cd5ec9e7cf56e6dbfb3930ef6cbba914c0883eeced396988ae8320","impliedFormat":1},{"version":"118ea3f4e7b9c12e92551be0766706f57a411b4f18a1b4762cfde3cd6d4f0a96","impliedFormat":1},{"version":"2ad163aaddfa29231a021de6838f59378a210501634f125ed04cfa7d066ffc53","impliedFormat":1},{"version":"6305acbe492b9882ec940f8f0c8e5d1e1395258852f99328efcb1cf1683ca817","impliedFormat":1},{"version":"7619b1f6087a4e9336b2c42bd784b05aa4a2204a364b60171e5a628f817a381e","impliedFormat":1},{"version":"15be9120572c9fbcd3c267bd93b4140354514c9e70734e6fcca65ff4a246f83a","impliedFormat":1},{"version":"412482ab85893cec1d6f26231359474d1f59f6339e2743c08da1b05fc1d12767","impliedFormat":1},{"version":"858e2315e58af0d28fcd7f141a2505aba6a76fd10378ba0ad169b0336fee33fc","impliedFormat":1},{"version":"02da6c1b34f4ae2120d70cf5f9268bf1aedf62e55529d34f5974f5a93655ce38","impliedFormat":1},{"version":"3ecf179ef1cc28f7f9b46c8d2e496d50b542c176e94ed0147bab147b4a961cd6","impliedFormat":1},{"version":"b145da03ce7e174af5ced2cbbd16e96d3d5c2212f9a90d3657b63a5650a73b7f","impliedFormat":1},{"version":"c7aadab66a2bc90eeb0ab145ca4daebcbc038e24359263de3b40e7b1c7affba6","impliedFormat":1},{"version":"99518dc06286877a7b716e0f22c1a72d3c62be42701324b49f27bcc03573efff","impliedFormat":1},{"version":"f4575fd196a7e33c7be9773a71bcc5fbe7182a2152be909f6b8e8e7ba2438f06","impliedFormat":1},{"version":"05cba5acd77a4384389b9c62739104b5a1693efd66e6abac6c5ffc53280ae777","impliedFormat":1},{"version":"acacda82ebd929fe2fe9e31a37f193fc8498a7393a1c31dc5ceb656e2b45b708","impliedFormat":1},{"version":"1b13e7c5c58ab894fe65b099b6d19bb8afae6d04252db1bf55fe6ba95a0af954","impliedFormat":1},{"version":"4355d326c3129e5853b56267903f294ad03e34cc28b75f96b80734882dedac80","impliedFormat":1},{"version":"37139a8d45342c05b6a5aa1698a2e8e882d6dca5fb9a77aa91f05ac04e92e70b","impliedFormat":1},{"version":"e37191297f1234d3ae54edbf174489f9a3091a05fe959724db36f8e58d21fb17","impliedFormat":1},{"version":"3fca8fb3aab1bc7abb9b1420f517e9012fdddcbe18803bea2dd48fad6c45e92e","impliedFormat":1},{"version":"d0b0779e0cac4809a9a3c764ba3bd68314de758765a8e3b9291fe1671bfeb8a1","impliedFormat":1},{"version":"d2116b5f989aa68e585ae261b9d6d836be6ed1be0b55b47336d9f3db34674e86","impliedFormat":1},{"version":"d79a227dd654be16d8006eac8b67212679d1df494dfe6da22ea0bd34a13e010c","impliedFormat":1},{"version":"b9c89b4a2435c171e0a9a56668f510a376cb7991eaecef08b619e6d484841735","impliedFormat":1},{"version":"44a298a6c52a7dab8e970e95a6dabe20972a7c31c340842e0dc57f2c822826eb","impliedFormat":1},{"version":"6a79b61f57699de0a381c8a13f4c4bcd120556bfab0b4576994b6917cb62948b","impliedFormat":1},{"version":"c5133d7bdec65f465df12f0b507fbc0d96c78bfa5a012b0eb322cf1ff654e733","impliedFormat":1},{"version":"00b9ff040025f6b00e0f4ac8305fea1809975b325af31541bd9d69fa3b5e57b1","impliedFormat":1},{"version":"9f96b9fd0362a7bfe6a3aa70baa883c47ae167469c904782c99ccc942f62f0dc","impliedFormat":1},{"version":"54d91053dc6a2936bfd01a130cc3b524e11aa0349da082e8ac03a8bf44250338","impliedFormat":1},{"version":"89049878a456b5e0870bb50289ea8ece28a2abd0255301a261fa8ab6a3e9a07d","impliedFormat":1},{"version":"55ae9554811525f24818e19bdc8779fa99df434be7c03e5fc47fa441315f0226","impliedFormat":1},{"version":"24abac81e9c60089a126704e936192b2309413b40a53d9da68dadd1dd107684e","impliedFormat":1},{"version":"f13310c360ecffddb3858dcb33a7619665369d465f55e7386c31d45dfc3847bf","impliedFormat":1},{"version":"e7bde95a05a0564ee1450bc9a53797b0ac7944bf24d87d6f645baca3aa60df48","impliedFormat":1},{"version":"62e68ce120914431a7d34232d3eca643a7ddd67584387936a5202ae1c4dd9a1b","impliedFormat":1},{"version":"91d695bba902cc2eda7edc076cd17c5c9340f7bb254597deb6679e343effadbb","impliedFormat":1},{"version":"e1cb8168c7e0bd4857a66558fe7fe6c66d08432a0a943c51bacdac83773d5745","impliedFormat":1},{"version":"a464510505f31a356e9833963d89ce39f37a098715fc2863e533255af4410525","impliedFormat":1},{"version":"ebbe6765a836bfa7f03181bc433c8984ca29626270ca1e240c009851222cb8a7","impliedFormat":1},{"version":"ac10457b51ee4a3173b7165c87c795eadd094e024f1d9f0b6f0c131126e3d903","impliedFormat":1},{"version":"468df9d24a6e2bc6b4351417e3b5b4c2ca08264d6d5045fe18eb42e7996e58b4","impliedFormat":1},{"version":"954523d1f4856180cbf79b35bd754e14d3b2aea06c7efd71b254c745976086e9","impliedFormat":1},{"version":"a8af4739274959d70f7da4bfdd64f71cfc08d825c2d5d3561bc7baed760b33ef","impliedFormat":1},{"version":"090fda1107e7d4f8f30a2b341834ed949f01737b5ec6021bb6981f8907330bdb","impliedFormat":1},{"version":"cc32874a27100c32e3706d347eb4f435d6dd5c0d83e547c157352f977bbc6385","impliedFormat":1},{"version":"e45b069d58c9ac341d371b8bc3db4fa7351b9eee1731bffd651cfc1eb622f844","impliedFormat":1},{"version":"7f3c74caad25bfb6dfbf78c6fe194efcf8f79d1703d785fc05cd606fe0270525","impliedFormat":1},{"version":"54f3f7ff36384ca5c9e1627118b43df3014b7e0f62c9722619d19cdb7e43d608","impliedFormat":1},{"version":"2f346f1233bae487f1f9a11025fc73a1bf9093ee47980a9f4a75b84ea0bb7021","impliedFormat":1},{"version":"013444d0b8c1f7b5115462c31573a699fee7458381b0611062a0069d3ef810e8","impliedFormat":1},{"version":"0612b149cabbc136cb25de9daf062659f306b67793edc5e39755c51c724e2949","impliedFormat":1},{"version":"2579b150b86b5f644d86a6d58f17e3b801772c78866c34d41f86f3fc9eb523fe","impliedFormat":1},{"version":"0353e05b0d8475c10ddd88056e0483b191aa5cdea00a25e0505b96e023f1a2d9","impliedFormat":1},{"version":"8c4df93dafcf06adc42a63477cc38b352565a3ed0a19dd8ef7dfacc253749327","impliedFormat":1},{"version":"22a35275abc67f8aba44efc52b2f4b1abc2c94e183d36647fdab5a5e7c1bdf23","impliedFormat":1},{"version":"99193bafaa9ce112889698de25c4b8c80b1209bb7402189aea1c7ada708a8a54","impliedFormat":1},{"version":"70473538c6eb9494d53bf1539fe69df68d87c348743d8f7244dcb02ca3619484","impliedFormat":1},{"version":"c48932ab06a4e7531bdca7b0f739ace5fa273f9a1b9009bcd26902f8c0b851f0","impliedFormat":1},{"version":"df6c83e574308f6540c19e3409370482a7d8f448d56c65790b4ac0ab6f6fedd8","impliedFormat":1},{"version":"32f19b665839b1382b21afc41917cda47a56e744cd3df9986b13a72746d1c522","impliedFormat":1},{"version":"8db1ed144dd2304b9bd6e41211e22bad5f4ab1d8006e6ac127b29599f4b36083","impliedFormat":1},{"version":"843a5e3737f2abbbbd43bf2014b70f1c69a80530814a27ae1f8be213ae9ec222","impliedFormat":1},{"version":"6fc1be224ad6b3f3ec11535820def2d21636a47205c2c9de32238ba1ac8d82e6","impliedFormat":1},{"version":"5a44788293f9165116c9c183be66cefef0dc5d718782a04847de53bf664f3cc1","impliedFormat":1},{"version":"afd653ae63ce07075b018ba5ce8f4e977b6055c81cc65998410b904b94003c0a","impliedFormat":1},{"version":"9172155acfeb17b9d75f65b84f36cb3eb0ff3cd763db3f0d1ad5f6d10d55662f","impliedFormat":1},{"version":"71807b208e5f15feffb3ff530bec5b46b1217af0d8cc96dde00d549353bcb864","impliedFormat":1},{"version":"1a6eca5c2bc446481046c01a54553c3ffb856f81607a074f9f0256c59dd0ab13","impliedFormat":1},{"version":"6ecc423e71318bafbd230e6059e082c377170dfc7e02fccfa600586f8604d452","impliedFormat":1},{"version":"772f9bdd2bf50c9c01b0506001545e9b878faa7394ad6e7d90b49b179a024584","impliedFormat":1},{"version":"f204b03cb07517d71715ac8bc7552542bfab395adb53e31c07fbc67de6856de1","impliedFormat":1},{"version":"7467736a77548887faa90a7d0e074459810a5db4bbc6de302a2be6c05287ccae","impliedFormat":1},{"version":"39504a2c1278ee4d0dc1a34e27c80e58b4c53c08c87e3a7fc924f18c936bebb5","impliedFormat":1},{"version":"cd1ccdd9fd7980d43dfede5d42ee3d18064baed98b136089cf7c8221d562f058","impliedFormat":1},{"version":"d60f9a4fd1e734e7b79517f02622426ea1000deb7d6549dfdece043353691a4e","impliedFormat":1},{"version":"403d28b5e5f8fcff795ac038902033ec5890143e950af45bd91a3ed231e8b59c","impliedFormat":1},{"version":"c73b59f91088c00886d44ca296d53a75c263c3bda31e3b2f37ceb137382282be","impliedFormat":1},{"version":"e7aa2c584edb0970cb4bb01eb10344200286055f9a22bc3dadcc5a1f9199af3e","impliedFormat":1},{"version":"bfeb476eb0049185cb94c2bfcadb3ce1190554bbcf170d2bf7c68ed9bb00458e","impliedFormat":1},{"version":"ae23a65a2b664ffe979b0a2a98842e10bdf3af67a356f14bbc9d77eb3ab13585","impliedFormat":1},{"version":"eccf6ad2a8624329653896e8dbd03f30756cbd902a81b5d3942d6cf0e1a21575","impliedFormat":1},{"version":"1930c964051c04b4b5475702613cd5a27fcc2d33057aa946ff52bfca990dbc84","impliedFormat":1},{"version":"2793d525d79404df346e4ef58a82f9b6d28a7650beeb17378cd121c45ba03f02","impliedFormat":1},{"version":"62463aa3d299ae0cdc5473d2ac32213a05753c3adce87a8801c6d2b114a64116","impliedFormat":1},{"version":"c9c2eabaad71c534d7de16385977f95184fdf3ddd0339dadbd5d599488d94f90","impliedFormat":1},{"version":"d0642c453e6af4c0700182bec4afc5b2cc9498fe27c9b1bcf2e6f75dd1892699","impliedFormat":1},{"version":"8f4469dd750d15f72ba66876c8bc429d3c9ce49599a13f868a427d6681d45351","impliedFormat":1},{"version":"d1e888a33faeb1f0e3c558bbe0ea4a55056318e0b2f8eba72ffd6729c3bbff4e","impliedFormat":1},{"version":"f689c0633e8c95f550d36af943d775f3fae3dac81a28714b45c7af0bbb76a980","impliedFormat":1},{"version":"fef736cfb404b4db9aa942f377dbbac6edb76d18aabd3b647713fa75da8939e9","impliedFormat":1},{"version":"45659c92e49dfca4601acc7e57fbb03a71513c69768984baf86ead8d20387a01","impliedFormat":1},{"version":"0239d8f6a3f51b26cbdbb9362f4fde35651c6bd0ff3d9fc09ee4a2da6065cb4e","impliedFormat":1},{"version":"6e5ab399ec7bd61d4f86421cc6074fd904379c3923706c899d15146e4f9a08c8","impliedFormat":1},{"version":"c9ffec02582eed74f518ae3e32a5dcf4ac835532e548300c5c5f950cdfeead5f","impliedFormat":1},{"version":"df343f5de08f5b607a3c7954ff1b512b7fa983d561e136cce0b6dc6849602a15","impliedFormat":1},{"version":"8fc97ef271771dc6f81a9c846d007ac4f0cb5779e3f441c1de54dfda5046fe7b","impliedFormat":1},{"version":"b5a060e2a4c54695076f871ddc0c91a0ff8eea1262177c4ede5593acbf1ca3bb","impliedFormat":1},{"version":"08ee70765d3fa7c5bad4afbbe1c542771e17f84bfd5e3e872ae1fdc5160836c8","impliedFormat":1},{"version":"1c225a18846203fafc4334658715b0d3fd3ee842c4cfd42e628a535eda17730d","impliedFormat":1},{"version":"7ce93da38595d1caf57452d57e0733474564c2b290459d34f6e9dcf66e2d8beb","impliedFormat":1},{"version":"d7b672c1c583e9e34ff6df2549d6a55d7ca3adaf72e6a05081ea9ee625dac59f","impliedFormat":1},{"version":"f3a2902e84ebdef6525ed6bf116387a1256ea9ae8eeb36c22f070b7c9ea4cf09","impliedFormat":1},{"version":"33bb0d96cea9782d701332e6b7390f8efae3af92fd3e2aa2ac45e4a610e705d6","impliedFormat":1},{"version":"ae3e98448468e46474d817b5ebe74db11ab22c2feb60e292d96ce1a4ee963623","impliedFormat":1},{"version":"f0a2fdee9e801ac9320a8660dd6b8a930bf8c5b658d390ae0feafdba8b633688","impliedFormat":1},{"version":"7beb7f04f6186bdac5e622d44e4cac38d9f2b9fcad984b10d3762e369524dd77","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},"eaf8514ce110fa428a93a27408df4d06d133dbd9ed0a775c315ddfdd507853a9","8a89eff88780d7f3815935533d4fab6df2faef13a726c14a1d205886057be0f1",{"version":"b8ad793dc17938bc462812e3522bbd3d62519d91d9b4a6422bed1383c2d3eb42","impliedFormat":1},{"version":"8b0b6a4c032a56d5651f7dd02ba3f05fbfe4131c4095093633cda3cae0991972","impliedFormat":1},{"version":"ff3c48a17bf10dfbb62448152042e4a48a56c9972059997ab9e7ed03b191809b","impliedFormat":1},{"version":"192a0c215bffe5e4ac7b9ff1e90e94bf4dfdad4f0f69a5ae07fccc36435ebb87","impliedFormat":1},{"version":"3ef8565e3d254583cced37534f161c31e3a8f341ff005c98b582c6d8c9274538","impliedFormat":1},{"version":"d7e42a3800e287d2a1af8479c7dd58c8663e80a01686cb89e0068be6c777d687","impliedFormat":1},{"version":"1098034333d3eb3c1d974435cacba9bd5a625711453412b3a514774fec7ca748","impliedFormat":1},{"version":"f2388b97b898a93d5a864e85627e3af8638695ebfa6d732ecd39d382824f0e63","impliedFormat":1},{"version":"6c6bd91368169cfa94b4f8cc64ebca2b050685ec76bc4082c44ce125b5530cca","impliedFormat":1},{"version":"f477375e6f0bf2a638a71d4e7a3da8885e3a03f3e5350688541d136b10b762a6","impliedFormat":1},{"version":"a44d6ea4dc70c3d789e9cef3cc42b79c78d17d3ce07f5fd278a7e1cbe824da56","impliedFormat":1},{"version":"272af80940fcc0c8325e4a04322c50d11f8b8842f96ac66cbd440835e958dd14","impliedFormat":1},{"version":"1803e48a3ec919ccafbcafeef5e410776ca0644ae8c6c87beca4c92d8a964434","impliedFormat":1},{"version":"875c43c5409e197e72ee517cb1f8fd358406b4adf058dbdc1e50c8db93d68f26","impliedFormat":1},{"version":"8854713984b9588eac1cab69c9e2a6e1a33760d9a2d182169059991914dd8577","impliedFormat":1},{"version":"e333d487ca89f26eafb95ea4b59bea8ba26b357e9f2fd3728be81d999f9e8cf6","impliedFormat":1},{"version":"2f554c6798b731fc39ff4e3d86aadc932fdeaa063e3cbab025623ff5653c0031","impliedFormat":1},{"version":"fe4613c6c0d23edc04cd8585bdd86bc7337dc6265fb52037d11ca19eeb5e5aaf","impliedFormat":1},{"version":"53b26fbee1a21a6403cf4625d0e501a966b9ccf735754b854366cee8984b711c","impliedFormat":1},{"version":"c503be3ddb3990ab27ca20c6559d29b547d9f9413e05d2987dd7c4bcf52f3736","impliedFormat":1},{"version":"598b15f0ae9a73082631d14cb8297a1285150ca325dbce98fc29c4f0b7079443","impliedFormat":1},{"version":"8c59d8256086ed17676139ee43c1155673e357ab956fb9d00711a7cac73e059d","impliedFormat":1},{"version":"cfe88132f67aa055a3f49d59b01585fa8d890f5a66a0a13bb71973d57573eee7","impliedFormat":1},{"version":"53ce488a97f0b50686ade64252f60a1e491591dd7324f017b86d78239bd232ca","impliedFormat":1},{"version":"50fd11b764194f06977c162c37e5a70bcf0d3579bf82dd4de4eee3ac68d0f82f","impliedFormat":1},{"version":"e0ceb647dcdf6b27fd37e8b0406c7eafb8adfc99414837f3c9bfd28ffed6150a","impliedFormat":1},{"version":"99579aa074ed298e7a3d6a47e68f0cd099e92411212d5081ce88344a5b1b528d","impliedFormat":1},{"version":"c94c1aa80687a277396307b80774ca540d0559c2f7ba340168c2637c82b1f766","impliedFormat":1},{"version":"ce7dbf31739cc7bca35ca50e4f0cbd75cd31fd6c05c66841f8748e225dc73aaf","impliedFormat":1},{"version":"942ab34f62ac3f3d20014615b6442b6dc51815e30a878ebc390dd70e0dec63bf","impliedFormat":1},{"version":"7a671bf8b4ad81b8b8aea76213ca31b8a5de4ba39490fbdee249fc5ba974a622","impliedFormat":1},{"version":"8e07f13fb0f67e12863b096734f004e14c5ebfd34a524ed4c863c80354c25a44","impliedFormat":1},{"version":"6f6bdb523e5162216efc36ebba4f1ef8e845f1a9e55f15387df8e85206448aee","impliedFormat":1},{"version":"aa2d6531a04d6379318d29891de396f61ccc171bfd2f8448cc1649c184becdf2","impliedFormat":1},{"version":"d422f0c340060a53cb56d0db24dd170e31e236a808130ab106f7ab2c846f1cdb","impliedFormat":1},{"version":"424403ef35c4c97a7f00ea85f4a5e2f088659c731e75dbe0c546137cb64ef8d8","impliedFormat":1},{"version":"16900e9a60518461d7889be8efeca3fe2cbcd3f6ce6dee70fea81dfbf8990a76","impliedFormat":1},{"version":"6daf17b3bd9499bd0cc1733ab227267d48cd0145ed9967c983ccb8f52eb72d6e","impliedFormat":1},{"version":"e4177e6220d0fef2500432c723dbd2eb9a27dcb491344e6b342be58cc1379ec0","impliedFormat":1},{"version":"ab710f1ee2866e473454a348cffd8d5486e3c07c255f214e19e59a4f17eece4d","impliedFormat":1},{"version":"db7ff3459e80382c61441ea9171f183252b6acc82957ecb6285fff4dca55c585","impliedFormat":1},{"version":"4a168e11fe0f46918721d2f6fcdb676333395736371db1c113ae30b6fde9ccd2","impliedFormat":1},{"version":"2a899aef0c6c94cc3537fe93ec8047647e77a3f52ee7cacda95a8c956d3623fb","impliedFormat":1},{"version":"ef2c1585cad462bdf65f2640e7bcd75cd0dbc45bae297e75072e11fe3db017fa","impliedFormat":1},{"version":"6a52170a5e4600bbb47a94a1dd9522dca7348ce591d8cdbb7d4fe3e23bbea461","impliedFormat":1},{"version":"6f6eadb32844b0ec7b322293b011316486894f110443197c4c9fbcba01b3b2fa","impliedFormat":1},{"version":"a51e08f41e3e948c287268a275bfe652856a10f68ddd2bf3e3aaf5b8cdb9ef85","impliedFormat":1},{"version":"16c144a21cd99926eeba1605aec9984439e91aa864d1c210e176ca668f5f586a","impliedFormat":1},{"version":"af48a76b75041e2b3e7bd8eed786c07f39ea896bb2ff165e27e18208d09b8bee","impliedFormat":1},{"version":"fd4107bd5c899165a21ab93768904d5cfb3e98b952f91fbf5a12789a4c0744e6","impliedFormat":1},{"version":"deb092bc337b2cb0a1b14f3d43f56bc663e1447694e6d479d6df8296bdd452d6","impliedFormat":1},{"version":"041bc1c3620322cb6152183857601707ef6626e9d99f736e8780533689fb1bf9","impliedFormat":1},{"version":"77165b117f552be305d3bc2ef83424ff1e67afb22bfabd14ebebb3468c21fcaa","impliedFormat":1},{"version":"128e7c2ffd37aa29e05367400d718b0e4770cefb1e658d8783ec80a16bc0643a","impliedFormat":1},{"version":"076ac4f2d642c473fa7f01c8c1b7b4ef58f921130174d9cf78430651f44c43ec","impliedFormat":1},{"version":"396c1e5a39706999ec8cc582916e05fcb4f901631d2c192c1292e95089a494d9","impliedFormat":1},{"version":"89df75d28f34fc698fe261f9489125b4e5828fbd62d863bbe93373d3ed995056","impliedFormat":1},{"version":"8ccf5843249a042f4553a308816fe8a03aa423e55544637757d0cfa338bb5186","impliedFormat":1},{"version":"93b44aa4a7b27ba57d9e2bad6fb7943956de85c5cc330d2c3e30cd25b4583d44","impliedFormat":1},{"version":"a0c6216075f54cafdfa90412596b165ff85e2cadd319c49557cc8410f487b77c","impliedFormat":1},{"version":"3c359d811ec0097cba00fb2afd844b125a2ddf4cad88afaf864e88c8d3d358bd","impliedFormat":1},{"version":"d8ec19be7d6d3950992c3418f3a4aa2bcad144252bd7c0891462b5879f436e4e","impliedFormat":1},{"version":"db37aa3208b48bdcbc27c0c1ae3d1b86c0d5159e65543e8ab79cbfb37b1f2f34","impliedFormat":1},{"version":"d62f09256941e92a95b78ae2267e4cf5ff2ca8915d62b9561b1bc85af1baf428","impliedFormat":1},{"version":"e6223b7263dd7a49f4691bf8df2b1e69f764fb46972937e6f9b28538d050b1ba","impliedFormat":1},{"version":"2daf06d8e15cbca27baa6c106253b92dad96afd87af9996cf49a47103b97dc95","impliedFormat":1},{"version":"1db014db736a09668e0c0576585174dbcfd6471bb5e2d79f151a241e0d18d66b","impliedFormat":1},{"version":"8a153d30edde9cefd102e5523b5a9673c298fc7cf7af5173ae946cbb8dd48f11","impliedFormat":1},{"version":"abaaf8d606990f505ee5f76d0b45a44df60886a7d470820fcfb2c06eafa99659","impliedFormat":1},{"version":"8109e0580fc71dbefd6091b8825acf83209b6c07d3f54c33afeafab5e1f88844","impliedFormat":1},{"version":"d92a80c2c05cf974704088f9da904fe5eadc0b3ad49ddd1ef70ca8028b5adda1","impliedFormat":1},{"version":"fbd7450f20b4486c54f8a90486c395b14f76da66ba30a7d83590e199848f0660","impliedFormat":1},{"version":"ece5b0e45c865645ab65880854899a5422a0b76ada7baa49300c76d38a530ee1","impliedFormat":1},{"version":"62d89ac385aeab821e2d55b4f9a23a277d44f33c67fefe4859c17b80fdb397ea","impliedFormat":1},{"version":"f4dee11887c5564886026263c6ee65c0babc971b2b8848d85c35927af25da827","impliedFormat":1},{"version":"fb8dd49a4cd6d802be4554fbab193bb06e2035905779777f32326cb57cf6a2c2","impliedFormat":1},{"version":"df29ade4994de2d9327a5f44a706bbe6103022a8f40316839afa38d3e078ee06","impliedFormat":1},{"version":"82d3e00d56a71fc169f3cf9ec5f5ffcc92f6c0e67d4dfc130dafe9f1886d5515","impliedFormat":1},{"version":"d38f45cb868a830d130ac8b87d3f7e8caff4961a3a1feae055de5e538e20879a","impliedFormat":1},{"version":"4c30a5cb3097befb9704d16aa4670e64e39ea69c5964a1433b9ffd32e1a5a3a1","impliedFormat":1},{"version":"1b33478647aa1b771314745807397002a410c746480e9447db959110999873ce","impliedFormat":1},{"version":"7b3a5e25bf3c51af55cb2986b89949317aa0f6cbfb5317edd7d4037fa52219a9","impliedFormat":1},{"version":"3cd50f6a83629c0ec330fc482e587bfa96532d4c9ce85e6c3ddf9f52f63eee11","impliedFormat":1},{"version":"9fac6ebf3c60ced53dd21def30a679ec225fc3ff4b8d66b86326c285a4eebb5a","impliedFormat":1},{"version":"8cb83cb98c460cd716d2a98b64eb1a07a3a65c7362436550e02f5c2d212871d1","impliedFormat":1},{"version":"07bc8a3551e39e70c38e7293b1a09916867d728043e352b119f951742cb91624","impliedFormat":1},{"version":"e47adc2176f43c617c0ab47f2d9b2bb1706d9e0669bf349a30c3fe09ddd63261","impliedFormat":1},{"version":"7fec79dfd7319fec7456b1b53134edb54c411ba493a0aef350eee75a4f223eeb","impliedFormat":1},{"version":"189c489705bb96a308dcde9b3336011d08bfbca568bcaf5d5d55c05468e9de7a","impliedFormat":1},{"version":"98f4b1074567341764b580bf14c5aabe82a4390d11553780814f7e932970a6f7","impliedFormat":1},{"version":"dadfa5fd3d5c511ca6bfe240243b5cf2e0f87e44ea63e23c4b2fce253c0d4601","impliedFormat":1},{"version":"2e252235037a2cd8feebfbf74aa460f783e5d423895d13f29a934d7655a1f8be","impliedFormat":1},{"version":"763f4ac187891a6d71ae8821f45eef7ff915b5d687233349e2c8a76c22b3bf2a","impliedFormat":1},{"version":"dff93e0997c4e64ff29e9f70cad172c0b438c4f58c119f17a51c94d48164475a","impliedFormat":1},{"version":"fd1ddf926b323dfa439be49c1d41bbe233fe5656975a11183aeb3bf2addfa3bb","impliedFormat":1},{"version":"6dda11db28da6bcc7ff09242cd1866bdddd0ae91e2db3bea03ba66112399641a","impliedFormat":1},{"version":"ea4cd1e72af1aa49cf208b9cb4caf542437beb7a7a5b522f50a5f1b7480362ed","impliedFormat":1},{"version":"903a7d68a222d94da11a5a89449fdd5dd75d83cd95af34c0242e10b85ec33a93","impliedFormat":1},{"version":"e7fe2e7ed5c3a7beff60361632be19a8943e53466b7dd69c34f89faf473206d7","impliedFormat":1},{"version":"b4896cee83379e159f83021e262223354db79e439092e485611163e2082224ff","impliedFormat":1},{"version":"5243e79a643e41d9653011d6c66e95048fc0478eb8593dc079b70877a2e3990e","impliedFormat":1},{"version":"70521b6ab0dcba37539e5303104f29b721bfb2940b2776da4cc818c07e1fefc1","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab41ef1f2cdafb8df48be20cd969d875602483859dc194e9c97c8a576892c052","affectsGlobalScope":true,"impliedFormat":1},{"version":"d153a11543fd884b596587ccd97aebbeed950b26933ee000f94009f1ab142848","affectsGlobalScope":true,"impliedFormat":1},{"version":"21d819c173c0cf7cc3ce57c3276e77fd9a8a01d35a06ad87158781515c9a438a","impliedFormat":1},{"version":"98cffbf06d6bab333473c70a893770dbe990783904002c4f1a960447b4b53dca","affectsGlobalScope":true,"impliedFormat":1},{"version":"ba481bca06f37d3f2c137ce343c7d5937029b2468f8e26111f3c9d9963d6568d","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d9ef24f9a22a88e3e9b3b3d8c40ab1ddb0853f1bfbd5c843c37800138437b61","affectsGlobalScope":true,"impliedFormat":1},{"version":"1db0b7dca579049ca4193d034d835f6bfe73096c73663e5ef9a0b5779939f3d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"9798340ffb0d067d69b1ae5b32faa17ab31b82466a3fc00d8f2f2df0c8554aaa","affectsGlobalScope":true,"impliedFormat":1},{"version":"f26b11d8d8e4b8028f1c7d618b22274c892e4b0ef5b3678a8ccbad85419aef43","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e9c23ba78aabc2e0a27033f18737a6df754067731e69dc5f52823957d60a4b6","impliedFormat":1},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"1d6e127068ea8e104a912e42fc0a110e2aa5a66a356a917a163e8cf9a65e4a75","impliedFormat":1},{"version":"5ded6427296cdf3b9542de4471d2aa8d3983671d4cac0f4bf9c637208d1ced43","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"ea0148f897b45a76544ae179784c95af1bd6721b8610af9ffa467a518a086a43","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"dd3273ead9fbde62a72949c97dbec2247ea08e0c6952e701a483d74ef92d6a17","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"e61be3f894b41b7baa1fbd6a66893f2579bfad01d208b4ff61daef21493ef0a8","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"b52476feb4a0cbcb25e5931b930fc73cb6643fb1a5060bf8a3dda0eeae5b4b68","affectsGlobalScope":true,"impliedFormat":1},{"version":"e2677634fe27e87348825bb041651e22d50a613e2fdf6a4a3ade971d71bac37e","impliedFormat":1},{"version":"7394959e5a741b185456e1ef5d64599c36c60a323207450991e7a42e08911419","impliedFormat":1},{"version":"8c0bcd6c6b67b4b503c11e91a1fb91522ed585900eab2ab1f61bba7d7caa9d6f","impliedFormat":1},{"version":"8cd19276b6590b3ebbeeb030ac271871b9ed0afc3074ac88a94ed2449174b776","affectsGlobalScope":true,"impliedFormat":1},{"version":"696eb8d28f5949b87d894b26dc97318ef944c794a9a4e4f62360cd1d1958014b","impliedFormat":1},{"version":"3f8fa3061bd7402970b399300880d55257953ee6d3cd408722cb9ac20126460c","impliedFormat":1},{"version":"35ec8b6760fd7138bbf5809b84551e31028fb2ba7b6dc91d95d098bf212ca8b4","affectsGlobalScope":true,"impliedFormat":1},{"version":"5524481e56c48ff486f42926778c0a3cce1cc85dc46683b92b1271865bcf015a","impliedFormat":1},{"version":"68bd56c92c2bd7d2339457eb84d63e7de3bd56a69b25f3576e1568d21a162398","affectsGlobalScope":true,"impliedFormat":1},{"version":"3e93b123f7c2944969d291b35fed2af79a6e9e27fdd5faa99748a51c07c02d28","impliedFormat":1},{"version":"9d19808c8c291a9010a6c788e8532a2da70f811adb431c97520803e0ec649991","impliedFormat":1},{"version":"87aad3dd9752067dc875cfaa466fc44246451c0c560b820796bdd528e29bef40","impliedFormat":1},{"version":"4aacb0dd020eeaef65426153686cc639a78ec2885dc72ad220be1d25f1a439df","impliedFormat":1},{"version":"f0bd7e6d931657b59605c44112eaf8b980ba7f957a5051ed21cb93d978cf2f45","impliedFormat":1},{"version":"8db0ae9cb14d9955b14c214f34dae1b9ef2baee2fe4ce794a4cd3ac2531e3255","affectsGlobalScope":true,"impliedFormat":1},{"version":"15fc6f7512c86810273af28f224251a5a879e4261b4d4c7e532abfbfc3983134","impliedFormat":1},{"version":"58adba1a8ab2d10b54dc1dced4e41f4e7c9772cbbac40939c0dc8ce2cdb1d442","impliedFormat":1},{"version":"641942a78f9063caa5d6b777c99304b7d1dc7328076038c6d94d8a0b81fc95c1","impliedFormat":1},{"version":"714435130b9015fae551788df2a88038471a5a11eb471f27c4ede86552842bc9","impliedFormat":1},{"version":"855cd5f7eb396f5f1ab1bc0f8580339bff77b68a770f84c6b254e319bbfd1ac7","impliedFormat":1},{"version":"5650cf3dace09e7c25d384e3e6b818b938f68f4e8de96f52d9c5a1b3db068e86","impliedFormat":1},{"version":"1354ca5c38bd3fd3836a68e0f7c9f91f172582ba30ab15bb8c075891b91502b7","affectsGlobalScope":true,"impliedFormat":1},{"version":"7e20d899c28ca26a2a7afc98beaa69e63ff7fba0a8bc47b4e3bf3ede5e09e424","impliedFormat":1},{"version":"2d2fcaab481b31a5882065c7951255703ddbe1c0e507af56ea42d79ac3911201","impliedFormat":1},{"version":"a192fe8ec33f75edbc8d8f3ed79f768dfae11ff5735e7fe52bfa69956e46d78d","impliedFormat":1},{"version":"ca867399f7db82df981d6915bcbb2d81131d7d1ef683bc782b59f71dda59bc85","affectsGlobalScope":true,"impliedFormat":1},{"version":"372413016d17d804e1d139418aca0c68e47a83fb6669490857f4b318de8cccb3","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e043a1bc8fbf2a255bccf9bf27e0f1caf916c3b0518ea34aa72357c0afd42ec","impliedFormat":1},{"version":"b4f70ec656a11d570e1a9edce07d118cd58d9760239e2ece99306ee9dfe61d02","impliedFormat":1},{"version":"3bc2f1e2c95c04048212c569ed38e338873f6a8593930cf5a7ef24ffb38fc3b6","impliedFormat":1},{"version":"6e70e9570e98aae2b825b533aa6292b6abd542e8d9f6e9475e88e1d7ba17c866","impliedFormat":1},{"version":"f9d9d753d430ed050dc1bf2667a1bab711ccbb1c1507183d794cc195a5b085cc","impliedFormat":1},{"version":"9eece5e586312581ccd106d4853e861aaaa1a39f8e3ea672b8c3847eedd12f6e","impliedFormat":1},{"version":"085f552d005479e2e6a7311cdbbe5d8c55c497b4d19274285df161ee9684cd9c","impliedFormat":1},{"version":"37ba7b45141a45ce6e80e66f2a96c8a5ab1bcef0fc2d0f56bb58df96ec67e972","impliedFormat":1},{"version":"45650f47bfb376c8a8ed39d4bcda5902ab899a3150029684ee4c10676d9fbaee","impliedFormat":1},{"version":"007faacc9268357caa21d24169f3f3f2497af3e9241308df2d89f6e6d9bb3f2e","affectsGlobalScope":true,"impliedFormat":1},{"version":"74cf591a0f63db318651e0e04cb55f8791385f86e987a67fd4d2eaab8191f730","impliedFormat":1},{"version":"5eab9b3dc9b34f185417342436ec3f106898da5f4801992d8ff38ab3aff346b5","impliedFormat":1},{"version":"12ed4559eba17cd977aa0db658d25c4047067444b51acfdcbf38470630642b23","affectsGlobalScope":true,"impliedFormat":1},{"version":"f3ffabc95802521e1e4bcba4c88d8615176dc6e09111d920c7a213bdda6e1d65","impliedFormat":1},{"version":"809821b8a065e3234a55b3a9d7846231ed18d66dd749f2494c66288d890daf7f","impliedFormat":1},{"version":"ae56f65caf3be91108707bd8dfbccc2a57a91feb5daabf7165a06a945545ed26","impliedFormat":1},{"version":"a136d5de521da20f31631a0a96bf712370779d1c05b7015d7019a9b2a0446ca9","impliedFormat":1},{"version":"c3b41e74b9a84b88b1dca61ec39eee25c0dbc8e7d519ba11bb070918cfacf656","affectsGlobalScope":true,"impliedFormat":1},{"version":"4737a9dc24d0e68b734e6cfbcea0c15a2cfafeb493485e27905f7856988c6b29","affectsGlobalScope":true,"impliedFormat":1},{"version":"36d8d3e7506b631c9582c251a2c0b8a28855af3f76719b12b534c6edf952748d","impliedFormat":1},{"version":"1ca69210cc42729e7ca97d3a9ad48f2e9cb0042bada4075b588ae5387debd318","impliedFormat":1},{"version":"f5ebe66baaf7c552cfa59d75f2bfba679f329204847db3cec385acda245e574e","impliedFormat":1},{"version":"ed59add13139f84da271cafd32e2171876b0a0af2f798d0c663e8eeb867732cf","affectsGlobalScope":true,"impliedFormat":1},{"version":"b7c5e2ea4a9749097c347454805e933844ed207b6eefec6b7cfd418b5f5f7b28","impliedFormat":1},{"version":"b1810689b76fd473bd12cc9ee219f8e62f54a7d08019a235d07424afbf074d25","impliedFormat":1},{"version":"76e7352249c42b9d54fe1f9e1ebcef777da1cb2eb33038366af49469d433597b","impliedFormat":1},{"version":"88cb622dd0ec1ef860e5c27fa884e60d2eba5ae22c7907dff82c56a69bdd2c8a","impliedFormat":1},{"version":"eb234b3e285e8bc071bdddc1ec0460095e13ead6222d44b02c4e0869522f9ba3","impliedFormat":1},{"version":"c85114872760189e50fef131944427b0fb367f0cc0b6dce164bb427a6fd89381","impliedFormat":1},{"version":"5ad69b0d7e7bdbcd3adfdb6a3e306e935c9c2711b1c60493646504a2f991346e","impliedFormat":1},{"version":"a12a667efdeb03b529bd4ebb4032998ddd32743799f59f9f18b186f8e63a2cf1","impliedFormat":1},{"version":"cee7efa0ae4c58deab218d1df0d1bf84abfd5c356cff28bca1421489cba13a19","impliedFormat":1},{"version":"f9e034b1ae29825c00532e08ea852b0c72885c343ee48d2975db0a6481218ab3","impliedFormat":1},{"version":"1193f49cbb883f40326461fe379e58ffa4c18d15bf6d6a1974ad2894e4fb20f3","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"2e2bc02af7b535d267be8cecbc5831466dd71c5af294401821791b26cb363c47","impliedFormat":1},{"version":"986affe0f60331f20df7d708ee097056b0973d85422ec2ce754af19c1fa4e4b1","impliedFormat":1},{"version":"8f06c2807459f1958b297f4ad09c6612d7dbd7997c9ccfc6ea384f7538e0cea8","impliedFormat":1},{"version":"a7de30cd043d7299bfe9daaca3732b086e734341587c3e923b01f3fd74d31126","impliedFormat":1},{"version":"78f7fad319e4ac305ffe8e03027423279b53a8af4db305096aa75d446b1ec7af","impliedFormat":1},{"version":"3bf58923a1d27819745bdad52bca1bdced9fef12cc0c7f8a3fd5f4e0206b684a","impliedFormat":1},{"version":"8fc11f102df58f03d36fcbf0da3efa37c177f5f18f534c76179ceef0c3a672cd","impliedFormat":1},{"version":"e6935ab0f64a886e778c12a54ed6e9075ce7e7f44723ff0d52020a654b025a09","impliedFormat":1},{"version":"9829af7653a29f1b85d3dd688a6c6256087c0b737b85d84b630e7f93fd420faf","impliedFormat":1},{"version":"3d9d985d41e536fcf79fc95082925c2f1ae5ade75814ad2bd70c0944747f7ac4","impliedFormat":1},{"version":"03b419ce598d77fe4d1705c8281a797a908f57ce24a15d6174d7e7276d355a65","impliedFormat":1},{"version":"b0e6f1b1569779cf567317c2265d67460d1d3b4de4e79126533109d87dc16d50","impliedFormat":1},{"version":"18cb8be1326ffa4158abd8d84c9b0a189c0f52201f12f7af2d2af830c077f2bf","impliedFormat":1},{"version":"9c15e2b87cd3d8b18881bcc7d72b2d1dc6d5fe078b674ae12c12c19ec09a6a1a","impliedFormat":1},{"version":"0de68916e23c1e3df800f9f61cdd7c506ceb0656fcbc245ee9974aad26786781","impliedFormat":1},{"version":"80c538ee6a62249e77ba3de07efb23d4a7ca8946499c065261bf5079f1cd3cf0","impliedFormat":1},{"version":"ad4277862bdcbe1cf5c1e0d43b39770e1ccc033da92f5b9ff75ca8c3a03a569b","impliedFormat":1},{"version":"46a86c47400a564df04a1604fcac41cb599ebbada392527a1462c9dfe4713d78","impliedFormat":1},{"version":"f342dcb96ad26855757929a9f6632704b7013f65786573d4fdcd4da09f475923","impliedFormat":1},{"version":"dcd467dc444953a537502d9e140d4f2dc13010664d4216cc8e6977b3c5c3efa3","impliedFormat":1},{"version":"ca476924dfa6120b807a14e0a8aea7b061b8bdaa7eecdb303d7957c769102e96","impliedFormat":1},{"version":"848fe622fac070f8af9255e5d63fe829e3da079cae30be48fb6deb5dbf2c27c6","impliedFormat":1},{"version":"f3bb275073b5db8931c042d347fdce888775436a4774836221af57fdccec32ff","impliedFormat":1},{"version":"03cb8cb2f8ef002a5cac9b8c9a0c02e5fd09de128b9769c5b920a6cbfc080087","impliedFormat":1},{"version":"3e5ebc3a6a938a03a361f4cdb9a26c9f5a1bac82b46273e11d5d37cd8eccc918","impliedFormat":1},{"version":"a0a7800e71c504c21f3051a29f0f6f948f0b8296c9ebffeb67033822aabf92e0","impliedFormat":1},{"version":"6a219f12b3e853398d51192736707e320699a355052687bad4729784649ff519","impliedFormat":1},{"version":"4294a84634c56529e67301a3258448019e41c101de6b9646ea41c0ecdc70df92","impliedFormat":1},{"version":"80fc027e10234b809a9a40086114a8154657dcb8478d58c85ef850592d352870","impliedFormat":1},{"version":"27f24ba43083d406b372e9eff72dbc378afa0503dac1c1dd32499cc92fc9cb22","impliedFormat":1},{"version":"12594611a054ca7fe69962f690a4e79922d563b4b434716eb855d63a9d11a78f","impliedFormat":1},{"version":"1440eca2d8bc47ebdbc5a901b369de1b7b39c3297e5b4ac9631899f49ea9740b","impliedFormat":1},{"version":"fc9897fbada879bda954603ea204c6e5df913262a90ad848b5efaab182b58033","impliedFormat":1},{"version":"93443b2da120bea58eb48bd7da86559d4cf868dc2d581eebf9b48b51ba1e8894","impliedFormat":1},{"version":"94be5c5f8cf26bbf53554cba4b112e30134349b14f3c0fd0ede3b51ec25a7174","impliedFormat":1},{"version":"c2956026078814be6dc01515213aeb1eb816e81715085952bbc97b7c81fe3f6d","impliedFormat":1},{"version":"ac3a69c529ab256532825b08902aec65d0d88c66963e39ae19a3d214953aedc5","impliedFormat":1},{"version":"fe29108f3ddf7030c3d573c5226ebe03213170b3beca5200ca7cb33755184017","impliedFormat":1},{"version":"04d5bfb0a0eecd66c0b3f522477bf69065a9703be8300fbea5566a0fc4a97b9d","impliedFormat":1},{"version":"d5e3e13faca961679bed01d80bc38b3336e7de598ebf9b03ec7d31081af735ad","impliedFormat":1},{"version":"de05a488fb501de32c1ec0af2a6ddfe0fdef46935b9f4ffb3922d355b15da674","impliedFormat":1},{"version":"9f00f2bc49f0c10275a52cb4f9e2991860d8b7b0922bfab6eafe14178377aa72","impliedFormat":1},{"version":"7bd94408358caf1794ad24546ca0aa56f9be6be2d3245d0972fcb924b84a81fd","impliedFormat":1},{"version":"0e7c3660d1df392b6f6ae7fa697f0629ae4404e5b7bac05dd81136247aff32d5","impliedFormat":1},{"version":"b0b3636502dc0c50295f67747968f202f7b775eac5016329606d1bc2888d5dd9","impliedFormat":1},{"version":"f9ede7ea553dc197fd5d2604f62cda1be1aea50024ed73237d9e3144f0c93608","impliedFormat":1},{"version":"b1005ae67226fd9b7b65333d9a351917f517d421a0c63b7cde59bec3b8e3562f","impliedFormat":1},{"version":"c6688fd4c2a8a24c9b80da3660a7a06b93ed37d12d84f3ba4aa071ffc125e75f","impliedFormat":1},{"version":"20efc25890a0b2f09e4d224afaaf84917baa77b1aee60d9dfd11ff8078d73f93","impliedFormat":1},{"version":"d00b48096854d711cee688e7ff1ca796c1bf0d27ca509633c2a98b85cc23d47d","impliedFormat":1},{"version":"30f116226d0e53c6cbbdbc967479d5c8036935f771b2af51987c2e8d4cc7fc6a","impliedFormat":1},{"version":"8be98ffc3c54fb40b220796b796388f8ade50c8ba813a811bffccf98006566d5","impliedFormat":1},{"version":"4e82eed3c1b5084132708ce030f8ec90b69e4b7bb844dcaacd808045ae24c0e2","impliedFormat":1},{"version":"eae8c7cbcb175b997ce8e76cd6e770eca5dba07228f6cb4a44e1b0a11eb87685","impliedFormat":1},{"version":"b3ded8e50b3cdf548d7c8d3b3b5b2105932b04a2f08b392564f4bc499407e4e5","impliedFormat":1},{"version":"4ed2d8fb4c598719985b8fbef65f7de9c3f5ae6a233fc0fe20bd00193c490908","impliedFormat":1},{"version":"6da51da9b74383988b89e17298ceca510357f63830f78b40f72afe4d5a9cee3e","impliedFormat":1},{"version":"512a079a1a3de2492c80aa599e173b2ea8cc6afb2800e3e99f14330b34155fe1","impliedFormat":1},{"version":"f281f20b801830f2f94b2bc0b18aba01d4fb50c2f4a847ffcadff39de31c8b80","impliedFormat":1},{"version":"738ddac5ab5b61d70d3466f3906d6b3c83c8786e922c6e726a6597296181ae87","impliedFormat":1},{"version":"90d202ace592f7b51b131a5890ec93e4df774c8677a485391c280cef0ea53f48","impliedFormat":1},{"version":"b34e1861949a545916696ef40f4a7fe71793661e72dd4db5e04cacc60ef23f7a","impliedFormat":1},{"version":"dd3f42651cfa863ded8fa0b0608fb892b826e254a0a6cbc447388cb5e11bffd5","impliedFormat":1},{"version":"8e3842ba15690ab4b340893a4552a8c3670b8f347fbb835afe14be98891eef10","impliedFormat":1},{"version":"e7b9673dcd3d1825dbd70ad1d1f848d68189afc302ecdafc6eb30cbe7bd420b5","impliedFormat":1},{"version":"15911b87a2ad4b65b30c445802d55fa6186c66068603113042e8c3dfa4a35e2a","impliedFormat":1},{"version":"a9dc7b8d06b1f69d219f61fa3f7ac621e6e3a8d5a430e800cd7d1a755cc058c3","impliedFormat":1},{"version":"f8c496656cb5fd737931b4d6c60bd72a97c48f37c07dcb74a593dd24ac3f684a","impliedFormat":1},{"version":"f2cf1d33c458ac091983e5dac1613f264d48a69b281e43c5b055321320082358","impliedFormat":1},{"version":"0fa43815d4b05eafe97c056dae73c313f23a9f00b559f1e942d042c7a04db93c","impliedFormat":1},{"version":"e769097e5ea39d2ed548eeb9c093e90f26dde167f95eb80fbdd4efb041778387","impliedFormat":1},{"version":"a02db6aabaa291a85cf52b0c3f02a75301b80be856db63d44af4feea2179f37b","impliedFormat":1},{"version":"e1e94e41f47a4496566a9f40e815687a2eca1e7b7910b67704813cf61248b869","impliedFormat":1},{"version":"557ba6713b2a6fefd943399d5fb6c64e315dc461e9e05eaa6300fdbeeda5d0a1","impliedFormat":1},{"version":"1f7eeb69504ad94d16f4731f707d2af879adc7487dc35b146e2d86825bb779b4","impliedFormat":1},{"version":"c1b5c480e4d38377c82f9f517c12014d3d4475c0e607c4845e0836e0e89bbf7d","impliedFormat":1},{"version":"1a014a8365354f37ea245349a4361d3b46589be7921fe7f1dbf408cc0f084bab","impliedFormat":1},{"version":"87fc4a324b9fa5c9b93a13b5ae1b55ea390929ec1b0450afebff9620921a9cc1","impliedFormat":1},{"version":"73c0b8df0e282e26a53820f53502847a043bd77a9cda78782207d5349842fba2","impliedFormat":1},{"version":"5c7391307b9a7c540d678f015d687c277269aa9171f441467e20bab15694db40","impliedFormat":1},{"version":"082aa8710bbf3d16b877e798341c69599fdd487b4dc34d374ab3e3ec6d46f690","impliedFormat":1},{"version":"acb9367f45f12526ea808d6da48ab77eee1ceb2b6fe47ab02bbcc7cce4c972b0","impliedFormat":1},{"version":"d6db974317fd9ff66a923555464850dcf87976054a7adacf09d53323f64686d1","impliedFormat":1},{"version":"79f4812dffe8f933c12c341d68eee731cb6dd7f2a4bb20097c411560c97a6263","impliedFormat":1},{"version":"c446e8f3bd5b16e121252e05ba7696524ca95ec3f819c12fb8c37e7836744769","impliedFormat":1},{"version":"23386bb0bcb20fcb367149f22f5c6468b53f1987e86fd25de875ffb769e4d241","impliedFormat":1},{"version":"3913806467307a4bd874b105ac3e79ac261ab986fbdce7f0feea26cbcee95765","impliedFormat":1},{"version":"a9417a980a4300048d179d0295e5b7dd76e4db7b566344779ee576cbd084b3c4","impliedFormat":1},{"version":"b96760c030c41fa078b35ea05fc3e7e4d2a81710a8329271d42b6abc110d5dbe","impliedFormat":1},{"version":"ef8ff23609cec5eb95e2beb98132ad90c0c5075415b50228b12f89ffaf981a4a","impliedFormat":1},{"version":"80bbc9365ca8398c69eae77cdf7284d07192a17dacf1904095ab4c89f4520a5d","impliedFormat":1},{"version":"174a3381f98fc78c451528cb1aa1baaa37a51852ec6fa90d42efd876301537c1","impliedFormat":1},{"version":"2c0de27d99a9331cfac8bc5c6bbd174e0593628bf3df268faa6c4188962a9549","impliedFormat":1},{"version":"1a17bcbc124a098987f7b1adbbcd412f8372ecb37e352b1c50165dac439eee5e","impliedFormat":1},{"version":"0ef49170735d9e5902f55b72465accadd0db93cae52544e3c469cbc8fbdbf654","impliedFormat":1},{"version":"f68a30e88dfa7d12d8dd4609bc9d5226a31d260bf3526de5554feed3f0bf0cb6","impliedFormat":1},{"version":"d8acc6f92c85e784acbbc72036156a4c1168a18cba5390c7d363040479c39396","impliedFormat":1},{"version":"1fffef141820a0556f60aa6050eccb17dbcdc29ecd8a17ee4366573fd9c96ce3","impliedFormat":1},{"version":"d2598c755c11170e3b5f85cd0c237033e783fd4896070c06c35b2246879612b8","impliedFormat":1},{"version":"8d2044a28963c6c85a2cf4e334eb49bb6f3dd0c0dfe316233148a9be74510a0e","impliedFormat":1},{"version":"2660eb7dba5976c2dcbea02ec146b1f27109e7bee323392db584f8c78a6477dd","impliedFormat":1},{"version":"54a4f21be5428d7bff9240efb4e8cae3cb771cad37f46911978e013ff7289238","impliedFormat":1},{"version":"10837df0382365c2544fb75cb9a8f6e481e68c64915362941b4ea4468fd0ef61","impliedFormat":1},{"version":"cc4483c79688bd3f69c11cb3299a07d5dcf87646c35b869c77cde553c42893cf","impliedFormat":1},{"version":"faf76eeb5dd5d4d1e37c6eb875d114fa97297c2b50b10e25066fed09e325a77a","impliedFormat":1},{"version":"b741703daf465b44177ef31cc637bde5cd5345e6c048d5807108e6e868182b01","impliedFormat":1},{"version":"9c3e59360437a3e2a22f7f1032559a4c24aba697365b62fb4816b7c8c66035b8","impliedFormat":1},{"version":"393446ab3f0dd3449ad6fd4c8abd0c82b711c514b9e8dfbf75222bbc48eb0cb6","impliedFormat":1},{"version":"ea02a962453ec628e886a6c5d0fc03bf4da9dfa38e1f8d42e65e07b2651edd85","impliedFormat":1},{"version":"5eb09226bfa1928721a438e37c004647fc19d8d1f4817bddcc350e57fb32935f","impliedFormat":1},{"version":"5994ed389d7fc28c03dad647ecb62e5349160bde443b0c7a54e0e10d6368bcbd","impliedFormat":1},{"version":"e1ff7df643e1aa1dbf1863113a913358844ed66f1af452e774834b0008e578b2","impliedFormat":1},{"version":"c5114285d0283d05e09cd959e605a4f76e5816c2fbe712241993fd66496083e5","impliedFormat":1},{"version":"2752e949c871f2cbd146efa21ebc34e4693c0ac8020401f90a45d4e150682181","impliedFormat":1},{"version":"c349cea980e28566998972522156daac849af8a9e4a9d59074845e319b975f5d","impliedFormat":1},{"version":"0370682454d1d243b75a7c7031bc8589531a472e927b67854c1b53b55ee496ea","impliedFormat":1},{"version":"cf6b4dbb5a1ac9ece24761c3a08682029851b292b67113a93b5e2bfd2e64e49d","impliedFormat":1},{"version":"baa9fbd480342a1d5e3e11ba3629f2826d18d4a765f1f9693ab87bfb3ce54adb","impliedFormat":1},{"version":"cb2fea712720bb7951d7e5d63db8670bf4a400d3e0fb197bceb6ef44efe36ec3","impliedFormat":1},{"version":"1b4fcfc691980d63a730d47d5309d9f85cdddc18a4c83f6e3af20936d103e3ff","impliedFormat":1},{"version":"ef19d5fe42541f8b529bccd10f488d12caefa3b57a0deb1ed6143219cba716b4","impliedFormat":1},{"version":"84b5e6269d7cf53008a479eeb533ef09d025eafb4febe3729301b8d4daf37ff2","impliedFormat":1},{"version":"04196b5d9edd60b9648daa329c3355d7c95f33b7e520e7835eb21002174a8b8c","impliedFormat":1},{"version":"637c0d7d8cedbc64a3c228c3fa6bef884746f7a16a631e7532f9828c9ac06b8a","impliedFormat":1},{"version":"9e665aea79b702fd612ffb7ac741e4160d35d8d696a789129ebcbaea003beb3d","impliedFormat":1},{"version":"c8eeffebe6c2c6800f73aa59d1436d4dadbad7f3ddda02a831ffa66114c3122d","impliedFormat":1},{"version":"caf3f141f93cbf527ad18ecce326311d70342fe1e16ce93e5ce8d6bcdf02bd48","impliedFormat":1},{"version":"4283d88023e6e9645626475e392565464eae99068f17e324cfc40a27d10fe94f","impliedFormat":1},{"version":"51e3b73dea24e2a9638345fb7a2a7ef5d3aa2e7a285ad6bd446b45fab826def1","impliedFormat":1},{"version":"77c4c9f71f3736ed179043a72c4fad9832023855804fbe5261a956428b26a7a6","impliedFormat":1},{"version":"7232467057ec57666b884924f84fd21cd3a79cc826430c312e61a5bc5758f879","impliedFormat":1},{"version":"624f5dbfd76f2d77f20ace318e8cb918608a296106e55587fb443ef3030c595d","impliedFormat":1},{"version":"c78bb1275f640e4902ad5c3383ab4f54f73322a59c95924ab671125ba9546294","impliedFormat":1},{"version":"1cb0838371e8213ce116a1497bb86bcf01a11a755b77587980ee7cfb2d625ece","impliedFormat":1},{"version":"f5d29fd7099274774c203d94d8c0238770ab411b922b978be15a2c3ec8ab845c","impliedFormat":1},{"version":"6d99b5b226a65890ce27796e086d58c6351f601757c1e9f217a69e944d05e7e6","impliedFormat":1},{"version":"10b322f5bc001bec9bf08513c978c120adb0abe3c82793b11bdaf75873426c05","impliedFormat":1},{"version":"51b4efdc8dc92bc6ae2c44d4edad265decad70e8577d5653fc7f85200cbf6c6e","impliedFormat":1},{"version":"c3fa40ac56aa2598d9133c90b115eeb39bbad56c6dfca350dc8435b8b107fe26","impliedFormat":1},{"version":"cc542183b68b048a8cf64eb6231b3d0852f7f4d0191d4637c9d1d4c3f44b83b5","impliedFormat":1},{"version":"4b954a3d432dca82c787c06d2f1cca0fe673a4b440c5e0195429bd1fe43b324a","impliedFormat":1},{"version":"c6fd975d319a70d6ba90bf38c34ac8efebe531214038fe561a27f89f2203f78e","impliedFormat":1},{"version":"a818204639081cf07d80885b88aff5120e5a4135211162f5e08cfc00ef3bf5b6","impliedFormat":1},{"version":"c194ca06da86829b836bb188dffc05543bbea3cbda797667c7a7cade2f907646","impliedFormat":1},{"version":"6df6afb0424a7c7581ee98a9333d30e893b943d0a4709b88f18c252ddc3101b4","impliedFormat":1},{"version":"59c2cbf84c22fae87f4f506f36a7258a72b931b602115067dfd6008ee526f8c0","impliedFormat":1},{"version":"1e09cd1bc6b6baa0733e1e799c4533105ea79cbb109937c71e8c870e14693216","impliedFormat":1},{"version":"0b60cfcd94fa9bd9fa58176650c7e4c72f99b9d30a50d0b55aa08b510276af96","impliedFormat":1},{"version":"ba25681012e5117866a2456dd3557e24aa5a946ed641126aa4469880db526883","impliedFormat":1},{"version":"2b1e058a8c3944890c7ce7c712ecfd0f2645420ee67537ac031d7afe6feda6e0","impliedFormat":1},{"version":"175dbcd1f226eebd93fd9628e9180fb537bb1171489b33db7b388ef0f4e73b37","impliedFormat":1},{"version":"69ec6331ee3a7cd6bade5d5f683f1705c1041ff77432aa18c50d2097e61f93db","impliedFormat":1},{"version":"06f34a0f2151b619314fc8a54e4352a40fd5606bda50623c326c3be365cc1ef9","impliedFormat":1},{"version":"6c6dcb49af3d72d823334f74a554b2f9917e3a59b3219934b7ae9e6b03a3e8b4","impliedFormat":1},{"version":"9628be9799a060a3f7fe2e1f08fab2b21cdd7e97a2bbc3ef2f0029be46e0d7da","impliedFormat":1},{"version":"3d24aec533fe2f035b0675ba1c0e55e8680a714fff2a517e0fb388279476701c","impliedFormat":1},{"version":"224e2edff4c1e67d9c5179aa70e31d0dc7dd4ea5a9e80ffde121df9e5254eef2","impliedFormat":1},{"version":"e324c3b2058f9525cf5c11915284f9dfdf7550c98f103429b271fe723c4f8e14","impliedFormat":1},{"version":"70a3659d557bb683091f9d318762a330a3acb3954f5e89e5134d24c9272192f1","impliedFormat":1},{"version":"d9fe2c804f7db2f19e4323601278b748dc2984798f265c37cd37bb84e6c88ab8","impliedFormat":1},{"version":"3525647a73ae2124fa8f353f0a078b44ff1ee6f82958c2bb507de61575f12fff","impliedFormat":1},{"version":"d7238315cbd18ebeed93f41ad756a0ed9759824b9b158c3d7a1e0b71682d8966","impliedFormat":1},{"version":"eeba7376ce9721610d3282a4159f3c60154b7b3877fb251f7b3211b085cfdc18","impliedFormat":1},{"version":"643efb9d7747ee1dd50ff5bd4b7a87351157e55988c7d2f90ffbdf124f063931","impliedFormat":1},{"version":"788c870cac6b39980a5cc41bf610b1873952ecdd339b781f0687d42682ffc5dc","impliedFormat":1},{"version":"d51a2e050c8a131b13ec9330a0869e5ac75b9ac4ebde52d5f474e819510b5263","impliedFormat":1},{"version":"b694593470a9bf370987e5b0757d5a9a88a46a703c9cf7921969f3379ce16148","impliedFormat":1},{"version":"6c034655fa83236bd779cacfc1d5b469d6e2150a1993e66ecca92376a8b2c6a7","impliedFormat":1},{"version":"6bd6933efe9d6263d9f1a534a28a8f88b1e4c331b95d85d39350cf02eca8dce0","impliedFormat":1},{"version":"658cf468a05b2b591fcd5455a76d9927face59ac4a21b4965982b3c234f5d289","impliedFormat":1},{"version":"6bf893d1b824bde22ee5880c0c760c1dd0a5163c38d22311441a3341b6965d2d","impliedFormat":1},{"version":"579d9d3c25058b854a6f7cc6368a473efcaa0740f45db13cb508761d35fc0156","impliedFormat":1},{"version":"68705604f0666ba3862670153eb4f965c3079415e7ab30a35b3126e36277dc9e","impliedFormat":1},{"version":"28b415e70f9da0346545b7d2bcf361844a8e5778bd6b45bc1a2859f99700ff5b","impliedFormat":1},{"version":"a905f2f6785e3971bd97c42191394209d97f2aefb11841f7353dd9789821fa8c","impliedFormat":1},{"version":"e099c5ebddf80ae7285d380c7dd3b5d49c1347346ced51ae121b846833a8d102","impliedFormat":1},{"version":"aec91730b9f4d83758b4a45596317d34d6ecdbe9330a44629f53af47641b96ee","impliedFormat":1},{"version":"2321197343254570a8d4c868572059bfdfb683cf9d4099b6d4694250dac69471","impliedFormat":1},{"version":"18a3be03c31356b60ea1090bcc905d99e4983ca911cc70b34ad0b9b4d4e050c3","impliedFormat":1},{"version":"9833a67663f960dc2d1908a19365ddde55c0651235596ac60d7078a9be6f6e56","impliedFormat":1},{"version":"2bcb8920601b80911430979b6db4a58a7908a31334e74e4e22b75c65edce3587","impliedFormat":1},{"version":"c3186dc74d62d0fb6fba29841ccbf995614992526c37fac5c082d0f28b351e54","impliedFormat":1},{"version":"2306daed18f7f59542a99857a678ef818058eefa30c2a556af123a1cf53889cd","impliedFormat":1},{"version":"b41ed9285a09710807ce2c423e038dfe538e46e9183c0c05aadc27bfb9ae256a","impliedFormat":1},{"version":"56b9f9de03f28eb5922750a213d3f47b21a4f00a48c7c9b89bf1733623873d3a","impliedFormat":1},{"version":"2bdd736078e445858cb1d9df809ff3a2f00445d78664dd70b6794fb2156bdd53","impliedFormat":1},{"version":"2653fb2893a65c610ec17d0e454e2b16726f16118425f0bc8a38c801943ef7f5","impliedFormat":1},{"version":"74ffa4541a56571f379060acaf9ab86da6c889dfe1f588425807e0117e62bba5","impliedFormat":1},{"version":"cf4dc15ca9dc6c0995dd2a9264e5ec37d09d9d551c85f395034e812abdf60a99","impliedFormat":1},{"version":"73e8b003f39c7ce46d2811749dab1dd1b309235fd5c277bd672c30a98b5cf90f","impliedFormat":1},{"version":"4cb49e79595c6413fcb01af55a8a574705bf385bd2ec5cf8b777778952e2914a","impliedFormat":1},{"version":"d6b44382b2670f38c8473e7c16b6e8a9bfa546b396b920afc4c53410eeb22abf","impliedFormat":1},{"version":"3b5c6f451b7ad87e3fcd2008d3a6cb69bd33803e541e9c0fe35754201389158f","impliedFormat":1},{"version":"8329556a2e85e3c3ff3dff43141790ff624b0f5138cedec5bb793164cf8b088f","impliedFormat":1},{"version":"4c889ce7e61ca7f3b7733e0d2be80b3af373e080c922e04639aa25f22963ae63","impliedFormat":1},{"version":"2239a8cd90c48e0b5c075e51099e7e3b4fc3d4741e4d9cc4410d2544d4216946","impliedFormat":1},{"version":"f5aa57712223d7438799be67b0c4a0e5ac3841f6397b5e692673944374f58a83","impliedFormat":1},{"version":"774c37f8faed74c238915868ccc36d0afedfbafb1d2329d6a230966457f57cbd","impliedFormat":1},{"version":"bc41b711477270e8d6f1110d57863284d084b089a22592c7c09df8d4cc3d1d20","impliedFormat":1},{"version":"0c792fe4e5f383b4f085a0033553fb84ed9322b7923fd59d4575aa43135e050d","impliedFormat":1},{"version":"228ed3721f42cc25bfebceef33754ce4766414d975ff71d012f01f141dbe3549","impliedFormat":1},{"version":"08985cdb65bbfe3c70d0037794a3d0f0a5613f55c278c77277a7acc17205db57","impliedFormat":1},{"version":"22bdefb6b2107006ab203073218566443a52ab65eb5e4e8e86c3d38efe776588","impliedFormat":1},{"version":"0f01b48cee64391fabef3f344e6e86197dc921f0f88a6d45d133ac58283d9690","impliedFormat":1},{"version":"c86fea295c21ea01c93410eba2ec6e4f918b97d0c3bf9f1bb1960eabe417e7eb","impliedFormat":1},{"version":"05d41b3e7789381ff4d7f06d8739bf54cc8e75b835cb28f22e59c1d212e48ff3","impliedFormat":1},{"version":"6fbcfc270125b77808679b682663c7c6ad36518f5a528c5f7258bcd635096770","impliedFormat":1},{"version":"9d3bd4ee558de42e9d8434f7293b404c4b7a09b344e77c36bbe959696328d594","impliedFormat":1},{"version":"f63be9b46a22ee5894316cf71a4ba7581809dd98cf046109060a1214ee9e2977","impliedFormat":1},{"version":"dd3cc41b5764c9435b7cae3cc830be4ee6071f41a607188e43aa1edeba4fbb3e","impliedFormat":1},{"version":"b2dbb9485701a1d8250d9a35b74afd41b9a403c32484ed40ed195e8aa369ae70","impliedFormat":1},{"version":"5aa7565991c306061181bd0148c458bcce3472d912e2af6a98a0a54904cd84fc","impliedFormat":1},{"version":"9629e70ae80485928a562adb978890c53c7be47c3b3624dbb82641e1da48fd2f","impliedFormat":1},{"version":"c33d86e1d4753d035c4ea8d0fdb2377043bc894e4227be3ceabc8e6a5411ab2e","impliedFormat":1},{"version":"f9ec74382c95cbc85804daf0e9dabed56511a6dfb72f8a2868aa46a0b9b5eafc","impliedFormat":1},{"version":"1ff7a67731e575e9f31837883ddfc6bfcef4a09630267e433bc5aea65ad2ced4","impliedFormat":1},{"version":"0c4f6b6eb73b0fa4d27ce6eef6c2f1e7bd93d953b941e486b55d5d4b22883350","impliedFormat":1},{"version":"af9692ce3b9db8b94dcfbaa672cb6a87472f8c909b83b5aeea043d6e53e8b107","impliedFormat":1},{"version":"782f2628a998fd03f4ccbe9884da532b8c9be645077556e235149ca9e6bd8c7d","impliedFormat":1},{"version":"269b7db8b769d5677f8d5d219e74ea2390b72ea2c65676b307e172e8f605a74a","impliedFormat":1},{"version":"ae731d469fae328ba73d6928e4466b72e3966f92f14cd1a711f9a489c6f93839","impliedFormat":1},{"version":"90878ed33999d4ff8da72bd2ca3efb1cde76d81940767adc8c229a70eb9332b2","impliedFormat":1},{"version":"d7236656e70e3a7005dba52aa27b2c989ba676aff1cab0863795ac6185f8d54f","impliedFormat":1},{"version":"e327901e9f31d1ad13928a95d95604ee4917d72ad96092da65612879d89aba42","impliedFormat":1},{"version":"868914e3630910e58d4ad917f44b045d05303adc113931e4b197357f59c3e93e","impliedFormat":1},{"version":"7d59adb080be18e595f1ce421fc50facd0073672b8e67abac5665ba7376b29b9","impliedFormat":1},{"version":"275344839c4df9f991bcf5d99c98d61ef3ce3425421e63eeb4641f544cb76e25","impliedFormat":1},{"version":"c4f1cc0bd56665694e010a6096a1d31b689fa33a4dd2e3aa591c4e343dd5181c","impliedFormat":1},{"version":"81c3d9b4d90902aa6b3cbd22e4d956b6eb5c46c4ea2d42c8ff63201c3e9676da","impliedFormat":1},{"version":"5bfc3a4bd84a6f4b992b3d285193a8140c80bbb49d50a98c4f28ad14d10e0acc","impliedFormat":1},{"version":"a7cf6a2391061ca613649bc3497596f96c1e933f7b166fa9b6856022b68783ab","impliedFormat":1},{"version":"864c844c424536df0f6f745101d90d69dd14b36aa8bd6dde11268bb91e7de88e","impliedFormat":1},{"version":"c74a70a215bbd8b763610f195459193ab05c877b3654e74f6c8881848b9ddb7f","impliedFormat":1},{"version":"3fa94513af13055cd79ea0b70078521e4484e576f8973e0712db9aab2f5dd436","impliedFormat":1},{"version":"48ffc1a6b67d61110c44d786d520a0cba81bb89667c7cdc35d4157263bfb7175","impliedFormat":1},{"version":"7cb4007e1e7b6192af196dc1dacd29a0c3adc44df23190752bef6cbbc94b5e0b","impliedFormat":1},{"version":"3d409649b4e73004b7561219ce791874818239913cac47accc083fad58f4f985","impliedFormat":1},{"version":"051908114dee3ca6d0250aacb0a4a201e60f458085177d5eda1fc3cde2e570f3","impliedFormat":1},{"version":"3e8240b75f97eb4495679f6031fb02ad889a43017cae4b17d572324513559372","impliedFormat":1},{"version":"d82609394127fb33eed0b58e33f8a0f55b62b21c2b6c10f1d7348b4781e392cb","impliedFormat":1},{"version":"b0f8a6436fbaf3fb7b707e2551b3029650bfaeb51d4b98e089e9a104d5b559b5","impliedFormat":1},{"version":"eae0ac4f87d56dcf9fbcf9314540cc1447e7a206eee8371b44afa3e2911e520c","impliedFormat":1},{"version":"b585e7131070c77b28cc682f9b1be6710e5506c196a4b6b94c3028eb865de4a7","impliedFormat":1},{"version":"b92ac4cc40d551450a87f9154a8d088e31cff02c36e81db2976d9ff070ba9929","impliedFormat":1},{"version":"6f99b4a552fbdc6afd36d695201712901d9b3f009e340db8b8d1d3415f2776f5","impliedFormat":1},{"version":"43700e8832b12f82e6f519b56fae2695e93bb18dddb485ddea6583a0d1482992","impliedFormat":1},{"version":"e8165ea64af5de7f400d851aeea5703a3b8ac021c08bebc958859d341fa53387","impliedFormat":1},{"version":"6db546ea3ced87efda943e6016c2a748e150941a0704af013dfe535936e820e1","impliedFormat":1},{"version":"f521c4293b6d8f097e885be50c2fef97de3dd512ad26f978360bb70c766e7eae","impliedFormat":1},{"version":"a0666dfd499f319cc51a1e6d9722ed9c830b040801427bbdd2984b73f98d292a","impliedFormat":1},{"version":"a7d86611d7882643dd8c529d56d2e2b698afd3a13a5adc2d9e8157b57927c0da","impliedFormat":1},{"version":"7e4615c366c93399f288c7bfbaa00a1dc123578be9d8ac96b15d489efc3f4851","impliedFormat":1},{"version":"f2e6c87a2c322ee1473cb0bd776eb20ee7bff041bc56619e5d245134ab73e83d","impliedFormat":1},{"version":"ee89bc94431b2dfaf6a7e690f8d9a5473b9d61de4ddcb637217d11229fe5b69f","impliedFormat":1},{"version":"a19c1014936f60281156dd4798395ad4ab26b7578b5a6a062b344a3e924a4333","impliedFormat":1},{"version":"5608be84dd2ca55fc6d9b6da43f67194182f40af00291198b6487229403a98fe","impliedFormat":1},{"version":"4a800f1d740379122c473c18343058f4bd63c3dffdef4d0edba668caa9c75f54","impliedFormat":1},{"version":"8e6868a58ca21e92e09017440fdb42ebfe78361803be2c1e7f49883b7113fdc2","impliedFormat":1},{"version":"2fbb72a22faefa3c9ae0dfb2a7e83d7b3d82ec625a74a8800a9da973511b0672","impliedFormat":1},{"version":"3e8c1a811bad9e5cd313c3d90c39a99867befa746098cdad81a9578ac3392541","impliedFormat":1},{"version":"d88f78b4e272864f414d98e5ed0996cd09f7a3bb01c5b7528320386f7383153d","impliedFormat":1},{"version":"0b9c34da2c6f0170e6a357112b91f2351712c5a537b76e42adfee9a91308b122","impliedFormat":1},{"version":"47adac87ec85a52ed2562cb4a3b441383551727ed802e471aa05c12e7cc7e27e","impliedFormat":1},{"version":"d1cacf181763c5d0960986f6d0abd1a36fc58fc06a707c9f5060b6b5526179ca","impliedFormat":1},{"version":"92610d503212366ff87801c2b9dc2d1bccfa427f175261a5c11331bc3588bb3f","impliedFormat":1},{"version":"805e2737ce5d94d7da549ed51dfa2e27c2f06114b19573687e9bde355a20f0ff","impliedFormat":1},{"version":"a37b576e17cf09938090a0e7feaec52d5091a1d2bbd73d7335d350e5f0e8be95","impliedFormat":1},{"version":"98971aa63683469692fef990fcba8b7ba3bae3077de26ac4be3e1545d09874b8","impliedFormat":1},{"version":"c6d36fa611917b6177e9c103a2719a61421044fb81cdd0accd19eba08d1b54de","impliedFormat":1},{"version":"088592cf2e218b99b02a5029ed8d1a763a3856cd25e012cfbb536b7494f08971","impliedFormat":1},{"version":"5eb39c56462b29c90cb373676a9a9a179f348a8684b85990367b3bbc6be5a6e9","impliedFormat":1},{"version":"52252b11bcbfaeb4c04dc9ec92ea3f1481684eee62c0c913e8ff1421dc0807e5","impliedFormat":1},{"version":"731d07940d9b4313122e6cc58829ea57dcc5748003df9a0cad7eb444b0644685","impliedFormat":1},{"version":"b3ead4874138ce39966238b97f758fdb06f56a14df3f5e538d77596195ece0b5","impliedFormat":1},{"version":"032b40b5529f2ecce0524974dbec04e9c674278ae39760b2ee0d7fce1bb0b165","impliedFormat":1},{"version":"c25736b0cb086cd2afa4206c11959cb8141cea9700f95a766ad37c2712b7772b","impliedFormat":1},{"version":"033c269cd9631b3f56bb69a9f912c1f0d6f83cf2cff4d436ee1c98f6e655e3b5","impliedFormat":1},{"version":"bd6d692a4a950abbfabe29131420abe804e7f3cc187c3c451f9811e9cf4408ce","impliedFormat":1},{"version":"a9b6411417d4bffd9a89c41dc9dedda7d39fb4fa378eaa0ab55ec9ea1a94eb6a","impliedFormat":1},{"version":"1329e7cd7aca4d223ef5a088d82bc3f6f302ce70581c8d3823a050ea155eec3b","impliedFormat":1},{"version":"09248c76437c5b1efce189b4050c398f76a9385135af75c5fb46308b0d1432e0","impliedFormat":1},{"version":"b8df115bf7b30cceeb4550c0be507082b9930ee6268539a1a1aaffb0791cc299","impliedFormat":1},{"version":"dde00f41a2d2b1e70df6df8ac33de7cb3a658956212c7bee326245cc01c990c2","impliedFormat":1},{"version":"115d092e2748990ff0f67f376f47e9a45a2f21f7c7784102419c14b32c4362d1","impliedFormat":1},{"version":"4ba068163c800094cd81b237f86f22c3a33c23cf2a70b9252aca373cfdf59677","impliedFormat":1},{"version":"53e65282ab040a9f535f4ad2e3c8d8346034d8d69941370886d17055874b348d","impliedFormat":1},{"version":"e6db934da4b03c1f4f1da6f4165a981ec004e9e7d956c585775326b392d4d886","impliedFormat":1},{"version":"6ecb85c8cbb289fe72e1d302684e659cc01ef76ae8e0ad01e8b2203706af1d56","impliedFormat":1},{"version":"fca410876e0302680190982f2fc5102d896e65e4f4f20547a185b60364838910","impliedFormat":1},{"version":"601bc70ff67ae9855fc65bad9bb2d135f72147cf22e2490f58ea0d209d95f2ee","impliedFormat":1},{"version":"5cd5a999e218c635ea6c3e0d64da34a0f112757e793f29bc097fd18b5267f427","impliedFormat":1},{"version":"de8a12540370f9f18b160a07ed57917d69fe24525d360531d42d4b1b5d0d9f0f","impliedFormat":1},{"version":"4a397c8a3d1cccf28751bcca469d57faeb637e76b74f6826e76ad66a3c57c7b8","impliedFormat":1},{"version":"34c1bb0d4cf216f2acb3d013ad2c79f906fe89ce829e23a899029dfa738f97e0","impliedFormat":1},{"version":"5c744f3cc0a266dd95b5769a70ddc85c8b6019adbb0954d4de61f89182202ce3","impliedFormat":1},{"version":"b50f05738b1e82cbb7318eb35a7aaf25036f5585b75bbf4377cfa2bad15c40bf","impliedFormat":1},{"version":"c682cb23f38a786bb37901b3f64727bd3c6210292f5bb36f3b11b63fbe2b23ee","impliedFormat":1},{"version":"d6592cf10dc7797d138af32800d53ff4707fdcd6e053812ce701404f5f533351","impliedFormat":1},{"version":"997f6604cd3d35281083706aa2862e8181ed1929a6cbb004c087557d6c7f23c4","impliedFormat":1},{"version":"9584dd669a3bf285e079502ebbb683e7da0bf7f7c1eb3d63f6ef929350667541","impliedFormat":1},{"version":"41a10e2db052a8bf53ed4d933d9b4f5caa30bdaee5a9d978af95f6641ce44860","impliedFormat":1},{"version":"d84761f8a994b5444529c7c294b194de6fd5350ccda974929ea7e8b3893b753a","impliedFormat":1},{"version":"652e51858bafd77e1abcc4d4e9d5e48cc4426c3dd2910021abd8cc664961e135","impliedFormat":1},{"version":"8c5c602045ffdfebeffc7a71cd2bf201fe147a371274b5fcbded765a92f2af78","impliedFormat":1},{"version":"6392ce794eef6f9b57818264bb0eeb24a46cf923f7695a957c15d3d087fbb6cc","impliedFormat":1},{"version":"b10f123e8100aa98723c133af16f1226a6360ec5b6990a0fe82b165d289549db","impliedFormat":1},{"version":"93d20368cdb5fff7f7398bfc9b2b474b2a2d5867277a0631a33b7db7fd53d5b4","impliedFormat":1},{"version":"b1e69b9834104482fabf7fba40e86a282ee10e0600ffd75123622f4610b0ef9e","impliedFormat":1},{"version":"ad5bb6c450cb574289db945ff82be103ed5d0ad8ee8c76164cee7999c695ae01","impliedFormat":1},{"version":"217761e8a5482b3ad20588a801521c2f5f9f7fb2fbb416d4eff3aff9b57f8471","impliedFormat":1},{"version":"7ad780687331f05998c62277d73b6f15ee3e8045b0187a515ffc49c0ad993606","impliedFormat":1},{"version":"e9aa5ccb42e118f5418721d2ac8c0ebdebeb9502007db9b4c1b7c9b8d493013e","impliedFormat":1},{"version":"d300868212b3cc4d13228f5dc2e9880d5959dc742c0c55be2fc43bcda8504c8f","impliedFormat":1},{"version":"0c55daad827669843bd2401f1ddd163b74d9f922680b08ae6e162ceb6c11b078","impliedFormat":1},{"version":"fe45a9bc654dfd1550c9466c0dad9c8017f2626476ed9d25c65ddfc1943f6b74","impliedFormat":1},{"version":"03abcbc7b5b68887525be71a194dd7f9f68276b5fb5b8989abae9a91585ddc33","impliedFormat":1},{"version":"5055e86e689cfe39104ab71298757e5aac839c2ea9d1f12299e76fa79303d47d","impliedFormat":1},{"version":"42266c387025558423c19d624f671352aac3e449c23906cb636f9ae317b72d7e","impliedFormat":1},{"version":"e578a36b3683d233e045a85c9adb0f10e83d2b48f777b9c05fbc363ccc6bdd34","impliedFormat":1},{"version":"0235d0ba0c7b64244d4703b7d6cabd88ba809abeb01da0c13e9ed111bf5e7059","impliedFormat":1},{"version":"9b21e8a79f4213c1cf29f3c408f85a622f9eb6f4902549ccb9a2c00717a0b220","impliedFormat":1},{"version":"d556e498591413e254793f9d64d3108b369a97bd50f9dd4015b5552888e975ef","impliedFormat":1},{"version":"e2c652c7a45072e408c1749908ca39528d3a9a0eb6634a8999b8cf0e35ef20c8","impliedFormat":1},{"version":"ec08224b320739d26aaf61cead7f1e0f82e6581df0216f6fe048aa6f5042cb8c","impliedFormat":1},{"version":"4eadaa271acca9bd20fc6ac1ea5e4bf9ab6698b8ccf3ec07c33df4970f8130f1","impliedFormat":1},{"version":"3238d2eee64423c8d41972c88673b0327d8b40174a78ea346bcd10954a8f3373","impliedFormat":1},{"version":"8f773ddff9070d725dd23f5cf6c8e62bd86984a57b5d5e3fc7583010b48cd8ac","impliedFormat":1},{"version":"5ecd8fdeb6c87db9c320eefbfa9ea27efccbdce853ed38d5ba58e2da482edf1f","impliedFormat":1},{"version":"19a4d116285e7d77e91411966930761a2204ce2d20915afdb12652681a4a88d7","impliedFormat":1},{"version":"c30ca82112586c5dae7477d7e82cc91a7e0d1e658c581f9ec3df07c4485bba84","impliedFormat":1},{"version":"68fca1813d17ee736f41124ccc958d0364cdef79ad1222951bfacc36b2630a58","impliedFormat":1},{"version":"7813329e568df1d42e5a6c52312b1a7c69700e35a561cf085158c345be155b22","impliedFormat":1},{"version":"561067dc7b6b7635277d3cad0a0e11f698d377063dd2c15dfac43ef78847eef4","impliedFormat":1},{"version":"438247e782a8a9b9abdce618e963667cf95157cc6d3f5194a452d3c7d9e9655c","impliedFormat":1},{"version":"0c293195f800014f1fa3ffacf979002c8c1886ab71750432813fb590738eeef5","impliedFormat":1},{"version":"7673348e0cc2f4e33d1db02ecda02f39e66e56ab2cc3c5602246e5532f2715ab","impliedFormat":1},{"version":"83724b26b711d85d6cfc9dd92fd5d666ffaae27fcfb1a0110401b98814ea26c0","impliedFormat":1},{"version":"869a27c929366c3c864013a991fd4c4c86af73eba25513e8ae915f814d3d349c","impliedFormat":1},{"version":"bfa105c32ed586b227188f7b568776d03202dc7aa4c3af2746579450c7d5e7f2","impliedFormat":1},{"version":"756e3f41a7f2501a34e1a070283c7f5550e200eeb43fed3c806e3f2edd924a75","impliedFormat":1},{"version":"59935cc13dcb7c3c7825e770a61e6696bfd11b65e3e47c28acc410dbdf8461c0","impliedFormat":1},{"version":"85e2808cc73ab3ac07774802b34a6ff0d7e1e46c26de7bc2dbe08e04b3340edb","impliedFormat":1},{"version":"f766e5cdea938e0c9d214533fd4501ab0ee23ab4efca9edba334fa02d2869f11","impliedFormat":1},{"version":"eb380820a3a1feda3a182a3d078da18e0d5b7da08ae531ce11133a84b479678c","impliedFormat":1},{"version":"7fba5cc3088ad9acada3daeff52dae0f2cac8d84d19508abd78af5924dc96bea","impliedFormat":1},{"version":"14176cfdbc3d1d633ad9b5daf044ab4c7d0d73be61ca2f14388800e21f0989cd","impliedFormat":1},{"version":"a24f510afe4d938d625a4b5a5374ac0478e56305e8743dd7d37d86d709754286","impliedFormat":1},{"version":"648acdbcbcd01b1a91e8b0ad390ed59fada685977f44b90e148b65bd8159dfe8","impliedFormat":1},{"version":"8309898ba0ac6f2856a94a11723d499091253a6d5df34ddebc6149d43480bfd2","impliedFormat":1},{"version":"a317ae0eb092da3fd799d1717a2da319a74abebe85e2914cb259222969f95705","impliedFormat":1},{"version":"36d76e2dbd5f5243bd566b018c589e2ba707e34b24ec7d285feb11ba6bf23fbe","impliedFormat":1},{"version":"f780879a2ca63dbb59b36f772bc28dccd2840f1377d8d632e8c978b99c26a45f","impliedFormat":1},{"version":"335c2e013b572967a9a282a70f9dded38631189b992381f1df50e966c7f315d6","impliedFormat":1},{"version":"8b7a519edbd0b7654491300d8e3cbd2cb3ef921003569ca39ebd33e77479bb99","impliedFormat":1},{"version":"c90f8038c75600e55db93d97bab73c0ab8fb618d75392d1d1ad32e2f6e9c7908","impliedFormat":1},{"version":"ca083f3bf68e813b5bded56ecbf177636aa75833eb86c7b40e3d75b8ce4c2f78","impliedFormat":1},{"version":"3c8bf00283ef468da8389119d3f5662c81106e302c8810f40ea86b1018df647e","impliedFormat":1},{"version":"67b248e4bac845c5139898b44cbd3e1213674bcc9831039701b5f0f957243a24","impliedFormat":1},{"version":"63d49516f359186f7b3e3115f2c829ed75c319b34022c97b56beead032a073b7","impliedFormat":1},{"version":"9f5f256c7b5cc4a98ef557ea9720f81e96319d569f731c897ddb4514936242b4","impliedFormat":1},{"version":"a20ded6c920f6e566537e93d69cbad79bc57d7e3ce85686003078cf88c1c9cfc","impliedFormat":1},{"version":"40b2d781df7b4a76d33454cb917c3883655ec1d8d05424b7a80d01610ad5082f","impliedFormat":1},{"version":"703ea2acd8b4741248897a5709cd46e22fcd9d13f01ff3481322a86505f0b77c","impliedFormat":1},{"version":"e09c56f8c446225e061b53cb2f95fcbbc8555483ab29165f6b0f39bc82c8d773","impliedFormat":1},{"version":"a571973bc2e34c898c3202452f957e6757f0c08cb66d50d6785f4a9042d74bad","impliedFormat":1},{"version":"a6a059446e66fbf5072eccce94eb5587cef2f99aa04d4bbd4ebe63d0a6592a4f","impliedFormat":1},{"version":"6e2533e27eba5ff02d6eed37e0a7eb69ae7982e0f72fd8f74c90ab201f061867","impliedFormat":1},{"version":"9c10dd3d85b7620ed3105b3f018125d0bb54198bf5847e39622afb22c651a1ad","impliedFormat":1},{"version":"58c62e415bf74b1423bf443587e33d7951a8bf19d7b03073f26e86d9b43ba9ea","impliedFormat":1},{"version":"dd6ec67ad168e92b8bf79ba975c6e0be8c60e403ba704d1c1b31a6059c12f967","impliedFormat":1},{"version":"bcaf468eea143f8e68ca40e5da58d640656b4f36697170c339042500be78ac5d","impliedFormat":1},{"version":"92de961d1db5fe075db8c0b6414a6eec430adaf9022465fe9d0a23f437aafcb3","impliedFormat":1},{"version":"7610ecdae59cea1a8db7580941ebc24d522d8ac1751ce718a6af22d41e1a1279","impliedFormat":1},{"version":"3e55a65822875e85f96e444b79787f619b9473e36c143dedc6d5441a2544b8ab","impliedFormat":1},{"version":"d49275f9098a8e7a5df7c55321b0242cef0bfdde51018b7b2709c4dc74917822","impliedFormat":1},{"version":"b25556c4111afad4cb174aa4674db2e5b23a6b191dc6a3e42c7c3417ea446a68","impliedFormat":1},{"version":"f9568a3a6c74013aee8b09d73ef04175596b51ce6f5d9dcd4885418170fe9306","impliedFormat":1},{"version":"bd3910ccd4fcd05ebd83fbfeb62f5a82a6674c85c6c0e4755c16298df7abe4d7","impliedFormat":1},{"version":"7c0541d0addc3007e5f5776023d5e6e44f96eae0684cdabe59ef04f2a294b116","impliedFormat":1},{"version":"70137204b720e4dd1b81260a70578f0f4f417c53837f8a13859b2f58e20d7150","impliedFormat":1},{"version":"b28b6875a761fd153ebf120fecb359660de80fd36e90c9b3d72a12318bd5d789","impliedFormat":1},{"version":"56d092bd6225f6e67d9acab3fd65ce0a4edb36cadba2f0370e67322e2f6f1bc8","impliedFormat":1},{"version":"a4709d5d466ad8dcf4ddccb905ad95348131df1616f964185be9739f96526bde","impliedFormat":1},{"version":"73b0fd6255f24e82be861f800a264f0175984062b6ccca3052578b03ed6f397b","impliedFormat":1},{"version":"4a3f7c6f02cb01eb7a9800548b41cfa03a57e476fc92a72869983f37efa8067a","impliedFormat":1},{"version":"ee428614fde0db27939b30094352fc85b58d1bdba6d63be6d8b2aec5929fd9a2","impliedFormat":1},{"version":"bc0b17d3fd0e34083fbc886367ed53563b569d1d05214f60b21117e2dbfb7fdd","impliedFormat":1},{"version":"6120bbd2dbac7d6bb7005b3e00ddb8e6acb9592a37e2bedf32218ad21da915e5","impliedFormat":1},{"version":"72fa257966dec421bf308d15ccf5ee43c588309942d51dd6330250bb0ab39891","impliedFormat":1},{"version":"9a7638d62db8cfa1466093d7d413fdf85c5e4a7c663ed76f2bfc8739c8e01505","impliedFormat":1},{"version":"058709777c09f2ef9b91ec305d4fd84cfa44eb4a0e39e39a3c759924b352f194","impliedFormat":1},{"version":"c338859b98f8a11f80e3e47e33767299e7a4facdf0870c01c8694fa8fa048d16","impliedFormat":1},{"version":"a6f9821e4b5f28264f61f7a8fccbedb30edf194313088242687c31ddf6c7a336","impliedFormat":1},{"version":"b113e9770d5be136c5e2add9e6cdf40d85051762ff2391f71d552975e66b1500","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"03464dcca517bcfb982cefdc316afe821aae8bbe02dcd4765dfa25bc2aecd097","impliedFormat":1},{"version":"59bdc8b3c0ca88ace4d08cf703a52a14f91ce05e3d66235df792915ea54f67c9","impliedFormat":1},{"version":"8a90c628f293590574bbeb66092271849d180a7f4812cb05233a2c4cb30e0c04","impliedFormat":1},{"version":"d2ab468a72716e9a385b9c0188ddd17045efb781ce90fd9f00141729cdc867e6","impliedFormat":1},{"version":"c3fbb898f4185e04b223a3c406f71be2ce89b58816b95096e91bd40bf74d2a08","impliedFormat":1},{"version":"7bac41f2fcdc718cb06a0caee8796305de3f435a1c3d5a700305f9cb26ab3041","impliedFormat":1},{"version":"e46abaadffe51343e4b50115f22ec40c55efc952e1a5ad8ea83a379e68fdc41b","impliedFormat":1},{"version":"56a44eae80f744ff0ed0ae54ed2c98873d9efaeb94b23102ce3882cbf3c80c87","impliedFormat":1},{"version":"c1608564db1e63ec542694ce8a173bb84f6b6a797c5baf2fdd05de87d96a087f","impliedFormat":1},{"version":"4205f1615444f90977138e01f4c6becc1ae84e09767b84c5a22185ddea2b8ffe","impliedFormat":1},{"version":"823fcbdb4319180e3f9094bc859d85c393200b9568c66f45ba4d5596ace5641d","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"0972ae3e0217c3591053f8db589e40b1bab85f7c126e5cf6cc6f016e757a0d09","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"165181dcaf69484f3a83fef9637de9d56cfa40ee31d88e1a6c3a802d349d32b2","impliedFormat":1},{"version":"823fcbdb4319180e3f9094bc859d85c393200b9568c66f45ba4d5596ace5641d","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"8e517fddbe9660901d0c741161c1ee6674967aaa83c0c84916058a2c21a47feb","impliedFormat":1},{"version":"30f2b1e9cecf6e992ee38c89f95d41aebdb14a109164dd47d7e2aa2a97d16ea9","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"f44bf6387b8c7ab8b6a4f9f82f0c455b33ca7abc499b950d0ef2a6b4af396c2a","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"0a7a83acf2bd8ece46aff92a9dedb6c4f9319de598764d96074534927774223a","impliedFormat":1},{"version":"4f9142ccaefd919a8fe0b084b572940c7c87b39f2fd2c69ecb30ca9275666b3d","impliedFormat":1},{"version":"b80840cbfda90fd14082608e38e9b9c5fde7a0263792c544cddc0034f0247726","impliedFormat":1},{"version":"dcd34efd697cf0e0275eb0889bdd54ca2c9032a162a8b01b328358233a8bcd49","impliedFormat":1},{"version":"98ca8492ccc686190021638219e1a172236690a8b706755abb8f9ff7bb97b63e","impliedFormat":1},{"version":"b61f91617641d713f3ab4da7fdda0ecef11906664550c2487b0ffa8bfbdc7106","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"61cc5aabafaa95e33f20f2c7d3289cf4cab048fc139b62b8b7832c98c18de9ef","impliedFormat":1},{"version":"811273181a8489d26cfa0c1d611178ddbeef85ced1faec1a04f62202697a38a5","impliedFormat":1},{"version":"487d2e38f52af45f6c183407858ea3e0a894fb3723c972140436f40878a27e85","impliedFormat":1},{"version":"15e56c8cb8c5515fe9794c5d223ca5c37a302c62183137a595ba657f5d961527","impliedFormat":1},{"version":"fda3db70b49ad94d08ec58caf0ca052e51d38c51d0461a28669a419c67edb396","impliedFormat":1},{"version":"bb7dd4601aaf41b0313503ffc43142a566a87224cc1720cbbc39ff9e26696d55","impliedFormat":1},{"version":"5ef05c11e0fe4120fb0413b18ca56c78e7fe5843682731fe89c6d35f46d0a4ae","impliedFormat":1},{"version":"02c3a89952ea1b30a3573246649c474cd27b17a26d532abed1e152d5981a6b97","impliedFormat":1},{"version":"d2873a33f67fd7d843ead8cebaeebd51ada53f5fc70d4a61e1874c5d2e3fde4b","impliedFormat":1},{"version":"94c6e873b76d2b5094bd2fddd026db85264bc24faa9cb23db9375f1a770312b5","impliedFormat":1},{"version":"2e8e67d756f97ff13764c81f098b9de13ff91e31028890f3dabe9e8d354f7e47","impliedFormat":1},{"version":"a3476600ff22e7d4845d951dbd0548f8d118f2bfe236aaa6ccd695f041f7a1fc","impliedFormat":1},{"version":"02c3a89952ea1b30a3573246649c474cd27b17a26d532abed1e152d5981a6b97","impliedFormat":1},{"version":"a86a43e07633b88d9b015042b9ea799661fe341834f2b9b6484cfa18a3183c74","impliedFormat":1},{"version":"8994f4c217d03e50957cc4693ae5fd35fd15c60c7d77a31528d90cbeb89311df","impliedFormat":1},{"version":"f5db90ab2b03fc1bc55b4d46df4aa6d4cacdbdd1491bcba0a3cf1a73777204d7","impliedFormat":1},{"version":"9fd04134a11f62f6b1523168945b42a74c35ffe1ea94dfdb08ecddf32218c5c2","impliedFormat":1},{"version":"dbe0161c1a41397e79211136cc6d595b10117aa23ac2f17f7484702ada81bc13","impliedFormat":1},{"version":"b21e6c15895ef16c12925295ebbb39f6731a0c74116f7bfdf5a9085040178bac","impliedFormat":1},{"version":"ea9911c1ac347d631cd840485aef26a8079f0ab64019cc90ae6c97d97dd65034","impliedFormat":1},{"version":"e9ff90fbab735e28c091315b542c620141a76f91bb0d56a14178908905e51b35","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"6fcdcc891e7f13ad8bd34c4de33d76d96c84f06d9ab6629620c8cf08d0cc6bea","impliedFormat":1},{"version":"16a187924c639631e4aab3d6ea031492dc0a5973bae7e1026b6a34116bd9ff5c","impliedFormat":1},{"version":"cd78f65631ff21afa0d2d72f47bd7783126e48c986ff47df22d1dc31347730e5","impliedFormat":1},{"version":"f5db90ab2b03fc1bc55b4d46df4aa6d4cacdbdd1491bcba0a3cf1a73777204d7","impliedFormat":1},{"version":"ad068305ead33649eb11b390392e091dbf5f77a81a4c538e02b67b18eb2c23b3","impliedFormat":1},{"version":"8994f4c217d03e50957cc4693ae5fd35fd15c60c7d77a31528d90cbeb89311df","impliedFormat":1},{"version":"caa292653f273a1cee0b22df63ce67417dbc84b795867bf3cd69f7386bb0f73c","impliedFormat":1},{"version":"cbe901efe10faaa15e14472d89b3a47892afc862b91f7a3d6e31abeb3546a453","impliedFormat":1},{"version":"717b25e589f53597f65f42e0ccff891cd22743511c79b50d534d2fa548484937","impliedFormat":1},{"version":"79d5d086cfd15de8c973783e166e689aa29100d0906ccfef52928504949cf8c2","impliedFormat":1},{"version":"15ecea8b0870ebf135faa352b43b8385f5a809e321bb171062da7ad257c9fd08","impliedFormat":1},{"version":"df9712034821067a7a2a0cf49c7bb90778dc39907083fa47b20c3e22c4e62da5","impliedFormat":1},{"version":"6b2394ca4ae40e0a6e693ad721e59f5c64c2d64b3a6271b4f20b27fce6d3c9c2","impliedFormat":1},{"version":"27ea6d85f1ba97aa339451165cae6992c8a6a7b17d3c8468e3d8dce1c97d16cd","impliedFormat":1},{"version":"05751acbcbf5d3ff3d565e17589834a70feb5638ae7ee3077de76f6442b9e857","impliedFormat":1},{"version":"54edf55c5a377ee749d8c48ca5132944906c09f68b86d1d7db4acc53eea70d57","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"bd0923e7cd1c54c64d7396fbd284983003f0e757bd67f3d6cf3a4e5d394128d7","impliedFormat":1},{"version":"b80840cbfda90fd14082608e38e9b9c5fde7a0263792c544cddc0034f0247726","impliedFormat":1},{"version":"4628d6640af9591f1671e0737b3b7de3abe790ff92686a46d6ca5b2e867162c1","impliedFormat":1},{"version":"50145df9cc9bdb77ac65e4622d11fb896b4730f6f727ffd42337a4fdcd2346da","impliedFormat":1},{"version":"0211a096d47b00b5ba4f6a2557184c649db02cb13a8d63f671428c09818b6df8","impliedFormat":1},{"version":"d32d132c14387d64aa1b776f426a5c3ddcf8211d8764526380dda04f9f4dd776","impliedFormat":1},{"version":"af1c879f74fa27f97cf8ae59ed33421826b7d00647c601cafbbeea129ed5ef5b","impliedFormat":1},{"version":"3b47ab89a1b5a0d3943aace80a68b9af7ae671e359836679ff07536c56ada3fa","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"ae66752cf1b4d08f0b1870dd7c848e491f078116e6395ee5171323c7ec30e92b","impliedFormat":1},{"version":"14a9ec5df1f55a6b37f36d5d91699092119dba1d81defd12151eb0069a26069d","impliedFormat":1},{"version":"ff49d78bd5a137f76e23cc9629105c1d216c43bf68f545acf3f997e838a47ba3","impliedFormat":1},{"version":"842f200637a0e0f390a6512e3e80c8f47c0193bbdff19b5700b070b6b29f1787","impliedFormat":1},{"version":"26a06ef0d60229641de4f9d0ac8566a471b99a3c124e567405a82e77116bee2a","impliedFormat":1},{"version":"f4f34cdbe509c0ae1a7830757a16c1ccb50093b3303af2c301c0007ec2ddf7e0","impliedFormat":1},{"version":"59ba962250bec0cde8c3823fd49a6a25dea113d19e23e0785b05afde795fad20","impliedFormat":1},{"version":"ea930c3c5a401f876daaec88bfc494d0f257e433eaa5f77208cc59e43d29c373","impliedFormat":1},{"version":"318ba92f9fcec5a9533d511ee430f1536e3e833ffe3ea8665d54fe73e28b1ad4","impliedFormat":1},{"version":"adc45c05969fc43d8b5eaac9d5cb96eccf87a6a1bd94498ddd675ea48f1ba450","impliedFormat":1},{"version":"5691d5365f48ff9de556f5883901586f2c9c428bcf75d6eff79615ae1fb67da6","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"a67a76d1886745066bd45956fdc5842812786be2a47285d2c59424882cefd6cf","impliedFormat":1},{"version":"66adf84e776d039acb0207f079934f389147264385fc8847b56481253da99fad","impliedFormat":1},{"version":"d2eee6a9d0b2f4123aba65f6e1bc4df3f973f73a7bdeaa9f76c3c0d3f369bef8","impliedFormat":1},{"version":"8f47038a38222bcbc8551a017ae2e32933ca4e6d2a4ec5cfa01179f1facfa975","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"73c82b8dd8ac2916e7cc44856da0dc795ca9952bb63baa220743d31f62b278e5","impliedFormat":1},{"version":"9e302a99187359decbfba11a58c6c1186722b956f90098bb34d8b161bc342a0d","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"9a06d96357b472809d65ea00b724b4309ba8c9bc1c73eadd3c465e1c336a1e2f","impliedFormat":1},{"version":"ac2b056c5c243b64e85fb8291efd5a1a5481f0bc246b92ea40827ed426ff408c","impliedFormat":1},{"version":"be78757555b38025ba2619c8eb9a3b2be294a2b7331f1f0c88e09bf94db54f3c","impliedFormat":1},{"version":"d68d6551207bf833d92fb7cda4d9428182f8c84eed1743d9a1e7135003e8e188","impliedFormat":1},{"version":"99394e8924c382a628f360a881171304a30e12ac3a26a82aba93c59c53a74a21","impliedFormat":1},{"version":"ed1f01a7eb4058da6d2cde3de9e8463da4351dbab110f50b55e6a7e6261e5e86","impliedFormat":1},{"version":"19ee405d4f1ae4cbacf4361f9a03092a9d69daa3b4ec147c346049d196b5656d","impliedFormat":1},{"version":"6d82ce2eadb900816fb1fa8b62eb4fcf375322bd1fe326b57ef521a0cac3c189","impliedFormat":1},{"version":"19ee405d4f1ae4cbacf4361f9a03092a9d69daa3b4ec147c346049d196b5656d","impliedFormat":1},{"version":"9d344fa3362148f3b55d059f2c03aa2650d5e030b4e8318596ee9bd083b9cf05","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"bfea7300ed7996fd03c8325ce6993eed134984b4bb994b0db8560b206c96f1f7","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"ca87e8ccd63c92b34fc734eee15d8ab2d64f0ffb85d762018bc0df29ca7185b4","impliedFormat":1},{"version":"4628d6640af9591f1671e0737b3b7de3abe790ff92686a46d6ca5b2e867162c1","impliedFormat":1},{"version":"a3913393d42c709b4faea550820241a262a4ba3577f9a00e2f8727eaa92be535","impliedFormat":1},{"version":"5e424456e19df83a4befc6cd24561c2564b7a846b7025a164ce7076ee43828ee","impliedFormat":1},{"version":"887dec57d4c44eaf8f5275c9f5e02721b55c0a34f21f5b6ed08a1414743d8fd9","impliedFormat":1},{"version":"2d53acf155ccbc6b7dca2cfdb01bac84e3571865d925411d2f08ff0445667ea8","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"a7161c3e94028388a80f7091eb2f7f60d2bdde6a58f76876ab30f66c26f6128e","impliedFormat":1},{"version":"381936e93d01e5697c8835df25019a7279b6383197b37126568b2e1dfa63bc14","impliedFormat":1},{"version":"9944093cbb81cc75243b5c779aebfb81fe859b1e465d50cd5331e35f35ef263a","impliedFormat":1},{"version":"fb19163944642017fcdcbdc61999ab21c108334c8b63377184a2a1095698889a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"1bd91f5355283c8fa33ad3b3aace6c4ebb499372943a49f57276f29f55fd62c4","impliedFormat":1},{"version":"6535056b39d5e025505b36ec189302e15af7d197a6afd9a3c853187eb1bea7b5","impliedFormat":1},{"version":"34f97cabd716ba01042042f6523183149c573b8fb15a08a3a9524bf1950216ef","impliedFormat":1},{"version":"01911dee2f91c28782c46d57e2e19e250f7c9db4388f8e9945476379e9392d56","impliedFormat":1},{"version":"95ce7b12742f82bddb85134d8ee20a759c698e5d8beefd559fd6e87112fbf72f","impliedFormat":1},{"version":"0b464435da3dd6473694a2128d49f37c9cf43951455c56f0aa5a940f290c69d2","impliedFormat":1},{"version":"75a5fcf80ec969763cb4a31d2cf8b8531b076d6f1ef8699bd9dacca43d34b571","impliedFormat":1},{"version":"b27117352bfa4f1e6fa6874c3f5518252ae2ff30e345d9e505409a75a232372c","impliedFormat":1},{"version":"d21630c0cd7409e8078cc0aeebf3cf8b915888553d7c9c2d9debd918bfd4bebb","impliedFormat":1},{"version":"7e7a2691f49c7d2623b8a531c9eb4005c22daa57e7789f1982c19fe3c1bf55eb","impliedFormat":1},{"version":"80c54f1d257a28de68ec6c23ca7da374071646182d9a2d2106a91606ebc15f52","impliedFormat":1},{"version":"55ba9e8cb3701eff791fccbe92ef441d19bc267b8aab1f93d4cac0d16fffa26a","impliedFormat":1},{"version":"a40e9367d94ec1db62a406d6e1cb589107ea6ad457af08b544e18d206a6ae893","impliedFormat":1},{"version":"12b260ecee756ba93760308b75a8445f2fe6a1cff3f918cf7e256e3d6d1066cc","impliedFormat":1},{"version":"181de508acbe6fe1b6302b8c4088d15548fb553cb00456081d1e8d0e9d284a24","impliedFormat":1},{"version":"ead149a41e9675c986e6d87c9309e751a8c2d0521839a1902f05ec92b2cba50b","impliedFormat":1},{"version":"d15a8152e6df11bfad2d6813f4517aa8664f6551b0200eca7388e5c143cd200d","impliedFormat":1},{"version":"98884645b61ad1aa2a0b6b208ebaab133f9dd331077a0af4ec395e9492c8d275","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"f660100bff4ca8c12762518ba1c1d62dd72ee1daa7ea42f7eae2f72e993bec6f","impliedFormat":1},{"version":"fd7140ce6b8fc050547d7da8696ed2bcdf4cabc4e65f40f4ac1b080f694711d8","impliedFormat":1},{"version":"8689dabe861fb0bdb3f577bdd9cca3990b14244d1d524c7bdb8d89e229c903a6","impliedFormat":1},{"version":"15d728b5790c39ce9abbd1363e0a5ed03ee6b59a38ee3c4d9d25476641baa7a5","impliedFormat":1},{"version":"95159570a0fc2b007b1a46ed8caf145ad6711030c0c4727cee979a3b770b0634","impliedFormat":1},{"version":"e5446a2b0c44d21a4e2ed885bbdb40a4e39a184f9155f13717993782e313bc7e","impliedFormat":1},{"version":"8683b5b593a5fd2cf99212195ba25106e61a546169068626c8a3745ec6e94bed","impliedFormat":1},{"version":"3f72337d957fd6c87b5c8628c85633d7314b8539cc641ea71a6f93a71f7533c2","impliedFormat":1},{"version":"5d0975641e296dba1ebaf16bb987a2b3abe0a62d18fa1396f57c9d4aaead48e8","impliedFormat":1},{"version":"7b08a55fd84cf8bbee204fa09e8ea402996a648c5af38b52d27231c60d9c8e4d","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"60d3271e8f6a7e952844b716a5f9f71744cb8d6fbeb9adaf35f1735ff7e44aa0","impliedFormat":1},{"version":"632e473a59bfaff109a4405851b56c61aab4a82cedd2a658b37931f98f64ba91","impliedFormat":1},{"version":"178871c23f0cac1cb358aa23f0ba3b1650ec3e962f575e82d33bce7550e55cce","impliedFormat":1},{"version":"94386e32c1da2a3dbff53bfa3aca55ef89397f09bfbb7546890031f246d65716","impliedFormat":1},{"version":"2b96e9789937d863abbb5e33861c941da0d0607fa548f965cdf4e0cf984579ce","impliedFormat":1},{"version":"ea80ad7543efdaeb5ee48a3951f5a32adaa8814fb2a8b9f8296170aa31083455","impliedFormat":1},{"version":"72aad439f7b0cf1c9b28cba809c6b818c72d09f8eeb5978f626d088c2d520f18","impliedFormat":1},{"version":"40d4add4a758635ba84308ecf486090c2f04d4d3524262c13bfb86c8979fac4e","impliedFormat":1},{"version":"72aad439f7b0cf1c9b28cba809c6b818c72d09f8eeb5978f626d088c2d520f18","impliedFormat":1},{"version":"f44c61ac2e275304f62aace3ebc52b844a154c3230f9e5b5206198496128e098","impliedFormat":1},{"version":"924f76dc7507df1c4140262ea2a2d8ef99b8c31e995edefc8271928a3e4807a6","impliedFormat":1},{"version":"3ffc5226ff4a96e2f1a1b12720f0f8c97ac958ac8dd73822bedf6f3ed3c35769","impliedFormat":1},{"version":"924f76dc7507df1c4140262ea2a2d8ef99b8c31e995edefc8271928a3e4807a6","impliedFormat":1},{"version":"9df26a86871f5e0959d47f10bff32add294bf75b8d5a4f77a19dfc41694649d2","impliedFormat":1},{"version":"bfdd4ae390e0cad6e6b23f5c78b8b04daef9b19aa6bb3d4e971f5d245c15eb9a","impliedFormat":1},{"version":"369364a0984af880b8d53e7abb35d61a4b997b15211c701f7ea84a866f97aa67","impliedFormat":1},{"version":"7143d8e984680f794ba7fb0aa815749f2900837fb142436fe9b6090130437230","impliedFormat":1},{"version":"f7b9862117ae65bea787d8baf317dcc7b749c49efeada037c42199f675d56b7b","impliedFormat":1},{"version":"78a29d3f67ea404727199efc678567919ecebbfdc3f7f7951f24e1014b722b46","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"e53b2d245026cefec043621d6648fab344fd04415b47270da9eb4e6796d2a9f4","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"f10a10d90bd1e3e12e1d7d027086a716dd6fa03d251597af77210e7a3081ac0b","impliedFormat":1},{"version":"b2bd6911e91dbb008938121d0fd7df51f00148652090bc9ccde4dc704f36f011","impliedFormat":1},{"version":"1bbdf84753428ed6f1533eabb066f9b467fade05180797e39cb32b4be4ba7d5d","impliedFormat":1},{"version":"e52d0f3e5073519a3a0a69fb0090c180f219fa04fc4053bb2bc5453a61296acd","impliedFormat":1},{"version":"24b30db28923568ff5274ec77c4c70c3e18a62e055f207633b95981ba94b0dee","impliedFormat":1},{"version":"e285a018fca2bcd32f25e2e048076b135086b3bd0d6215b1f72716129dce44ad","impliedFormat":1},{"version":"d9901d27accf8b30a3db21c9537e516427f55abd13ca53283c8237711bd37c16","impliedFormat":1},{"version":"46ded89297bd3856f536a6a990d64831ea69976626669e9371fe12e47a263ceb","impliedFormat":1},{"version":"823f27e48b1e7ff551b90d15351912470ab3cd0fa133bc2e1ddc22bea6c07d23","impliedFormat":1},{"version":"189abcb612878978d45a513656690710591b93860bc9cc2d2bf58c5f2ea9b3ae","impliedFormat":1},{"version":"e6251b50929025156877155e58eff37840da58c85d094e3f128b4f07e03aa66d","impliedFormat":1},{"version":"e6251b50929025156877155e58eff37840da58c85d094e3f128b4f07e03aa66d","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"657bfa91b3233a36081f7030fa35a16728be10e90b926a9e8ae218e9078a5e75","impliedFormat":1},{"version":"c6b1f54c34ab08126f8594801908410a93a64e0dff66df8a226a9b5460054f19","impliedFormat":1},{"version":"ca969c350e570c5fa395c4fb88ea52dfe50014890c445d2834e4f1fe96e93c2d","impliedFormat":1},{"version":"a6f374e4c41a9aaa10213ba98f7d1e520f4cc314c2f20770145124e2f207f11c","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"1481094055c14f5976d55446330cca137adf0b2a39dcae164f1d6460862e5e5b","impliedFormat":1},{"version":"914912142f2648f12b831ad10bcfacfbc02876161de095c479a1ae308067f646","impliedFormat":1},{"version":"b5f7732acfd56640a680acbd12caff991c839c3dfd5a4b48ad90bd7a730d501d","impliedFormat":1},{"version":"8b801973d33012fc9b97dcb37cfd2d5d30eed228b4d342ae3563972ba1004279","impliedFormat":1},{"version":"09c3bb9dac02114c00586e82c825655ea0c5031097667855544d436063322760","impliedFormat":1},{"version":"14e64ceb540cc27093ba1a04948aec14707da94a6ff1d9675efca976e10fea49","impliedFormat":1},{"version":"da6e2dde5747e6e71bdc00a26978fe29027a9e59afe7c375e2c040a07ef9ff25","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"da20ac2b80ec650f4c36df8ebff9493625634329eb0f901a0971dd6619e0978c","impliedFormat":1},{"version":"ef51ac3ae8d6ddc8ee29937a039cbb4a9bfe6ab34267d4c9d998645e73f91237","impliedFormat":1},{"version":"cc45a177fe3864f8a5579ddb987cb5db0ee47c4d39335832635c241b5f98337e","impliedFormat":1},{"version":"3aaf74018283ef4c49f52bcab37f09cd6ec57fff27503090bc4bb75194fd68a8","impliedFormat":1},{"version":"69578d34fa63a8314823b04f6f57a60671755666055a9990b070f5403f21d417","impliedFormat":1},{"version":"c9aa17bf9f1d631f01764ad9087de52f8c7e263313d79ac023f7cd15967b85cb","impliedFormat":1},{"version":"78d05f11e878fe195255ac49d0c2414a1c7fa786b24e8d35c0659d5650d37441","impliedFormat":1},{"version":"b93a1522b0ae997d2b4dc0e058c1d34f029b34370ee110b49654deeef5829a41","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"ae2104bdc52ab3722b5c0cfa26aa65b077e09d7288695f9e0ee9ffde08721b3d","impliedFormat":1},{"version":"a4038d37487d8535f99ba99adc4a01b08f038515dd939e57bd80a3743c0e5662","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"483095dc7d04bc24cc55e72a807fa8d786a52981068c6f484947f63956b0fa92","impliedFormat":1},{"version":"4539884fadd3b91977560c64de4e5a2f894a656a9288882e1307ba11c47db82e","impliedFormat":1},{"version":"430016e60c428c9c8bfa340826ff7ed5988e522348838700f3c529dc48376c10","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"2e1b0586468b145f432257bfc0dc8d40a82b04ebd00c5f92efdde426d14d122b","impliedFormat":1},{"version":"976d79fce50c222b3aa23d34e4165e1c8424060c3744a4a5b5834bbc644e64a6","impliedFormat":1},{"version":"d61d7221ed4b74db0568ffae7765f6c2a48afc64a076dd627e98dfecd1ad9897","impliedFormat":1},{"version":"89ac12f3bd077e0d31abc0142b41a3dbbdb7ae510c6976f0a957a1f3ca8c46c9","impliedFormat":1},{"version":"694d279f9a6012c39bba6411e08b27706e0d31ea6049c69ff59d39a50de331cc","impliedFormat":1},{"version":"e27f95d214610d9d7831fdeccba54fbe463ae7e89bd1783d828668072c2d2c92","impliedFormat":1},{"version":"ed48328b38a82b98abf873153e939c9baed42cbd5d5289830dd832c552db5024","impliedFormat":1},{"version":"6ca43ca6b5f1794be3eee4993c66f15083c3b47ee45615163ee49f450e4b464a","impliedFormat":1},{"version":"8d8381e00cd14cf97b708210657e10683f7d53a4eddcfc3f022be2c9bdf591dd","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"ec85bf4283c2ec8108b0b6161f155aeedfc770f42dca27bb6fca2cfb0abf1a8a","impliedFormat":1},{"version":"ec2ba248e2ad73cfd1989cb7f53ff1df5612f63b628e03a472308c1bab10c0f9","impliedFormat":1},{"version":"ea763067ac7adab4741f87de9fec3fc154ac1f3578b7e3bc0c64b42c6f6c912e","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"d54fa16b15959ed42cd81ad92a09109fadbb94f748823e2f6b4ad2fbbee6e01f","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"2e2ffb8593c9db471bac9f97c0b1f1c7ef524946a462936e5e68858ac3e71566","impliedFormat":1},{"version":"d4c081ae5c343c754ac0dd7212f6308d07f55ab398cee4586ee0a76480517ae5","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"a4f2c605bbc73124b1bb76faa66be28937ccfb7f5b77c45cd8022071bd53696c","impliedFormat":1},{"version":"be4c58de8fd3ddd0e84076c26416ce5ffcf193a1238704692e495bc32e0a6ec5","impliedFormat":1},{"version":"af9491fcc19d5157b074871bdceafc18dd61972020fb8778c7d3cd789cd8186a","impliedFormat":1},{"version":"64da3dee7d98bdc4b99b24de094a08ffb2dda8aa14270cd51fc936dc8af1cdb2","impliedFormat":1},{"version":"a4038d37487d8535f99ba99adc4a01b08f038515dd939e57bd80a3743c0e5662","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"152532087c2a91adb4527e96ccd7b3640f1b08c92301fa2f41ed6a53130bda67","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"6e39d03aa07f268eed05dd88e1bd493cb10429c1d2809e1aaa61fbcd33978196","impliedFormat":1},{"version":"aa7384441d37522532179359964184e5c8cf649db32a419542e7b5605208b45c","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"36d27819ece3bf0eefe61ecda9e3aa2e86b5949c89dba79f17dd78a2c4587a61","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"18a20ae79049147b460771dfd6b63b3b477772d763c26b367efa499c98e9fb5f","impliedFormat":1},{"version":"4c91908ebcc1b1c91f5c9cd7e9ffff83fc443e6926013b0b0082a6c2778b729e","impliedFormat":1},{"version":"ee51a4032beba0b38ff75838b386627a38c53008b8ca350bb42f192d0fb3cf58","impliedFormat":1},{"version":"b14b8756b166914ab1cb68c44bb579566833449d5e9d68655726f6ffc6d5e457","impliedFormat":1},{"version":"a09ae8631b5e442bbcdb93e3b60d6f71a54d192452af841616e2b49c5a03fb26","impliedFormat":1},{"version":"7a254103740333c7fb870f95ab9a26fb028cb298478f43e4750b8eddefafa11f","impliedFormat":1},{"version":"d54b449b0eff66bc26e09593df44512725b9e9fce4d86ea436bed9e7af721ff1","impliedFormat":1},{"version":"91991180db9a4d848bd9813c38a56d819a41376a039a53f0e7461cc3d1a83532","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"637ffc16aeaadb1e822bffc463fcc2ca39691dea13f40829c1750747974c43d4","impliedFormat":1},{"version":"7955f3e66404ff9a4ac41f40b09457fe1c0e135bde49e4d77c3ea838956041bf","impliedFormat":1},{"version":"f6d23ab8669e32c22f28bdbdf0c673ba783df651cafcbdcc2ead0ff37ba9b2b5","impliedFormat":1},{"version":"c90ef12b8d68de871f4f0044336237f1393e93059d70e685a72846e6f0ebbbff","impliedFormat":1},{"version":"ecefe0dd407a894413d721b9bc8a68c01462382c4a6c075b9d4ca15d99613341","impliedFormat":1},{"version":"9ec3ba749a7d20528af88160c4f988ad061d826a6dd6d2f196e39628e488ccd8","impliedFormat":1},{"version":"71ce93d8e614b04d49be0251fb1d5102bb248777f64c08078ace07449700e207","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"4818c918c84e9d304e6e23fdd9bea0e580f5f447f3c93d82a100184b018e50f5","impliedFormat":1},{"version":"6e39d03aa07f268eed05dd88e1bd493cb10429c1d2809e1aaa61fbcd33978196","impliedFormat":1},{"version":"eab3b41a54d5bc0e17a61b7b09639dc0d8640440e3b43715a3621d7fa721ae85","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"36d27819ece3bf0eefe61ecda9e3aa2e86b5949c89dba79f17dd78a2c4587a61","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"ce8eb80dad72ac672d0021c9a3e8ab202b4d8bccb08fa19ca06a6852efedd711","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"d12e9c3d5e2686b5c82f274fb06227748fc71b3a6f58f7b3a6f88f4b8f6921fb","impliedFormat":1},{"version":"5f9a490be2c894ac65814a1a9e465b99882490ed3bce88c895362dc848f74a8d","impliedFormat":1},{"version":"2d5935948312241d3195b5e24df67775c6736dec1e1373efb1b6f04447106867","impliedFormat":1},{"version":"686ccf874ccbf999a155208a7ec8358a718d211f779980c2fe7cca176025d769","impliedFormat":1},{"version":"48bf56f3c8b3d0b27f94587996400c129773ab9c4810354d89850b0bee92b3d7","impliedFormat":1},{"version":"e6e9bdd2f65408a0b52d8e8ca9ddb7827c5f3496561788c974e4f2fb485427eb","impliedFormat":1},{"version":"193772121770797ee600739d86de128cd7244e3e3e101684473eb49590dbfce1","impliedFormat":1},{"version":"7a6208fa971deb77dbd7c59d56f7eb5b2516d76a3372a55917b75fc931c44483","impliedFormat":1},{"version":"b9aa4ed5dc603ad443dac26b9c27b0680b1cf4614f321b8d3663e26c1b7ef552","impliedFormat":1},{"version":"8613d707dc7f47e2d344236136010f32440bebfdf8d750baccfb9fad895769ee","impliedFormat":1},{"version":"59ebb6007bce20a540e273422e64b83c2d6cddfd263837ddcbadbbb07aa28fcc","impliedFormat":1},{"version":"23d8df00c021a96d2a612475396e9b7995e0b43cd408e519a5fb7e09374b9359","impliedFormat":1},{"version":"9a3c859c8d0789fd17d7c2a9cd0b4d32d2554ce8bb14490a3c43aba879d17ffb","impliedFormat":1},{"version":"431dc894a90414a26143bbf4ca49e75b15be5ee2faa8ba6fcc9815e0ce38dd51","impliedFormat":1},{"version":"5d5af5ceb55b5ec182463fe0ffb28c5c0c757417cbed081f4afd258c53a816c5","impliedFormat":1},{"version":"f43eee09ead80ae4dcfc55ba395fe3988d8eb490770080d0c8f1c55b1bd1ef67","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"4c9784ca0ab39916b498c54db858ea27c929777f161a2450f8712a27cec1b017","impliedFormat":1},{"version":"9c92db9255eab1e3d218bdeca593b99355bbf41fa2a73a9c508ad232a76cda96","impliedFormat":1},{"version":"bf2cc5b962f3823a8af297abe2e849227dbfb3a39a7f7301c2be1c0a2ecb8d32","impliedFormat":1},{"version":"eaed6473e830677fd1b883d81c51110fcb5e8c87a3da7a0f326e9d01bf1812ff","impliedFormat":1},{"version":"3ac0952821b7a43a494a093b77190a3945c12f6b34b19f2392f20c644ac8d234","impliedFormat":1},{"version":"ed5877de964660653409f2561c5d0a1440777b2ef49df2d145332c31d56b4144","impliedFormat":1},{"version":"c05da4dd89702a3cc3247b839824bdf00a3b6d4f76577fcb85911f14c17deae5","impliedFormat":1},{"version":"f91967f4b1ff12d26ad02b1589535ebe8f0d53ec318c57c34029ee68470ad4a3","impliedFormat":1},{"version":"f6ac182bf5439ec39b1d9e32a73d23e10a03fe7ec48c8c9ace781b464ecc57c3","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"687b26db97685fcadeb8e575b6bc252ea621fef8217acd2bb788ce781a4b05b3","impliedFormat":1},{"version":"e4a88ca598bf561ec253c0701eea34a9487766c69a8d8e1b80cf67e60dcc10d7","impliedFormat":1},{"version":"281cf6513fcf7b7d88f2d69e433ebbd9248d1e1f7571715dd54ca15676be482e","impliedFormat":1},{"version":"dc9f827f956827ec240cec3573e7215dc08ed812c907363c6653a874b0f5cabb","impliedFormat":1},{"version":"baa40541bd9b31a6f6b311d662252e46bad8927d1233d67e105b291d62ace6e6","impliedFormat":1},{"version":"d3fa2e4b6160be0ab7f1bc4501bf0c969faa59c6b0f765dc8ca1000ca8172b18","impliedFormat":1},{"version":"cf24c5c94e5e14349df49a69fb963bee9cd2df39f29ddd1d4d153d7a22dfb23f","impliedFormat":1},{"version":"18a20ae79049147b460771dfd6b63b3b477772d763c26b367efa499c98e9fb5f","impliedFormat":1},{"version":"c5ad2bd5f2243c6fade8a71a752b4333b0ba85ae3ea97d5323f7d938b743cb26","impliedFormat":1},{"version":"cf1e804f283ae1ca710f90dba66404c397b7b39682dbdfa436a6b8cc0b52b0ab","impliedFormat":1},{"version":"25fd641b32d4f7d6811cec4b00c0c9a74cb8822ec216f3b74bae205a32b1de08","impliedFormat":1},{"version":"658f07f1b7c327ecc8b18ed95ada19a90f9fc3f0282d536ca9d6cd2d597631f4","impliedFormat":1},{"version":"35c8e20c61bffc19a0391f42db2fe8f7bb77caa414bd2145a8891826bfdb9667","impliedFormat":1},{"version":"658f07f1b7c327ecc8b18ed95ada19a90f9fc3f0282d536ca9d6cd2d597631f4","impliedFormat":1},{"version":"b3279a079db8ea0c8b76f7f3098f4b10266c3bb24fa21e5838fe6008e3d40043","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"8aec152ae554311c39f87fc5ec3c1f4c5d5d44e1145704782a4fdd6b16c2f1d7","impliedFormat":1},{"version":"9b4a1b563bc6d3d02a4a9d3e72bf699d486a6b117fdcf29199d49d3650abe122","impliedFormat":1},{"version":"803e87c5c27720886ff9f591a47e3281b02bf737f6c67964d72a4d8e7b905a21","impliedFormat":1},{"version":"ce762eb7d3137473f6b50c2cd5e5f44be81334550d9eb624dadb553342e9c6ed","impliedFormat":1},{"version":"3a4d63e0d514e2b34487f84356984bd4720a2f496e0b77231825a14086fb05c1","impliedFormat":1},{"version":"22856706f994dec08d66fcbf303a763f351bc07394fb9e1375f0f36847f6d7a5","impliedFormat":1},{"version":"1f2b07381e5e78133e999e7711b84a5d65b1ab50413f99a17ffccfc95b3f5847","impliedFormat":1},{"version":"39aa109cb3f83642b99d9f47bf18824f74eaaa04f2664395b0875a03d4fc429a","impliedFormat":1},{"version":"15ca7cf99d213ac6a059a5f81ff17dd2c0d4e31260821719ef7e78ea6163f518","impliedFormat":1},{"version":"ee130bd48bc1fb67a0be58ab5708906f8dc836a431b0e3f48732a82ad546792e","impliedFormat":1},{"version":"9d32f274f0b2388e27a83b6b88b33616a4b73b4d045c00d814e942c07a5c9a57","impliedFormat":1},{"version":"06a6defbd61ec1f028c44c647c7b8a5424d652b3330ff4f6e28925507e8fde35","impliedFormat":1},{"version":"9d32f274f0b2388e27a83b6b88b33616a4b73b4d045c00d814e942c07a5c9a57","impliedFormat":1},{"version":"15ca7cf99d213ac6a059a5f81ff17dd2c0d4e31260821719ef7e78ea6163f518","impliedFormat":1},{"version":"9df4d5273810ea069628b1efd0ea6ca9932af9694bfbc8dcea17c8253f1790c2","impliedFormat":1},{"version":"9b3ca716ad96d961aa8f2bab5fbd6752637af2da898f54c8d4021ef8ab2607d2","impliedFormat":1},{"version":"60d53d724e5854f545fd4753881466043628eb886159a73568878f18b3020afe","impliedFormat":1},{"version":"c53d0b758384bd45cd3a051a5227805b57eae8f2142e906d65ae97c8868fd45f","impliedFormat":1},{"version":"a844bbf1cb0bb844743b2d78eee9bdc78df80a98989deab32ff8cd3228b41289","impliedFormat":1},{"version":"b641f9357511425b12ad981f9ba66d964fc114b78a5761ead8595599f036a22f","impliedFormat":1},{"version":"3537c3f024e3bed94fedcce3444fca3c1bce744942912a5a4857f7050ab25429","impliedFormat":1},{"version":"96a5c70389556c62902487f56bb34259ef57439a4cba6c9bdbbbb55225b32e63","impliedFormat":1},{"version":"54895ba2b529f7c369600228dbb88c842c311d1fb7de4ccbc43123b357c26a90","impliedFormat":1},{"version":"9d0050ae8481d6e0731ed80b55f6b475ae3a1cffbc61140e92816a0933dba206","impliedFormat":1},{"version":"68867d1d1560d31165f817de3fceb4b2bedbd41e39acdf7ae9af171cdc056c47","impliedFormat":1},{"version":"1c193e68e159296fded0267475b7172231c94e66b3d2f6f4eb42ffde67111cc5","impliedFormat":1},{"version":"f025c51bcc3c7dacbedb4b9a398815f4d5c6f4c645db40880cee4ac6f89588de","impliedFormat":1},{"version":"b94704c662a31e0d061abb006d38f6211ade97422f0ae45d751ef33d46ce3042","impliedFormat":1},{"version":"c3e2f2b328bd55ae9a401673bd33f86d25a7d53a4f5e1fad216f5071c86c0b79","impliedFormat":1},{"version":"5f6e56ac166b7a5bde756afd2e573af1e38fdd5f10ddb72e46bc44f3c0a42369","impliedFormat":1},{"version":"9b65fd7edfcf3c4c6538d735d269647edc14856dc062e9dde80412c45ff2cf29","impliedFormat":1},{"version":"fbb26af430ebc8743161f6026a0722a4cee3df8c08bdc2610a1d037f733fa823","impliedFormat":1},{"version":"65de396834768bf2b3548447b84b774310f83f33d00f9fb951c1b338dd9b5395","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"75b022f6a48640ca4e048da35132eef2cb9445680c7e1080021ccc15f4d2bf59","impliedFormat":1},{"version":"ea7c9f9c4b1cd2573d49dd628d446fa7611052e00ea1a3aa385a83a7b07c7fbb","impliedFormat":1},{"version":"a74eec58a6011f6ba3d6bbe4eacea0935f7fce9ad34f8c8bd8ed8872ae68f826","impliedFormat":1},{"version":"6bd326162475f1661612f9bb68aa7833e548c7a726940f042e354086cd9b7c2d","impliedFormat":1},{"version":"4b3d55b3d962f8773ea297be1b7f04093a5e5f0ea71cb8b28cef89d3d66f39b0","impliedFormat":1},{"version":"39d7517763d726ce19f25aacf1ccb48ec4f1339978c529abdf88c863418b9316","impliedFormat":1},{"version":"4ce8ae09e963394e7ffe3a5189007f00a54e2b18295585bb0dae31c7d55c1b3f","impliedFormat":1},{"version":"b29b65017a631dff06b789071cdf7a69f67be35238b79f05e5f33523e178feaf","impliedFormat":1},{"version":"58cb40faa82010f10f754e9839e009766e4914586bdb7a4cceff83765fa5e46c","impliedFormat":1},{"version":"efa190d15d9b3f8a75496c9f7c95905fca255a7ce554f4f0b91ba917b61c3b7e","impliedFormat":1},{"version":"303fd31bbed55c8cdf2d3d9851668f4e67746f0a79861a3b4d947a6c1c9e35c5","impliedFormat":1},{"version":"0fe6e8d738df018108bd3ca0e208dfa771d4e34641242b45423eca7d7ade80a7","impliedFormat":1},{"version":"8210e3bdbeeb9f747efdf7dad7c0ed6db9d13cd0acd9a31aa9db59ddbbac5a15","impliedFormat":1},{"version":"d6791734d0fce30014c94846a05cb43560bce15cfdc42827a4d42c0c5dafa416","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"8c4f5b888d7d2fc1283b7ce16164817499c58180177989d4b2bd0c3ebd0197f7","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"ea7c9f9c4b1cd2573d49dd628d446fa7611052e00ea1a3aa385a83a7b07c7fbb","impliedFormat":1},{"version":"3108920603f7f0bbf0cebce04bcaf90595131c9170adb84dc797e3948f7b6d06","impliedFormat":1},{"version":"8aded022b77ae3c07af72765bca9421f2d990814e0f4bfca0aa97395aa4c9010","impliedFormat":1},{"version":"f817987f543a452afa3035a00aa92800dbd7ff3246fcbe4cecb29bc18552b081","impliedFormat":1},{"version":"6ab1e8b5d0a0f4123b82158ea498222a5eacbffa1354abe8770030ba722c13b7","impliedFormat":1},{"version":"3cda89b540ed1ea9a3d1e302a489a4157a98b62b71c7abb34f8f15c13da9717a","impliedFormat":1},{"version":"a1ebece06e1ac47fb3a1b07997e57aa2e6a8f5ece26ea3c4a4fcb591e05d1e05","impliedFormat":1},{"version":"8aded022b77ae3c07af72765bca9421f2d990814e0f4bfca0aa97395aa4c9010","impliedFormat":1},{"version":"fb3b5ff3f5fe7767c07b755f2c22ce73ba46d98e6bc4a4603fde8888eed14e19","impliedFormat":1},{"version":"41c53632da296cf700f8553a48522e993949ea8499ceac4a483d1813beed3017","impliedFormat":1},{"version":"03b97deb8a168b27af94dca96eba747e19faf077445102d52c618210829cb85f","impliedFormat":1},{"version":"6a3589af6b9ec75cd87d9516ccfb9b06ab6be6f938790aeb4b1cd4dbaef92c45","impliedFormat":1},{"version":"722a667fe3b290be746d3ea6db20965ec669614e1f6f2558da3d922f4559d9c4","impliedFormat":1},{"version":"0f1c68ddd4573b2e135748377c3705a96d6a6c123910b00d0c7e8dc2edcd7f6b","impliedFormat":1},{"version":"a63781a8662205b9b6d2c7c5f3bad1747a28e2327804477463ebb15e506508e1","impliedFormat":1},{"version":"0f1c68ddd4573b2e135748377c3705a96d6a6c123910b00d0c7e8dc2edcd7f6b","impliedFormat":1},{"version":"80d8f42128925d6f1c82268a3f0119f64fd522eec706c5925b389325fb5256de","impliedFormat":1},{"version":"b4c189c9be8cf4a7cce177fc49678e29d170e67279195207f36a4f4d184d60f2","impliedFormat":1},{"version":"d16a18dfc505a7174b98f598d1b02b0bf518c8a9c0f5131d2bd62cfcaaa50051","impliedFormat":1},{"version":"b4c189c9be8cf4a7cce177fc49678e29d170e67279195207f36a4f4d184d60f2","impliedFormat":1},{"version":"d3ceb0f254de2c13ffe0059a9a01ab295ccf80941c5429600ffdbaaec57410a7","impliedFormat":1},{"version":"8e172ba46195a56e4252721b0b2b780bf8dc9e06759d15bc6c9ad4b5bb23401d","impliedFormat":1},{"version":"41c53632da296cf700f8553a48522e993949ea8499ceac4a483d1813beed3017","impliedFormat":1},{"version":"0fe5f22bc0361f3e8eacf2af64b00d11cfa4ed0eacbf2f4a67e5805afd2599bc","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"226dc98afab126f5b99f016ec709f74c3bcc5c0275958613033e527a621ad062","impliedFormat":1},{"version":"ec7197e94ffb2c4506d476df56c2e33ff52d4455373ecb95e472bb4cedb87a65","impliedFormat":1},{"version":"343865d96df4ab228ff8c1cc83869b54d55fa764155bea7db784c976704e93ec","impliedFormat":1},{"version":"f3f8a9b59a169e0456a69f5c188fb57982af2d79ec052bf3115c43600f5b09e4","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"15ddffc9b89470a955c0db3a04aec1f844d3f67e430b244236171877bdb40e50","impliedFormat":1},{"version":"7ca1ed0b7bd39d6912d810562413fb0dad45300d189521c3ca9641a5912119a5","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"74766ac445b27ae31cc47f8338fd0d316a103dd4d9eb766d54b468cb9aacbf0e","impliedFormat":1},{"version":"65873070c21b3ce2ccdf220fe9790d8a053035a25c189f686454353d00d660f9","impliedFormat":1},{"version":"d767c3cc8b1e117a3416dda1d088c35b046b82a8a7df524a177814b315bde2e3","impliedFormat":1},{"version":"bf834cd64464f9217cb642a48c2f5f5f1cd509e13088adac6773715fb8536212","impliedFormat":1},{"version":"40258ea27675f7891614c8bd2b3e4ee69416731718f35ec28c0b1a68f6d86cd6","impliedFormat":1},{"version":"bf834cd64464f9217cb642a48c2f5f5f1cd509e13088adac6773715fb8536212","impliedFormat":1},{"version":"c61aa5b694977909ef7e4a3fdad86b3c8cd413c8d8e05b74a2def595165ba7ce","impliedFormat":1},{"version":"bfef3048352341739d810997dcd32f78527c3c426fac1bbb2b8c14293e1fa505","impliedFormat":1},{"version":"1dd31462ed165900a141c2e159157be0e8701ce2a2ed0977636f1d021894887d","impliedFormat":1},{"version":"872321f2e59009fad1f2efde489b20508a3631e16a86860740044e9c83d4b149","impliedFormat":1},{"version":"fa381c11f336210a8c10d442c270c35165dcf6e76492618ee468dba325a3fc98","impliedFormat":1},{"version":"857857dbb4d949686de80a138aeab8e669d23397100dc1e645190ff8be5787de","impliedFormat":1},{"version":"d6a9fe9c13a14a8d930bb90f3461dc50945fa7152e1a20a1f5d740d32f50b313","impliedFormat":1},{"version":"4162a1f26148c75d9c007dd106bd81f1da7975256f99c64f5e1d860601307dad","impliedFormat":1},{"version":"63f1d9ad68e55d988c46dab1cbc2564957fcbd01f6385958a6b6f327a67d5ff4","impliedFormat":1},{"version":"8df3b96fbafb9324e46b2731bb267e274e516951fbf6c26165a894cae6fd0142","impliedFormat":1},{"version":"822e61c3598579070f6da4275624f34db9eb4af4c27a2f152a467b4a54f4302f","impliedFormat":1},{"version":"a8f83bf864a5dea43d30c9035d74069b1820f0c49824960764cf21d6bfbb8e66","impliedFormat":1},{"version":"f9449f2b807f14c9ff9db943e322385875cca5faa26775f64a137e4d1a21b158","impliedFormat":1},{"version":"8855c7125e06a2001f726b4f2f9905e916d122377f7d938936fb49606ccb55c5","impliedFormat":1},{"version":"8855c7125e06a2001f726b4f2f9905e916d122377f7d938936fb49606ccb55c5","impliedFormat":1},{"version":"d24f0b133a979dc915411e1c76d2dada47e3624b42d5838e9d6b9eef1f067cc7","impliedFormat":1},{"version":"755611714dbab5b9b351b51e7875195f83bb26169ae6b31486dcb1e6654ed14c","impliedFormat":1},{"version":"a82213450f0f56aab5e498eaae787cf0071c5296ea4847e523cf7754a6239c99","impliedFormat":1},{"version":"f2882c5afda246fa0c63489d1c1dff62bf4ddf66c065b4285935d03edaec3e71","impliedFormat":1},{"version":"d38c1b0fd8bc7e301fd467a2afd6d32b2457813c48c16afabc06d2ca5b6bda41","impliedFormat":1},{"version":"d38c1b0fd8bc7e301fd467a2afd6d32b2457813c48c16afabc06d2ca5b6bda41","impliedFormat":1},{"version":"4ed8f12983c82690e8fecd9b24f143d4a7c86d3156be7b2bff73e0761f820c8c","impliedFormat":1},{"version":"1d920699becb8e60a0cbbc916d8559a3579b204dd21655dd242c98fd8ae986ea","impliedFormat":1},{"version":"c278288183ec3690f63e50eb8b550ef0aa5a7f526337df62474f47efea57382b","impliedFormat":1},{"version":"3c0486004f75de2873a34714069f34d6af431b9b335fa7d003be61743ecb1d0a","impliedFormat":1},{"version":"99300e785760d84c7e16773ee29ac660ed92b73120545120c31b72166099a0e4","impliedFormat":1},{"version":"8056212dad7fd2da940c54aeb7dfbf51f1eb3f0d4fe1e7e057daa16f73c3e840","impliedFormat":1},{"version":"e58efb03ad4182311950d2ee203807913e2ee298b50e5e595729c181f4c07ce3","impliedFormat":1},{"version":"67b16e7fa0ef44b102cc4c10718a97687dabfa1a4c0ba5afe861d6d307400e00","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"f29c608ba395980d345144c0052c6513615c0ab0528b67d74cacbfac2639f1d4","impliedFormat":1},{"version":"e094afe0a81b08444016e3532fbf8fae9f406cdb9da8dbe8199ba936e859ced7","impliedFormat":1},{"version":"e4bcab0b250b3beb978b4a09539a9dfe866626a78b6df03f21ae6be485bc06e2","impliedFormat":1},{"version":"a89246c1a4c0966359bbbf1892f4437ff9159b781482630c011bb2f29c69638f","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"98ca77869347d75cd0bb3d657b6dcd082798ef2419f1ab629ccf8c900f82d371","impliedFormat":1},{"version":"73acfe8f7f57f1976d448d9569b345f907a6cf1027a08028fe5b8bb905ef8718","impliedFormat":1},{"version":"ed8a781d8b568d8a425869029379d8abc967c7f74d6fe78c53600d6a5da73413","impliedFormat":1},{"version":"90ead73acfd0f21314e8cbef2b99658d88cc82124cfc20f565d0bdda35e3310a","impliedFormat":1},{"version":"8ecfec0e00878d6d26a496cf5afc715b72c3da465494081851da85269b0aef8e","impliedFormat":1},{"version":"4c78fccd1c5cd8eebde42cc078e7332f3d9b4eb1a542d9a5ec66899dfd71b93e","impliedFormat":1},{"version":"4c78fccd1c5cd8eebde42cc078e7332f3d9b4eb1a542d9a5ec66899dfd71b93e","impliedFormat":1},{"version":"e54b165a2a5a5fbcf4bcd09176e4388b514ca70a20635841937f1cc36e37fbef","impliedFormat":1},{"version":"6eb0dcefcf4cc9088174209028db705572e7fb7e38f3f93275bf6778afa2cd19","impliedFormat":1},{"version":"fa572fa0d1b1b1a7d356d5942b1d57f342880a68d1bf1ab5d00490221c471c18","impliedFormat":1},{"version":"17694dd0223346fa0a17e87e9ce00335569166368357b9963571aa623c5e3c27","impliedFormat":1},{"version":"207d46e6e557df62460be9021502fc3af96c927cef0cc5add32cb6f2d60b2e23","impliedFormat":1},{"version":"cf0cf6556adc9178a6251d9b12837e5d514b805cebe8de6d7a16e1e4248ec1ef","impliedFormat":1},{"version":"3d3d28a294ca0d5caea84d58eec474891dd1df7015f8fb2ee4dabf96d938333c","impliedFormat":1},{"version":"0b5b95f3b76e6cc9b716e08274d0f7486bee9d99e42dd6a99c55e4cb4ff5569e","impliedFormat":1},{"version":"94fb6c136acee366e3a4893df5ddbecadde49738de3c4d61a2923c6ada93e917","impliedFormat":1},{"version":"95669998e1e807d41471cebed41ede155911da4b63511345571f5b7e13cbef9c","impliedFormat":1},{"version":"48cca9861e6f91bde2435e5336b18bdc9ed3e83a6e7ea4cf6561e7f2fee4bad6","impliedFormat":1},{"version":"b6b8be8a70f487d6a2fd80b17c4b524b632f25c6c19e76e45a19ad1130209d64","impliedFormat":1},{"version":"76d7fadbb4ff94093be6dd97ea81a0b330a3a41fc840c84a2a127b32311200e6","impliedFormat":1},{"version":"856a8b0060b0e835bccba7909190776f14d8871b8170b186d507d3e12688086d","impliedFormat":1},{"version":"e39aaeef0aea93bdda6f00d27ca9ebda885f233ecc52b40e32db459916f24183","impliedFormat":1},{"version":"14f3c0b1b5e6adac892607ecefc1d053c50bc8a5f14d05f24e89e87073d2f7e3","impliedFormat":1},{"version":"f877dcc12cc620dede9c200625692cf614b06aadc026f6b59e5967cd2e30cbc4","impliedFormat":1},{"version":"5a37547f8a18bc0738e670b5043819321ae96aee8b6552266f26d8ce8f921d17","impliedFormat":1},{"version":"4d3e13a9f94ac21806a8e10983abcf8f5b8c2d62a02e7621c88815a3a77b55ae","impliedFormat":1},{"version":"938cb78a2ad0894a22e7d7ebd98cdc1719ee180235c4390283b279ea8616e2a9","impliedFormat":1},{"version":"84ba4c2edb231b1568dae0820f82aca1256a04599d398ec526615c8a066f69ec","impliedFormat":1},{"version":"cd80a8f16c92fe9f03899f19c93783dce3775ef4c8cdf927ac6313354765a4f2","impliedFormat":1},{"version":"25df98970954ccd743fe5e68c99b47d0e02720e2bf6584a6de60e805395b6bf7","impliedFormat":1},{"version":"251983cb99df8c624ca1abd6335ca5d44d0dd7cdcab3ef9c765b4acc79fae8fb","impliedFormat":1},{"version":"7c4965812974ebd1333cb09f95c4a3669e19008dfbb1e931321e08ae1f7cff09","impliedFormat":1},{"version":"31d3f4757bece74c888df52c8bdc4373e3f58deb518000051cadb5e85deb54de","impliedFormat":1},{"version":"a2be4cad298b3b474a0a71c1dd78a8bfc70b322f44704cf4329aecb873687a3a","impliedFormat":1},{"version":"ca8b04bea4ba551b47ddea18e385e76e555a9f7ff823dcae668d05e255fdc241","impliedFormat":1},{"version":"de0d160ecc8e643727bb93018015ae89510d59b7bdad4550f4318fba0a0ce2e6","impliedFormat":1},{"version":"acf3fff2afb5ceb54bd5ddb697b1d337338e3c23b93385f100a2046cfa700184","impliedFormat":1},{"version":"a2be4cad298b3b474a0a71c1dd78a8bfc70b322f44704cf4329aecb873687a3a","impliedFormat":1},{"version":"15c7f60f69f663374a7bc57afe164e70e3b6310bd1ee476ba911646b09c7852b","impliedFormat":1},{"version":"d71becf074ceaa0e91558fe51ed8640fa83a0fbf45a31e8069716edbf38de99a","impliedFormat":1},{"version":"ef681b070e9f3b9b28f1886bbe67faa12237c8d4691604a1f1cba614a10ef2e1","impliedFormat":1},{"version":"b15f5e077245fef1ecf45327fd94aa67fc4da288bfd42bf1b8a80f297afd561e","impliedFormat":1},{"version":"b7091d79a6e7be7bb10ca9477b6c71db4cf7b44f155912266ecfba92c1a126c1","impliedFormat":1},{"version":"e585a113e0abcaf3022f5cf1318e17f299f0935d7b389a23dcad9074c3922946","impliedFormat":1},{"version":"ae545310dfa53a7b33f574e621b14f423373dea930218d2ad290b4da0c5e8e50","impliedFormat":1},{"version":"ae545310dfa53a7b33f574e621b14f423373dea930218d2ad290b4da0c5e8e50","impliedFormat":1},{"version":"4428e4d440e1914859e8aee558f90b4829c6a45b717078490dfc2d297dcef46c","impliedFormat":1},{"version":"ad205fc7116808509e19ee71277d8da74157751d7388f0134d91c009b987f69f","impliedFormat":1},{"version":"4428e4d440e1914859e8aee558f90b4829c6a45b717078490dfc2d297dcef46c","impliedFormat":1},{"version":"8900bf61f4ce9517567cc6c9e41638a5bd0c4a0e9cc094190bc07644bbeedf24","impliedFormat":1},{"version":"cf5414a97c345c8f3294e0513a7613f5a263e1b56b3a61b810ba8279716fd38c","impliedFormat":1},{"version":"7778bc213be81351a01867789728c7780467c84e3ec94cfcef53a4e2dccf1b57","impliedFormat":1},{"version":"41a934d2efbb6cb08b205a76206fb015ebda692db4d78382ec5bec9689d6f4ac","impliedFormat":1},{"version":"67881ba5e3ec9746193075b020876fa94a1437fd78b2e0ad6351d11e1062941a","impliedFormat":1},{"version":"953cbf62815703fa9970c9cfec3c8d033da04a90c2409af6070dcc6858cf6b98","impliedFormat":1},{"version":"68065ce3af3ef8599af8338068cf336be35249eff281ee393186a0ef40db3abf","impliedFormat":1},{"version":"5339f84dfcb7b04aa1c2b4d7713d6128039381447f07abc2e48d36685e2eef44","impliedFormat":1},{"version":"fb35a61a39c933d31b5b2549d906b2c932a1486622958586f662dbd4b2fe72e6","impliedFormat":1},{"version":"24e2728268be1ad2407bab004549d2753a49b2acb0f117a04c4e28ffb3ecdd4f","impliedFormat":1},{"version":"aff159b14eba59afe98a88fe6f57881ba02895fb9763512dda9083497bdcd0e6","impliedFormat":1},{"version":"1f2bddea07543ccda708134cca0600b4d9ac9bd774ec1ede0a69935b04df1496","impliedFormat":1},{"version":"6e8997d08f6798d0a9416df24312cafd084e6184a205d9283eba95ef56f8ef8b","impliedFormat":1},{"version":"ac6968717607889d24d6e407effb48dd5af82005925b4725b1d9eb52a8a047e2","impliedFormat":1},{"version":"26080058b725ac0b480241751255b4391f722263778e84e66a62068705aafd3c","impliedFormat":1},{"version":"46afbf46c3d62eac2afead3a2011d506637bf4f2c05e1fd64bbf7e2bb2947b7c","impliedFormat":1},{"version":"84d02daa32c7a8bff4946bbc7d878ffb7114c19879f7bfceeeb39bef48e93c42","impliedFormat":1},{"version":"29723e0bc48036a127c3b8874f3abe9b695c56103f685f2b817fc532b8995e33","impliedFormat":1},{"version":"991cf4ed946cdf4c140ccaad45c61fc36a25b238a8fa95af51e93cb20c4b0503","impliedFormat":1},{"version":"81ef252ff5df76bccf7863bb355ccbb8af69f7d1064b3ef87b2b01c30fb2c1f4","impliedFormat":1},{"version":"0f17f5f14a5f53e5709404b5b59fe816eaad15a469412b73330e6f69834234e0","impliedFormat":1},{"version":"01edea77be9c2bef3a5f3fc46324c5e420e5bd72b499c5dec217c91866be5a99","impliedFormat":1},{"version":"39209d2b85d238810ef19ab3905c9498918343bc8f72a1dcae7fc0b08270d9a0","impliedFormat":1},{"version":"92a130d875262e78c581f98faa07c62f4510885df6d98213c72f3b83a1be93c1","impliedFormat":1},{"version":"6029af83ecb3d7815daf2dfb53b978087288d018e0681043b1c8c586e787af4b","impliedFormat":1},{"version":"0aa14ffe353b8bab88046e64a92efa5cd039f095759fe884d188702956e2cba2","impliedFormat":1},{"version":"68d3eee1d509f45625e39ba325a72c6ce1d2116e3d5c3a40f513472e66622e02","impliedFormat":1},{"version":"4e5f1234308de112f09920e0a0b99f35a9780b3abbc13a84445f32a490d0bb87","impliedFormat":1},{"version":"9ac0e5aea87c4a1d37b4677145e9a75bc8e13bf887bd1148a4acb21ab7398d00","impliedFormat":1},{"version":"625b802ecd18feb6a9d69ef8ef58d6c08c9c9022b8105cdeaa3fc77acaab5667","impliedFormat":1},{"version":"2ac33d7f6999e0fb363d1e483d80f087d3e7d712ff6fcc2b4f7b18b5dab92f37","impliedFormat":1},{"version":"195749d135be639001a554e4b4025b66b3c5c627d90b68266c14399bde120cec","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"e4ffb6aa67b82aca99315bd54365892ece9ed76ad37667a8dea478b7ac9a755a","impliedFormat":1},"819cd6f38755f4fc8d473cb9721c704775fef186e791b2d6058fdcca98cb5b11",{"version":"d34aa8df2d0b18fb56b1d772ff9b3c7aea7256cf0d692f969be6e1d27b74d660","impliedFormat":1},{"version":"f4db16820c99b6db923ab18af5fecb02331d785c4c2a8a88373a0cfc08256589","impliedFormat":1},{"version":"2f5747b1508ccf83fad0c251ba1e5da2f5a30b78b09ffa1cfaf633045160afed","impliedFormat":1},{"version":"90407bbaa24977b8a6a90861148ac98d8652afe69992a90d823f29e9807fe2d7","affectsGlobalScope":true,"impliedFormat":1},{"version":"b71c603a539078a5e3a039b20f2b0a0d1708967530cf97dec8850a9ca45baa2b","impliedFormat":1},{"version":"168d88e14e0d81fe170e0dadd38ae9d217476c11435ea640ddb9b7382bdb6c1f","impliedFormat":1},{"version":"104c67f0da1bdf0d94865419247e20eded83ce7f9911a1aa75fc675c077ca66e","impliedFormat":1},{"version":"cc0d0b339f31ce0ab3b7a5b714d8e578ce698f1e13d7f8c60bfb766baeb1d35c","impliedFormat":1},{"version":"6823ccc7b5b77bbf898d878dbcad18aa45e0fa96bdd0abd0de98d514845d9ed9","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e04cf0688e0d921111659c2b55851957017148fa7b977b02727477d155b3c47","impliedFormat":1},"c90e979e96b3086214f9bbf8efb103b9afc1cc36dade7927897072f5c8da129f","5d6c897d2b3dbce72049755a5d050ec40802cd648ed21d43c65be8698e4473a7","9fefdc8f36dda49e2d81481d5a239a66d9af727a3a761a7256a4b7d4bc136244","8ed1d2ebbdf5bfbc5b5480c2b8dbfd36c92cb678a2bbe12d082e3905981b3975","44cf69c4d1095e6ff3a2670085009bf3b597ca6b988711056defb7cbdefb4966","1ece02b356dfb8a53f279c0c8fb7bf15e5a016da2572a07c293f5e380782bd1d","6a716651e2ed41daaa310b35bba3b74e20e06b47122d239494af3e6a851b285f","fb200f76d128756dbc9db40339bd028755ebc8a1beacb2689229b123165fc949","c987f3abd1b0485fc5a2839a7fd70d776786175c789557db4a55f62266f602d7","65d73290c9f8274468dfa2673cfc3265ab4aa543d586ea5d93a349965f9dd08e","58340b7bfc63ea9b98e91e89e9a46d0ea6cc67dde0e7b43fe3ad287012c65849","c72d5ef4633b767f56b51b29efe49506299d3dda8e53a7a1ca123682f13750ec","0fdb7b65c8c17860faac5691a37e477ad73fd9fa519f1d1ead63c4644fb9a599",{"version":"6c1b497aeb9135ac66891d783a34dec6d4df347126ebe9c3841110f0a614e0c6","impliedFormat":1},{"version":"cef73ddf0336cb343be88b61a0448483029d438dd66ca21722aeabc66223bded","impliedFormat":1},{"version":"8cb6c8db9e27d0c6dba28bf0fcd7ef7603d0b5b2b3dce6fffc86f3827a8a00e9","impliedFormat":1},{"version":"d07ef5953b1499ae335c75147c658d9c037fc649544a4c85883f10eb9e5878e8","impliedFormat":1},{"version":"34714fae00aa0544916ade4018d18a04432db2b4b49c6cd066825ac31734eb40","impliedFormat":1},{"version":"5cb3b7b2b0997e451f91ab009ff2d66e7cd5f77838dc729a2e335554fa098a12","impliedFormat":1},{"version":"bdbe3e5d1f1f3dd035c551b6f94883212ccdbe9b3610f65f49138980e0efc0d7","impliedFormat":1},{"version":"eadae8542e5f360490f84d8da987529e415e265da584dd12e3e7c07a74db2fc9","impliedFormat":1},{"version":"9a82178f67affe7ca9c8b20035956d1ad5b25d25b42b6265820232ba16ba0768","impliedFormat":1},{"version":"332680a9475bd631519399f9796c59502aa499aa6f6771734eec82fa40c6d654","impliedFormat":1},{"version":"191bee6605de2b5210f29f22df04f5b5e6bdcc1f6e21fb07091d40eeeb75fd72","impliedFormat":1},{"version":"d83f3c0362467589b3a65d3a83088c068099c665a39061bf9b477f16708fa0f9","impliedFormat":1},{"version":"180e527dbc1f5ae2bbb79d0a3db1ada49258783d7e6299559e0f2ed663b4afec","impliedFormat":1},{"version":"29994a97447d10d003957bcc0c9355c272d8cf0f97143eb1ade331676e860945","impliedFormat":1},{"version":"f4260022f7af38e533d364ea62eb7ae01b0a32050033d7f6772073e1dc908025","impliedFormat":1},{"version":"9cddf06f2bc6753a8628670a737754b5c7e93e2cfe982a300a0b43cf98a7d032","impliedFormat":1},{"version":"3f8e68bd94e82fe4362553aa03030fcf94c381716ce3599d242535b0d9953e49","impliedFormat":1},{"version":"63e628515ec7017458620e1624c594c9bd76382f606890c8eebf2532bcab3b7c","impliedFormat":1},{"version":"355d5e2ba58012bc059e347a70aa8b72d18d82f0c3491e9660adaf852648f032","impliedFormat":1},{"version":"311cc121259b3e0c3c08304fc25b525aa02ba0f9bf55b3e7c60b0dbb7422014e","impliedFormat":1},{"version":"74c269b43d39e5ece20b2cca49c14e64c05b01e46407200d7558301d0fcaabf4","impliedFormat":1},{"version":"ec09bd95866efe38cd00ebb79dfa7a26563d600fa4a30db0f7c6d68f8f6d2b06","impliedFormat":1},{"version":"482d0ac70d56aa79941be30da6df28e926a007f835eed70cf7b5f3135368d1f6","impliedFormat":1},{"version":"7dd19397d5a090c9f8cd762bae67bd0ad6f782abe422594fb71168fb578673b0","impliedFormat":1},{"version":"84cbf6204ada0ee2f80493e55e45befa079954788718efd6dcc103183104e3c0","impliedFormat":1},{"version":"ed849d616865076f44a41c87f27698f7cdf230290c44bafc71d7c2bc6919b202","impliedFormat":1},{"version":"9a0a0af04065ddfecc29d2b090659fce57f46f64c7a04a9ba63835ef2b2d0efa","impliedFormat":1},{"version":"10297d22a9209a718b9883a384db19249b206a0897e95f2b9afeed3144601cb0","impliedFormat":1},{"version":"034b8b5912823744c986986f24432bf3fa7bfa671e69316b672f3f2db5166ce4","impliedFormat":1},{"version":"34d206f6ba993e601dade2791944bdf742ab0f7a8caccc661106c87438f4f904","impliedFormat":1},{"version":"05ca49cc7ba9111f6c816ecfadb9305fffeb579840961ee8286cc89749f06ebd","impliedFormat":1},{"version":"2151db9166dfd90feaa67f0c3a07efcab39e1640f1b26abc81632d8e1bf95fcb","impliedFormat":1},{"version":"2f6fda81a8ee3f78205dcf8a69e6f5afd8ee577b8f423fed0a72c346509937a0","impliedFormat":1},{"version":"37389b1222c65e82f8e2670d586d788911f317548c3ead5c5535d2495fd08572","impliedFormat":1},{"version":"1ce0883eaaeea383c10e3274b4e5189915beaf4ec3f74fe609afd44d16bba02e","impliedFormat":1},{"version":"55a525e18db580413a78fa364a54faf071c028888d5432fcf015d229c5fafc28","impliedFormat":1},{"version":"9190c744aa6c9c2d69b5e283f5559a5543a201c518b2fe14ea4e3eb7e42f33e0","impliedFormat":1},{"version":"057ac92d0a839b0a3bd91d86b9288991ff6ac9ea72b64877464f2b12b1d117a1","impliedFormat":1},{"version":"528f3448c98e09174ca4186540000bf77f81fddcb587e0db9626ea825dead0ab","impliedFormat":1},{"version":"305af5e4e2f1f3b99d43e285d8dde8a39073dd9e40a2c5682839e7f19d66688c","impliedFormat":1},{"version":"5cc755647e5205f5acd69ad6fbb5bb41c150ea5ce229ab8ee34fc769ffecb7ac","impliedFormat":1},{"version":"aac2532c6e0b1183dd1f7d746013af0dcde78ad82879ac599f17c5563bb2f1ec","impliedFormat":1},{"version":"9ca2b093001037c017169b747ec9616b26b42ac9d8fd60aa2b0d9eb6b8c1cf95","impliedFormat":1},{"version":"5e8f7bf17b0f8382eff0d93f1ffaafc4f7fd15293b4a90edd517dca2a3ff6020","impliedFormat":1},{"version":"e7705224440c17c183317a861a0f1392a9c0746c3a06549c09e2d511a8c0c32a","impliedFormat":1},{"version":"eaccaaec4001b7c7e672f371d466e96fcbe3782cdbb7eb8ecbd132060515573f","impliedFormat":1},{"version":"f3eca6b9a668c7872bb132fafe6750c582771c40a66606745c2c01dbec8d4c5d","impliedFormat":1},{"version":"8f4e60cefea06a80cdd3a6a19fd2070910aa819fa934b58e9dc4ac726f1ff74f","impliedFormat":1},{"version":"325574dd9f2eb185c39a1ee5bb5bf656cd448954ca5485e0987d29574abdc699","impliedFormat":1},{"version":"529e1aa76e7ed000983d12eec4eca0f6d38045e9d0bc8440083d86aa4897549a","impliedFormat":1},{"version":"452234c0b8169349b658a4b5e2b271608879b3914fcc325735ed21b9cb88d58d","impliedFormat":1},{"version":"3860b1088a3d0edbe82d07b7fb13a0d04b5f23653e70714892d58e847e37bb13","impliedFormat":1},{"version":"02e6216fe46b07bbfdc787254cf085fe383ad957fe7a26aab34cb4a6e0f969b6","impliedFormat":1},{"version":"1481128ac360e7a5fc5944efc36b7634b8e5eea8870d3e5cef6647af83f98c8c","impliedFormat":1},{"version":"b5b9340f337ae17e2b59afc4c70a45b698a0227a81daf16f4bdea22757d7ba74","impliedFormat":1},{"version":"3aec561fe42dc4beb19e50b9711580620d5b0988ca0295ad0f4060a5669ee3ba","impliedFormat":1},{"version":"801e735da27b1fcb22b4d79bbe1240f211889d633026cbbd1469f941245ab419","impliedFormat":1},{"version":"5265fd19af035a75b0ea228cdd98820babea56b2b79c75517c0158ad022ae16c","impliedFormat":1},{"version":"d9fdea96fc90cc8d970044bb7bbd75766899f06a6214383bbc3b95c061bdf733","impliedFormat":1},{"version":"b3952aed8c195a401b42a8995800b5c1ea4d9d390c1a5e3521a1a3c3653f9b71","impliedFormat":1},{"version":"69c63d594f437c04b4971e171b8b3eff3d926141b87c4a898cc139b39ac86666","impliedFormat":1},{"version":"3284e33a45d6aa8324691ac5737d08695e35e99b5f69fdc9ef21b3c7e7fd8449","impliedFormat":1},{"version":"46899ea33977cc9709846fa0df32edbaa610d261a7020e487c09c5b499723634","impliedFormat":1},{"version":"df2ba32dfae996beb1face17a5bba909d7fb8f6fb80ac554e7cae50e8b00a4c7","impliedFormat":1},{"version":"b4a8d900684e3167a5251e7614843bc889a307bd79226054124286743475f2fa","impliedFormat":1},{"version":"5feab6c5b5962b943354bafc10e79ab8a495786c1141358f2a44fe2108003828","impliedFormat":1},{"version":"91dc8945750895c8ee8cc8549f81b4f0a7a6248af72e48359f8cbbc5b8bec77a","impliedFormat":1},{"version":"67e1ae275bb047700f4384559e596bcdc46a9e7ba1ef6ab275e60b8059f077ce","impliedFormat":1},{"version":"eeb24fa259f000f6b51a1fe89123f55de081eb2a0ef8d8f847afd67af49cfb68","impliedFormat":1},{"version":"9d9b52c50efdfc1d23d4aea99074009a932068cb786992776c233913bff1eeb2","impliedFormat":1},{"version":"e21bb2cfbcdd8ce7eebb72422f3660806724f2b16cd6ce126d527511abb3a379","impliedFormat":1},{"version":"c04146836a55ea071b435298335e47f569db0e4d3ae420e35c83e448f944192f","impliedFormat":1},{"version":"31f71fe23daabea143fc8bd21dae0d5908227180fcda38ad3674df70351f9761","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"d023752daf2a5c2e27a2a850aedc10a48a42fb507dceae37db91dc8294aafdec","impliedFormat":1},"614c7caf88abe8857ed7de9c32b25e4b4d20c4a0925623ad5a859fe561cbba9f",{"version":"ba63131c5e91f797736444933af16ffa42f9f8c150d859ec65f568f037a416ea","impliedFormat":1},{"version":"44372b8b42e8916b0ab379da38dcf4de11227bad4221aba3e2dbe718999bdfab","impliedFormat":1},{"version":"43ebfcc5a9e9a9306ea4de9fda3abdd9e018040e246434b48ad56d93b14d4a3d","impliedFormat":1},{"version":"0e9aa853b5eb2ca09e0e3e3eb94cbd1d5fb3d682ab69817d4d11fe225953fc57","impliedFormat":1},{"version":"179683df1e78572988152d598f44297da79ac302545770710bba87563ce53e06","impliedFormat":1},{"version":"793c353144f16601da994fa4e62c09b7525836ce999c44f69c28929072ca206a","impliedFormat":1},{"version":"ff155930718467b27e379e4a195e4607ce277f805cad9d2fa5f4fd5dec224df6","affectsGlobalScope":true,"impliedFormat":1},{"version":"599ac4a84b7aa6a298731179ec1663a623ff8ac324cdc1dabb9c73c1259dc854","impliedFormat":1},{"version":"95c2ab3597d7d38e990bf212231a6def6f6af7e3d12b3bb1b67c15fc8bfd4f4a","impliedFormat":1},{"version":"585bc61f439c027640754dd26e480afa202f33e51db41ee283311a59c12c62e7","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},"aded36ed2c593630ed47de618ee775edbca587662b4abd93d4103602dd59d557","d6a602db688c1e2ee9724115c069c021b42dd374b1f54cb013a6c0c6b32a38b9",{"version":"2e19656c513ded3efe9d292e55d3661b47f21f48f9c7b22003b8522d6d78e42f","impliedFormat":1},{"version":"ddecf238214bfa352f7fb8ed748a7ec6c80f1edcb45053af466a4aa6a2b85ffe","impliedFormat":1},{"version":"896eec3b830d89bc3fb20a38589c111bbe4183dd422e61c6c985d6ccec46a1e9","impliedFormat":1},{"version":"907dab3492fc59404ecf40f9ad655251741c5f2e471bb0376d11dae3e27cb1d8","impliedFormat":1},{"version":"8629340be5692664c52a0e242705616c92b21330cb20acf23425fff401ac771f","impliedFormat":1},{"version":"81477bb2c9b97a9dd5ce7750ab4ae655e74172f0d536d637be345ba76b41cd92","impliedFormat":1},{"version":"55a6b0318ec658ff37bc88e18a93e5f10ddad7257b379b71abf39e6868b8d4d2","impliedFormat":1},{"version":"b7d85dc2de8db4ca983d848c8cfad6cf4d743f8cb35afe1957bedf997c858052","impliedFormat":1},{"version":"83daad5d7ae60a0aede88ea6b9e40853abcbe279c10187342b25e96e35bc9f78","impliedFormat":1},{"version":"3a4e276e678bae861d453944cf92178deaf9b6dcd363c8d10d5dd89d81b74a0c","impliedFormat":1},{"version":"db9661c9bca73e5be82c90359e6217540fd3fd674f0b9403edf04a619a57d563","impliedFormat":1},{"version":"f7a5ab7b54bdc6a13cf1015e1b5d6eeb31d765d54045281bfeefcdfcc982a37c","impliedFormat":1},{"version":"ec99a3d23510a4cb5bdc996b9f2170c78cde2bfa89a5aee4ca2c009a5f122310","impliedFormat":1},{"version":"37c2d356ebbcf0d3d33b9213fd1922b47eba5c15ec0cb15a411c5dc4d35499bf","impliedFormat":1},"f201e42bb80a9f51ac45716c52a2c2c43feb3a0c3a912e33f679c4c6eef66357","ba4abf64dfa4614cba305421c7b953a21164d47b4ac1f189f429352b424b4ddd","977c30493d8edb920e7eb11bba5a07d03ed179271eb4e0d4b6f9cfe6f8d6b9df","1ecb531d91a84fd9438126106273b6b91b59bc77ae6226ff7ee4c09b0ec81ea8","0229dcc6fdcbaa1defa323f48a5d4f45a8dade502c2622866c9a8e43d9eec0df","f9d8d355171301ce78de5938b28d1c29a239a2ac66ffc518975148eda3253dc5","e781b9cc8968208faa1707147b8a7afb1d04de2f40834bbe93999af6b20ccec6","1115c69b64d8a1c5c875a8bf24bfae0eb164d2769598a8881b2561521969682f","3f290394d4b80644e0c000c4a87bd406b16c2ae5754dd6e2ae03bf78e11687b1","524cda3453bebf1fa0e23d92d3970ba17967b677b60392459d303967314b9c06","62b574f6d62b52bb35070da8e6bd329a2d6117b4f4bd05d80807cfa3977fa62a","c01d96a8f819e33e2a13b875bf2f902636ec55faff8d00f44cd823fc9a7fac3a","5ac19d9ec0d126f9cbb4f7d856efc184c690eedb92cc95edf2d665b6d45bdd2b","d3b14aab633af81b4d937068affcc8b7bd834acc72c68d103d9a880793ea29dc","d61e42599d683d8cb50feaf54ae3d56f339bec3215f37757ce31df932c131247","bbe137bfc9e929453688e402aecb256bf4190776d47bc91f58757165fbf6c5e2","063ffaed779734b3db736e27c34865a2f7f8980f06ebbb3801b7b689fae69363","959cf3c1275ac4b009468075e8c2b9f9a0700218d7e8feaafbc8f9d4fd1b5e78","a8fa8c1d13672bd046a86546d665c7b754576c5b63df708cf121aeb045ec25fb","f36dae78c1fa663aaf6d6dd7774bfd97d31772a86288cc7478409f5081f6a1f1","007913db99604a18adb8b99e75a08d58129891ebd1ebb2e08781730798e5d40f","c325c11427f06848e895cf55d0748eb62aaa903efa5197512b644f02c98c9c3e","8a57b2ae85040ce35b1b020cee78596025573556d7e71413ae52167d7bccbac3",{"version":"25e5c8b73c6ad21f39e8e72f954090f30b431a993252bccea5bdad4a3d93c760","impliedFormat":1},{"version":"5bf595f68b7c1d46ae8385e3363c6e0d4695b6da58a84c6340489fc07ffc73f8","impliedFormat":1},{"version":"b87682ddc9e2c3714ca66991cdd86ff7e18cae6fd010742a93bd612a07d19697","impliedFormat":1},{"version":"87d3ab3f2edb68849714195c008bf9be6067b081ef5a199c9c32f743c6871522","impliedFormat":1},{"version":"86bf2bfe29d0bc3fbc68e64c25ea6eab9bcb3c518ae941012ed75b1e87d391ae","impliedFormat":1},{"version":"8d9c4957c4feed3de73c44eb472f5e44dfb0f0cb75db6ea00f38939bd77f6e84","impliedFormat":1},{"version":"00b4f8b82e78f658b7e269c95d07e55d391235ce34d432764687441177ae7f64","impliedFormat":1},{"version":"57880096566780d72e02a5b34d8577e78cdf072bfd624452a95d65bd8f07cbe0","impliedFormat":1},{"version":"10ac50eaf9eb62c048efe576592b14830a757f7ea7ed28ee8deafc19c9845297","impliedFormat":1},{"version":"e75af112e5487476f7c427945fbd76ca46b28285586ad349a25731d196222d56","impliedFormat":1},{"version":"e91adad3da69c366d57067fcf234030b8a05bcf98c25a759a7a5cd22398ac201","impliedFormat":1},{"version":"d7d6e1974124a2dad1a1b816ba2436a95f44feeda0573d6c9fb355f590cf9086","impliedFormat":1},{"version":"464413fcd7e7a3e1d3f2676dc5ef4ebe211c10e3107e126d4516d79439e4e808","impliedFormat":1},{"version":"18f912e4672327b3dd17d70e91da6fcd79d497ba01dde9053a23e7691f56908c","impliedFormat":1},{"version":"2974e2f06de97e1d6e61d1462b54d7da2c03b3e8458ee4b3dc36273bc6dda990","impliedFormat":1},{"version":"d8c1697db4bb3234ff3f8481545284992f1516bc712421b81ee3ef3f226ae112","impliedFormat":1},{"version":"59b6cce93747f7eb2c0405d9f32b77874e059d9881ec8f1b65ff6c068fcce6f2","impliedFormat":1},{"version":"e2c3c3ca3818d610599392a9431e60ec021c5d59262ecd616538484990f6e331","impliedFormat":1},{"version":"e3cd60be3c4f95c43420be67eaa21637585b7c1a8129f9b39983bbd294f9513c","impliedFormat":1},"7e53cf2988b02956c00db1a5d28f1087aabbfd0e437351d093cffecd36ac4a1b",{"version":"b40885a4e39fb67eb251fb009bf990f3571ccf7279dccad26c2261b4e5c8ebcd","impliedFormat":1},{"version":"2d0e63718a9ab15554cca1ef458a269ff938aea2ad379990a018a49e27aadf40","impliedFormat":1},{"version":"530e5c7e4f74267b7800f1702cf0c576282296a960acbdb2960389b2b1d0875b","impliedFormat":1},{"version":"1c483cc60a58a0d4c9a068bdaa8d95933263e6017fbea33c9f99790cf870f0a8","impliedFormat":1},{"version":"07863eea4f350458f803714350e43947f7f73d1d67a9ddf747017065d36b073a","impliedFormat":1},{"version":"396c2c14fa408707235d761a965bd84ce3d4fc3117c3b9f1404d6987d98a30d6","impliedFormat":1},{"version":"0c46e15efeb2ff6db7c6830c801204e1048ccf0c8cc9ab1556b0b95832c9d1c9","impliedFormat":1},{"version":"c475aa6e8f0a20c76b5684658e0adaf7e1ba275a088ee6a5641e1f7fe9130b8a","impliedFormat":1},{"version":"a42db31dacd0fa00d7b13608396ca4c9a5494ae794ad142e9fb4aa6597e5ca54","impliedFormat":1},{"version":"4d2b263907b8c03c5b2df90e6c1f166e9da85bd87bf439683f150afc91fce7e7","impliedFormat":1},{"version":"db6eec0bf471520d5de8037e42a77349c920061fb0eb82d7dc8917262cbf0f17","impliedFormat":1},{"version":"13c83c04f3cbd2da8276c6290b75f295edf309b4f907f667f1b775d5f048f47e","impliedFormat":1},{"version":"ca70001e8ea975754a3994379faca469a99f81d00e1ff5b95cabac5e993359aa","impliedFormat":1},{"version":"d6fa1d345cf72ead5f57dd2561136b7d09b774891d81822087499b41b3f04913","impliedFormat":1},{"version":"3bdc578841f58bfd1087e14f81394ece5efd56b953362ef100bdd5bd179cd625","impliedFormat":1},{"version":"2bc15addade46dc6480df2817c6761d84794c67819b81e9880ab5ce82afb1289","impliedFormat":1},{"version":"247d6e003639b4106281694e58aa359613b4a102b02906c277e650269eaecede","impliedFormat":1},{"version":"fe37c7dc4acc6be457da7c271485fcd531f619d1e0bfb7df6a47d00fca76f19c","impliedFormat":1},{"version":"159af954f2633a12fdee68605009e7e5b150dbeb6d70c46672fd41059c154d53","impliedFormat":1},{"version":"a1b36a1f91a54daf2e89e12b834fa41fb7338bc044d1f08a80817efc93c99ee5","impliedFormat":1},{"version":"8bb4a5b632dd5a868f3271750895cb61b0e20cff82032d87e89288faee8dd6e2","impliedFormat":1},{"version":"2a3e6dfb299953d5c8ba2aca69d61021bd6da24acea3d301c5fa1d6492fcb0ec","impliedFormat":1},{"version":"017de6fdabea79015d493bf71e56cbbff092525253c1d76003b3d58280cd82a0","impliedFormat":1},{"version":"cf94e5027dd533d4ee448b6076be91bc4186d70f9dc27fac3f3db58f1285d0be","impliedFormat":1},{"version":"74293f7ca4a5ddf3dab767560f1ac03f500d43352b62953964bf73ee8e235d3d","impliedFormat":1},{"version":"06921a4f3da17bed5d4bc6316658ce0ea7532658a5fc575a24aa07034c1b0d3d","impliedFormat":1},{"version":"90ee466f5028251945ee737787ee5e920ee447122792ad3c68243f15efa08414","impliedFormat":1},{"version":"34c17533b08bd962570d7bdb838fcaf5bcf7b913c903bc9241b0696a635b8115","impliedFormat":1},{"version":"1d567a058fe33c75604d2f973f5f10010131ab2b46cf5dddd2f7f5ee64928f07","impliedFormat":1},{"version":"5af5ebe8c9b84f667cd047cfcf1942d53e3b369dbd63fbea2a189bbf381146c6","impliedFormat":1},{"version":"5e126f7796301203e1d1048c1e5709ff9251f872a19f5ac0ee1f375d8128ef9b","impliedFormat":1},{"version":"147734cfd0973548fb6ef75d1e7d2c0b56bb59aad72b280784e811d914dc47d6","impliedFormat":1},{"version":"d2594d95d465026ebbee361f4819dc7b3146f4a8b42091ffb5dd90f9ceb345ab","impliedFormat":1},{"version":"e399d54c1b272a400ed446ca35d5e43d6b820723c2e5727b188ebea261e7cc2e","impliedFormat":1},{"version":"123568587c36c9f2a75091d8cdf8f287193855ba5aa10797b4fc320c80920b7f","impliedFormat":1},{"version":"6deffa531bdb8817b363505e88d957653d0c454f42c69e31588d00102cd1a076","impliedFormat":1},{"version":"973551068756351486afe706b240eb4dc83678ab2d829a1c6b1a19871394fd5f","impliedFormat":1},{"version":"e647d13de80e1b6b4e1d94363ea6f5f8f77dfb95d562748b488a7248af25aabf","impliedFormat":1},{"version":"9b7b0209a8841f5ffa60ccdfae26f7dc70ea4e7e446a603ef4732e84f1bb1b4f","impliedFormat":1},{"version":"5edc4b81a61ea5e0319b32d8f581d9643cb747cf44477b16af048f62d358c433","impliedFormat":1},{"version":"d47c9f84b00def208cbfdd820f8d10425ead9dbf36350d77fb55d5ef6857dabc","impliedFormat":1},{"version":"3bc5f767d5e0cd548c92e4623e0a7f4486889a72d2ca9cbc81df760669270dcc","impliedFormat":1},{"version":"20cf19c8028a7b958e9c2000281d0f4c4cd12502fef7d63b088d44647cdd607b","impliedFormat":1},{"version":"799780c3726407eaa2e09e709c376ec459582f6f9c41d9643f863580cecf7ff8","impliedFormat":1},{"version":"37280465f8f9b2ea21d490979952b18b7f4d1f0d8fab2d627618fb2cfa1828e3","impliedFormat":1},{"version":"52e29afa525973fc7cff28c4b6b359d91ad030d4aa198f060f813d4abcadb099","affectsGlobalScope":true,"impliedFormat":1},{"version":"a890cccdc380629c6cd9e9d92fff4ca69b9adddde84cc503296ada99429b5a3b","impliedFormat":1},{"version":"168b6da36cf7b832173d7832e017bc6c6c7b4023bf6b2de293efb991b96bca44","impliedFormat":1},{"version":"05b39d7219bb2f55f865bca39a3772e1c0a396ea562967929d6b666560c85617","impliedFormat":1},{"version":"bcae62618c23047e36d373f0feac5b13f09689e4cd08e788af13271dbe73a139","impliedFormat":1},{"version":"2c49c6d7da43f6d21e2ca035721c31b642ebf12a1e5e64cbf25f9e2d54723c36","impliedFormat":1},{"version":"5ae003688265a1547bbcb344bf0e26cb994149ac2c032756718e9039302dfac8","impliedFormat":1},{"version":"e1744dbace6ba2051a32da3c6b40e0fc690810a87b9ad4a1925b59f8f7157a34","impliedFormat":1},{"version":"ba8a615335e3dfdf0773558357f15edfff0461db9aa0aef99c6b60ebd7c40344","impliedFormat":1},{"version":"6921769648e4b83bb10e8fcf7011ea2d8f7de5d056daacf661648935a407376e","impliedFormat":1},{"version":"dd21167f276d648aa8a6d0aacd796e205d822406a51420b7d7f5aa18a6d9d6d9","impliedFormat":1},{"version":"3dea56c1745af2c31af0c84ecc6082044dc14cfa4d7366251e5bf91693eecd8b","impliedFormat":1},{"version":"eb6360635bc14b96a243bd5134e471f3ad26b0ecaf52d9d28621e443edb56e5c","impliedFormat":1},{"version":"3ad73b5b1d43cb8ba3975b2999396e10bf3ed015b3337876079caf07f22501d9","impliedFormat":1},{"version":"62a64260ea1dada7d643377c1a0ef3495363f4cca36adf7345e8566e7d7f419b","impliedFormat":1},{"version":"8b15e8af2fc862870418d0a082a9da2c2511b962844874cf3c2bad6b2763ca10","impliedFormat":1},{"version":"3d399835c3b3626e8e00fefc37868efe23dbb660cce8742486347ad29d334edd","impliedFormat":1},{"version":"b262699ba3cc0cae81dae0d9ff1262accf9832b2b7ee6548c626d74076bff8fe","impliedFormat":1},{"version":"057cac07c7bc5abdcfba44325fcea4906dff7919a3d7d82d4ec40f8b4c90cf2f","impliedFormat":1},{"version":"d94034601782f828aa556791279c86c37f09f7034a2ab873eefe136f77a6046b","impliedFormat":1},{"version":"fd25b101370ee175be080544387c4f29c137d4e23cad4de6c40c044bed6ecf99","impliedFormat":1},{"version":"8175f51ec284200f7bd403cb353d578e49a719e80416c18e9a12ebf2c4021b2b","impliedFormat":1},{"version":"e3acb4eb63b7fc659d7c2ac476140f7c85842a516b98d0e8698ba81650a1abd4","impliedFormat":1},{"version":"04d4c47854061cc5cefc3089f38e006375ae283c559ab2ce00763bca2e49516b","impliedFormat":1},{"version":"6a2146116c2fa9ca4fefa5c1d3de821462fc22e5330cda1196be15d439728c51","impliedFormat":1},{"version":"3b10140aae26eca9f0619c299921e202351c891b34e7245762e0641469864ffd","impliedFormat":1},{"version":"c0c0b22cefd1896b92d805556fcabda18720d24981b8cb74e08ffea1f73f96c2","impliedFormat":1},{"version":"ceec94a0cd2b3a121166b6bfe968a069f33974b48d9c3b45f6158e342396e6b2","impliedFormat":1},{"version":"49e35a90f8bd2aa4533286d7013d9c9ff4f1d9f2547188752c4a88c040e42885","impliedFormat":1},{"version":"3261b6d56270a3d8535f34c2fdad217cfba860d0f74f154f0a6a2031d0c8daf9","impliedFormat":1},{"version":"7eca5b6e1cd1c28637103d2b6c44e8b89035a53e515ff31ae3babc82e6c8e1f9","impliedFormat":1},{"version":"49c9c8316d59f6175e6e0439b1d5ef1218f02ce622d1a599449de30645559eed","impliedFormat":1},{"version":"e4c48be0ffac936fb60b19394739847145674582cbc7e24000d9fd35ab037365","impliedFormat":1},{"version":"215de2c70639abaf351b8ff69041e44a767ecffc5e8d2ac13ca3f201853fa1fb","impliedFormat":1},{"version":"d228c7773484140fac7286c9ca4f0e04db4a62acb792a606a2dda24bef70dc21","impliedFormat":1},{"version":"8e464886b1ff36711539ffa15ec2482472220271100768c1d98acfdf355a23ba","impliedFormat":1},{"version":"fb0135c4906ff44d3064feebd84bae323ebb7b59b8ce7053d34e7283d27c9076","impliedFormat":1},{"version":"178c8707a575baddc8f529a6dbd5d574a090e3498b2d525753db7938c74227c3","impliedFormat":1},{"version":"ae81e464a7db70637d07b93582b051487c7d119ac7e1bab1b1582a96e631b3f7","impliedFormat":1},{"version":"148634fcee440c7bd8c1339b97455aaadc196b0229ffc8dc8b85965a7d65b380","impliedFormat":1},{"version":"d3c60c4cf88594f84f7f5ca5f87d59090787bfcf032e86d4f03d58394b826910","impliedFormat":1},{"version":"f3c3f17825c6a78681186da04c2f3a0f1c60cfa95f3d4b82bbbd6ebd57214a6a","impliedFormat":1},{"version":"eb45a1782ef50423c1ffac4d2a89c60004f4e2d25ed8e7dcb9e24e6cf984ccdb","impliedFormat":1},{"version":"e4c94ea55cac34117f423db62a62df7121006b0e69ab77d27f98cef27973f876","impliedFormat":1},{"version":"97e4a153ac1feffdc4cbdf8f2b8ecfe8a2c651c3ba949fd5fbcc1e29d8e7d283","impliedFormat":1},{"version":"0b801e25d3e163d246c70e059cd529e5edd280692c6b384c20314825b23511bb","impliedFormat":1},{"version":"97f2a8c88656c95104fe635a2c254b0ec431591d26838e1fbc2a6db6b822a1df","impliedFormat":1},{"version":"d120b9af0cd364387d676de7c662643006ea2d942f7d5af18ca3024fee73c149","impliedFormat":1},{"version":"274097c2dff5959c9a6242b87aa858e772f0d269dd2847e44e4f6c5c0922937f","impliedFormat":1},{"version":"9a5ce34f28948d1b84dea73942e10b8a36fa10dc958356589c8f03aea866c96d","impliedFormat":1},{"version":"75897ed38868fdd8ce289475d2812e4d3e5cb64bcf812e44ebbf3f5e21ae486d","impliedFormat":1},{"version":"d994fb6705faaae18b9d71ba2d89b4a7e5e77c2b801a3dae51c0821da4a90acb","impliedFormat":1},{"version":"b30cc18b84468d3fa20ac04ca5ba9bed5a03431fc8a22bcf2c266c132baa1d3f","impliedFormat":1},{"version":"fda3e93361b17be4f24a12ac951b3c87cbae8e81b351d849caec342c94f932fa","impliedFormat":1},{"version":"f0ce99f229795d96ffbcee89e1de6c6d9dc52b4c31bfe6fe002893e243c305b4","impliedFormat":1},{"version":"680bd67092c080fefa2045bba990378e0c0f2ad5ed56b7fe928b4287413d7842","impliedFormat":1},{"version":"0840cd0220ad530dbc6e8549161e9304dac9d520c3b78caeeeb6af194a9f0588","impliedFormat":1},{"version":"a54f60678f44415d01a810ca27244e04b4dde3d9b6d9492874262f1a95e56c7d","impliedFormat":1},{"version":"84058607d19ac1fdef225a04832d7480478808c094cbaedbceda150fa87c7e25","impliedFormat":1},{"version":"27abd2f2ed5aaac951b12b8332aac7970c9cf0cfd88c458f0f016228180b4293","impliedFormat":1},{"version":"901c640dced9243875645e850705362cb0a9a7f2eea1a82bb95ed53d162f38dd","impliedFormat":1},{"version":"ebb0d92294fe20f62a07925ce590a93012d6323a6c77ddce92b7743fa1e9dd20","impliedFormat":1},{"version":"b499f398b4405b9f073b99ad853e47a6394ae6e1b7397c5d2f19c23a4081f213","impliedFormat":1},{"version":"ef2cbb05dee40c0167de4e459b9da523844707ab4b3b32e40090c649ad5616e9","impliedFormat":1},{"version":"068a22b89ecc0bed7182e79724a3d4d3d05daacfe3b6e6d3fd2fa3d063d94f44","impliedFormat":1},{"version":"e70d18d1352550a028f48d74e126a919c830267b38c76ddae4dc1571476a462a","impliedFormat":1},{"version":"5624b09ca38ea604954f0422a9354e79ada3100305362a0da79555b3dd86f578","impliedFormat":1},{"version":"24830e279f5773a4108e0cbde02bdcb6c20b1d347ff1509f63eed031bf8b3190","impliedFormat":1},{"version":"8899fd9f8ab5ce2b3af7ba0e1a47eede6a2a30a269283cc4a934ab755d0aadaa","impliedFormat":1},{"version":"f10759ece76e17645f840c7136b99cf9a2159b3eabf58e3eac9904cadc22eee5","impliedFormat":1},{"version":"363dd28f6a218239fbd45bbcc37202ad6a9a40b533b3e208e030137fa8037b03","impliedFormat":1},{"version":"c6986e90cf95cf639f7f55d8ca49c7aaf0d561d47e6d70ab6879e40f73518c8d","impliedFormat":1},{"version":"224d293a02b7d22edb77b4ab89c0d4f63b95ecd7c0698776719f33863a77ffdc","impliedFormat":1},{"version":"1518707348d7bd6154e30d49487ba92d47b6bd9a32d320cd8e602b59700b5317","impliedFormat":1},{"version":"ede55f9bac348427d5b32a45ad7a24cc6297354289076d50c68f1692add61bce","impliedFormat":1},{"version":"d53a7e00791305f0bd04ea6e4d7ea9850ccc3538877f070f55308b3222f0a793","impliedFormat":1},{"version":"4ea5b45c6693288bb66b2007041a950a9d2fe765e376738377ba445950e927f6","impliedFormat":1},{"version":"7f25e826bfabe77a159a5fec52af069c13378d0a09d2712c6373ff904ba55d4b","impliedFormat":1},{"version":"7ffef1ed1c2bc7d9cf2fc134a7e8c68b10416cdbe8e70da8a4bd7ad5c8698d9c","impliedFormat":1},{"version":"63c0926fcd1c3d6d9456f73ab17a6affcdfc41f7a0fa5971428a57e9ea5cf9e0","impliedFormat":1},{"version":"eb524eabfa1809d54dd289374c0ce0ed4f145abb878687e4fd5e67f91d7d08a6","impliedFormat":1},{"version":"4ef0a17c5bcae3d68227136b562a4d54a4db18cfa058354e52a9ac167d275bbb","impliedFormat":1},{"version":"b748dd4ccc072a2b7194b898dc8996a2cb56bfa15ccdb60ac0d2f9eaa8e28e9d","impliedFormat":1},{"version":"64269ed536e2647e12239481e8287509f9ee029cbb11169793796519cc37ecd4","impliedFormat":1},{"version":"c06fd8688dd064796b41170733bba3dcacfaf7e711045859364f4f778263fc7b","impliedFormat":1},{"version":"b0a8bf71fea54a788588c181c0bffbdd2c49904075a7c9cb8c98a3106ad6aa6d","impliedFormat":1},{"version":"434c5a40f2d5defeede46ae03fb07ed8b8c1d65e10412abd700291b24953c578","impliedFormat":1},{"version":"c5a6184688526f9cf53e3c9f216beb2123165bfa1ffcbfc7b1c3a925d031abf7","impliedFormat":1},{"version":"cd548f9fcd3cebe99b5ba91ae0ec61c3eae50bed9bc3cfd29d42dcfc201b68b5","affectsGlobalScope":true,"impliedFormat":1},{"version":"14a8ec10f9faf6e0baff58391578250a51e19d2e14abcc6fc239edb0fb4df7c5","impliedFormat":1},{"version":"81b0cf8cd66ae6736fd5496c5bbb9e19759713e29c9ed414b00350bd13d89d70","impliedFormat":1},{"version":"4992afbc8b2cb81e0053d989514a87d1e6c68cc7dedfe71f4b6e1ba35e29b77a","impliedFormat":1},{"version":"1810b0b14614e53075d4d1b3e6be512bde19b1ed3a287925c0d24bae8585fa1b","impliedFormat":1},{"version":"1c390420d6e444195fd814cb9dc2d9ca65e86eb2df9c1e14ff328098e1dc48ae","impliedFormat":1},{"version":"ec8b45e83323be47c740f3b573760a6f444964d19bbe20d34e3bca4b0304b3ad","impliedFormat":1},{"version":"ab8b86168ceb965a16e6fc39989b601c0857e1fd3fd63ff8289230163b114171","impliedFormat":1},{"version":"62d2f0134c9b53d00823c0731128d446defe4f2434fb84557f4697de70a62789","impliedFormat":1},{"version":"12056fec06740770f1d14d47dad75ba54a9a8e0f40e766b38172a44b42112886","impliedFormat":1},{"version":"ec8db3ff5638196d3debc149d61fb4d604e73e12ffd28ea02a326ad40d2240bf","impliedFormat":1},{"version":"e8c0a909df5d29c77c9b5037b9a25ec6be0b70e0d3f5ce95ae0923f2582d6c1c","impliedFormat":1},{"version":"9fad83dff123fb56c6d11f8973ebc6cffec784f5d5c0cc2572344df246bfa0dc","impliedFormat":1},{"version":"c7865dbee3c1955c9c91b107ac95e51c27ee67dc2b85893ee91d4ce46f37ea99","impliedFormat":1},{"version":"5776c61de0f11da1c3cf8aafc3df524e8445201c96a7c5065a36dc74c2dc0ef6","impliedFormat":1},{"version":"94ffa91cb9f8eba1f468c5439994d4544ad24658e1192060f76267b767114445","impliedFormat":1},{"version":"7f0f90d0ffdd54875c464b940afaa0f711396f65392f20e9ffafc0af12ccbf14","impliedFormat":1},{"version":"483255952a9b6240575a67f7beb4768bd850999a32d44d2c6d0ae6dfcdafe35c","impliedFormat":1},{"version":"a1957cc53ce2402d4dc5c51b7ccc76b30581ab67bea12a030a76300be67c51d8","impliedFormat":1},{"version":"8149e534c91fc2bcb3bf59f7c1fab7584382abfc5348055e7f84d2552c3de987","impliedFormat":1},{"version":"c280ec77789efcf60ea1f6fd7159774422f588104dae9dfa438c9c921f5ab168","impliedFormat":1},{"version":"2826b3526af4f0e2c8f303e7a9a9a6bb8632e4a96fece2c787f2df286a696cea","impliedFormat":1},{"version":"77ced89806322a43991a88a9bd267d6dc9e03fd207a65e879804fa760292a03b","impliedFormat":1},{"version":"c8ff3a75cd1c990cbe56080b1d254695c989136c9521cb1252c739788fe55c83","impliedFormat":1},{"version":"485f7d76af9e2b5af78aac874b0ac5563c2ae8c0a7833f62b24d837df8561fb9","impliedFormat":1},{"version":"8bdf41d41ff195838a5f9e92e5cb3dfcdc4665bcca9882b8d2f82a370a52384e","impliedFormat":1},{"version":"4f42e9c177a3c68913560c086a889285cd51c20b4d54a44bae77c17dee7cfa46","impliedFormat":1},{"version":"fe8a3e5492c807cc5cfc8dda4e6464aff0f991dc54db09be5d620fb4968ba101","impliedFormat":1},{"version":"31f931c21c7a121317c5b4d0223e6c13f616312ddb481409e54393340bd61ed9","impliedFormat":1},{"version":"654bcc87bc095d6a2248a5889ec057b38cae6052744b48f4d2922a7efac4554f","impliedFormat":1},{"version":"cad0f26943006174f5e7508c0542873c87ef77fa71d265968e5aa1239ad4459c","impliedFormat":1},{"version":"0be66c79867b62eabb489870ba9661c60c32a5b7295cce269e07e88e7bee5bf3","impliedFormat":1},{"version":"eed82e8db4b66b1ea1746a64cd8699a7779138b8e45d495306016ce918b28440","impliedFormat":1},{"version":"3a19286bcc9303c9352c03d68bb4b63cecbf5c9b7848465847bb6c9ceafa1484","impliedFormat":1},{"version":"6cdf8f9ca64918a2f3c2679bc146d55f07490f7f5e91310b642bc1a587f2e17e","impliedFormat":1},{"version":"3b55c93b5d7a44834d9d0060ca8bad7166cf83e13ef0ed0e736da4c3dbe490a2","impliedFormat":1},{"version":"d1f8a829c5e90734bb47a1d1941b8819aeee6e81a2a772c3c0f70b30e3693fa9","impliedFormat":1},{"version":"3517c54fba6f0623919137ab4bdb3b3c16e64b8578f025b0372b99be48227ad7","impliedFormat":1},{"version":"19b3d0c212d241c237f79009b4cd0051e54971747fd89dc70a74f874d1192534","impliedFormat":1},{"version":"0d0202235a9c54a36e08502ddee21ad24013eff328831630b438ca927ff15b11","impliedFormat":1},{"version":"f17963b9935dd2142c08b006da53afeeaca2c9a600485f6eb9c018b96687275b","impliedFormat":1},{"version":"1afd33ee26eb73a31cca8870ef7da768d1f13827957847d68d0270ce4568b4c2","impliedFormat":1},{"version":"2eba01b5ecb62f6e3204c7a45918760a5b9047071dc98e2ba86ff777f8d605bf","impliedFormat":1},{"version":"a24adc6e8aa778f0b66578f842a3c4fb7bdca5f8bb183c5e90c9746676a33454","impliedFormat":1},{"version":"12a6a37d9676938a3a443a6bd9e8321d7221b6ad67b4485753322dc82a91e2a1","impliedFormat":1},{"version":"6c4833182ba7a753200bf30986d254653c1ac58855d784edd8dfe82f5db98954","impliedFormat":1},{"version":"69eeee4818209fdb59544d6f74bd6ff024944bdd4050a33577f62376d5cada8e","impliedFormat":1},{"version":"4cfbd2a7a4afee212bfb0c9c3cb6e4c7d48366e0565bf5b43a4cd96c91cf14bf","impliedFormat":1},{"version":"37c175e28375e157933b40ca98eeb608e05f2583821a0fae564dc04614d2d95e","impliedFormat":1},{"version":"3f20a041a051abfb2b47a66611cf4bcbf263605f5469ed7e8b51b3977892d83f","impliedFormat":1},{"version":"7de33f94f482eee2f6d1d8f24427b737e2c4006792ec4c2b87da0a426e741c4d","impliedFormat":1},{"version":"79134a050ccec1692c31f1dacccd05ce4fcdacdf98f0fa56546b98eb8bdefead","impliedFormat":1},{"version":"24f1b6865be734484de2baf99146122137654c5f5f28086c5cee97b998bfcd5c","impliedFormat":1},{"version":"398feb1537ae0409646b0489bac99a9f0d757a2048f0009255f8e35e9c0f9828","impliedFormat":1},{"version":"3da4432a9c24123f98f6f1ddc5cda9c9eedf0a8853d06321803dbc5a116e5270","impliedFormat":1},{"version":"afc60e07200c5eae65b702f95d83096de54d99fa6eb2e0154e83b5e11c520bda","impliedFormat":1},{"version":"f4651affee2900f19746d1bf0fb1c45e77f57576197561ddc90b7272835c3f37","impliedFormat":1},{"version":"19527fc5a08c68414a234b02ae9b9619cdb4b811435d12c0af528e5640236f6b","impliedFormat":1},{"version":"20a629bc3f82d238f596230637365b8aec8284c963d13dafdd4c8e2746be5e64","impliedFormat":1},{"version":"01c48e5bf524d3fc2a3fa5c08a2e18d113ad1985bc3caea0503a4ea3a9eee64a","impliedFormat":1},{"version":"68969a0efd9030866f60c027aedbd600f66ea09e1c9290853cc24c2dcc92000f","impliedFormat":1},{"version":"4dbfad496657abd078dc75749cd7853cdc0d58f5be6dfb39f3e28be4fe7e7af5","impliedFormat":1},{"version":"348d2fe7d7b187f09ea6488ead5eae9bfbdb86742a2bad53b03dff593a7d40d1","impliedFormat":1},{"version":"becdfb07610e16293af2937e5f315a760f90a40fec4ffd76eb46ebcb0b3d6e16","impliedFormat":1},{"version":"710926665f4ada6c854b47da86b727005cc0e0831097d43f8c30727a7499788c","impliedFormat":1},{"version":"3888f0e43cd987a0dfa4fc16dd2096459deea150be49a2d30d6cf29d47801c92","impliedFormat":1},{"version":"f4300c38f9809cf811d5a9196893e91639a9e2bb6edf9a4f7e640c3c4ce765ec","impliedFormat":1},{"version":"676c3327721e3410b7387b13af857f4be96f2be91b3813a724eedc06b9ce52d7","impliedFormat":1},{"version":"10716e50bcd2a25cecf2dd993f0aadf76f12a390d2f7e91dc2cac794831e865e","impliedFormat":1},{"version":"81a8f1f6218d0acc8cd2cf8b5089d21b45cf812bb5820affe3bab058b46cba7b","impliedFormat":1},{"version":"fa69921924cf112fa523a18215a3bfb352ac3f498b46e66b879e50ca46cc9203","impliedFormat":1},{"version":"9b82a268ba0a85015cb04cd558582c7949a1b91b6761292b9360d093c18e1dd1","impliedFormat":1},{"version":"ccfb77fcac04c34442ffca82ae90c8dd2a0ec1689ace547fab9a0ae337dd4752","impliedFormat":1},{"version":"7b464488950d74ca5037da375308fc0c94a539378fd0e9554556df45483aad02","impliedFormat":1},{"version":"beebde754323e430b4ecf5b9f837a05b1667b3df86bd924b52c4f80f20b3d660","impliedFormat":1},{"version":"40eda068f71d159edc51c273a01948282d6e3d38dd2430944595d526dc4b40b9","impliedFormat":1},{"version":"c790db6044ce1bbafc46f13bde46b9f0065de155b26a199f442fe064f6b05d63","impliedFormat":1},{"version":"f405e934163ed30905b4682eb542bb2d446e59c477871be9d29f92ab474d522a","impliedFormat":1},{"version":"8294ddd1c6ea4ed9ec190a2d41500539c1623e274d5a67786d6b09849cb98d45","impliedFormat":1},{"version":"666d6d6d9f2298f8d8d17ac7a34ac9ca9a59e09fc97b1ae505df6ab4934e2dbe","impliedFormat":1},{"version":"f3941ac359b8377c0ccce596a2bd3cde8986279f42d75290b0272f3ab1aa604d","impliedFormat":1},{"version":"de3d39262355af808ff74b3df62aaad0ad3cbde76c13fb4fa6fb6e4cc817e78e","impliedFormat":1},{"version":"8c38034476af70d7ad430f69cb960c5bd6efc9962f266b39ed54dd8e9cad566c","impliedFormat":1},{"version":"757f7967151a9b1f043aba090f09c1bdb0abe54f229efd3b7a656eb6da616bf4","impliedFormat":1},{"version":"786691c952fe3feac79aca8f0e7e580d95c19afc8a4c6f8765e99fb756d8d9d7","impliedFormat":1},{"version":"734614c9c05d178ceb1acf2808e1ca7c092cf39d435efc47417d8f744f3e4c0b","impliedFormat":1},{"version":"d65a7ea85e27f032d99e183e664a92f5be67c7bc7b31940957af6beaaf696844","impliedFormat":1},{"version":"5c26ad04f6048b6433f87556619fd2e50ba6601dcdf3276c826c65681197f79d","impliedFormat":1},{"version":"9c752e91fe237ce4857fbbef141bee357821e1e50c2f33a72c6df845703c87d5","impliedFormat":1},{"version":"f926160895757a498af7715653e2aedb952c2579a7cb5cc79d7b13538f9090bd","impliedFormat":1},{"version":"255be579a134ab321af2fefb52ace369a11ffb4df09d1fbfc1ed1a43c1e5eec5","impliedFormat":1},{"version":"fa05a4a765755e92c1dcab306ef3648fa4aa108494b6e10d2329db8b89e89908","impliedFormat":1},{"version":"ea385ec05b32ad43bbd1002a7c553bbc6935754504d60dc38ee64cc8b3c21768","impliedFormat":1},{"version":"d61821435a95c7a660d5850ce6fe9c4400787595009853d982343b8089724319","impliedFormat":1},{"version":"3e56d3093cd84dac94761258d6d98226671481d43e93321f9825692940fcc0c2","impliedFormat":1},{"version":"25091d25f74760301f1e094456e2e6af52ceb6ef1ece48910463528e499992d8","impliedFormat":1},{"version":"ed79978235b685e7e9d2ac149c6ddaf602ce7e3a30725c20023e57f011760593","impliedFormat":1},{"version":"3345fc785abb65f2263f91ba092bb77470d949eddb41fc208256b964c2ccd5cb","impliedFormat":1},{"version":"d899e749dab44c9c0c1cf2118324ca1e6651113268551aa7a1e320b161953505","impliedFormat":1},{"version":"8a49075f007383f24df5b52376e41198e341a7b715da34a90b2c54b8fc8d4bcc","impliedFormat":1},{"version":"83f5494ed714e7898414a220e56886e00ffb58c80cbf745ecc45d6aae55f051a","impliedFormat":1},{"version":"28a019a0ebc44efd6b9148bbaf688bed1778d6977950da3d277f09e17687a46a","impliedFormat":1},{"version":"a67eff604aaeeae1c93f410dd9f176ab75a3fac2d2761e0bd9d83f933ec3e657","impliedFormat":1},{"version":"76a0210cd218dd17365ac0b7a99936804ede147c0a899d6bd2a9ac29cc8d4fea","impliedFormat":1},{"version":"853d02f4f46ca9700fefd0d45062f5b82c9335ba2224ca4d7bd34d6ae4fc4a7f","impliedFormat":1},{"version":"5f9ab7ba179f92fa3c5dddafec778a621fe9f64e2ba8c264ddf76fe5cf9eaf93","impliedFormat":1},{"version":"a0bce0fb40b88d17305f113ed02c4014329be52e8168b01fe825c049e9a37028","impliedFormat":1},{"version":"364e53fe15122e9d37aa8ee2c8eb037cde59bf5890b46a8205f4516b529501c0","impliedFormat":1},{"version":"1a577fdc45901cf461d4edc7697860c63a60526f60b7b2ba8ff7c89a9e7a1932","impliedFormat":1},{"version":"a6da29e6495bf303eb5f0b65dca3f92711b9cd6729eb1bed3e29dbc8b0fc2604","impliedFormat":1},{"version":"c35bd33a53356146889d87a05b34fc5a130ba93bc1bb36d021c0a7c817c4cc8d","impliedFormat":1},{"version":"f6067be38e8661d7e68287db5a602501897e2fbb385dfa2ecd170dd9f4524e8a","impliedFormat":1},{"version":"506b96750c75c5e5af820ac8356f27b2bcd0b077a02d33d90e87be9f0604f59d","impliedFormat":1},{"version":"d9c87237f57a1ca503eebb89514f65c48800b10b5574f7cda978addab1474dfb","impliedFormat":1},{"version":"b3cc1bb7311f35569b531e781d4a42d2b91f8dfd8bc194cc310c8b61011d6e43","impliedFormat":1},{"version":"33cff836b608822d3b4a52f523d964472c5437d08d81dbd1171ccf743276a223","impliedFormat":1},{"version":"8d334e97bce2fca74d7f2a416e1bd0c434785bffbca381632e8676dd4dee5324","impliedFormat":1},{"version":"4b610a25fe9e65ebf0b80d83dd52d4b40044768016821f947de60ba1b59cda4d","impliedFormat":1},{"version":"a9452e81c28c642c2f095844c3473d979eba5ae89726ad52b15ea86b3e112ee2","impliedFormat":1},{"version":"dc4a2cf12254395c8ae3fb4c61e6fd9f7c16110be66483599f9641941416988f","impliedFormat":1},{"version":"82b4045609dc0918319f835de4f6cb6a931fd729602292921c443a732a6bb811","impliedFormat":1},{"version":"8a2c67c55dfab4ea1f6e378cfa4b5cb60d8e8931316500f5b59c201674f6105c","impliedFormat":1},{"version":"b7b45ff1345f8e6bd6109a5b6ef0394c2e3bcbe48830516d9e78e20592ce468a","impliedFormat":1},{"version":"e5eb4863b7fc8515078dc09cd2f98fd179ff1a55216ecdc57d2dec7ce13e36c1","impliedFormat":1},{"version":"81785a3ea03d6db981ddfcf8fb1bd1377f985564def845c55e49e16f171deec4","impliedFormat":1},{"version":"537a2b61594512c5e75fad7e29d25c23922e27e5a1506eb4fce74fe858472a6e","impliedFormat":1},{"version":"8f9a2a6ddbd11ecbbc430ae8ce25528e696206f799ef1f22528569caf6ce580c","impliedFormat":1},{"version":"e05e03e1687d7f80f1569fdae117bb7b97feef1e839a61e1b3c61ffca8cc67c9","impliedFormat":1},{"version":"b311d973a0028d6bc19dfbaae891ad3f7c5057684eb105cfbeec992ab71fbc13","impliedFormat":1},{"version":"8a49e533b98d5c18a8d515cd3ae3bab9d02b6d4a9ac916e1dba9092ca0ebff15","impliedFormat":1},{"version":"fcb26ad5a6c39ce71dfac5dc16b3ed0e1a06a6dc8b9ac69112c935ad95fcad69","impliedFormat":1},{"version":"6acdef608420511aa0c9e3290b37d671bab4f719ffc2a2992c2e63a24605a657","impliedFormat":1},{"version":"291df5da0d84d1452cd68abfbcca08a3f96af610bf0e748528ba8d25784ce2b1","impliedFormat":1},{"version":"176cda558a7f76813f463a46af4607a81f10de5330c0f7a43d55982163aa0493","impliedFormat":1},{"version":"6621af294bd4af8f3f9dd9bd99bd83ed8d2facd16faa6690a5b02d305abd98ab","impliedFormat":1},{"version":"5eada4495ab95470990b51f467c78d47aecfccc42365df4b1e7e88a2952af1a3","impliedFormat":1},{"version":"6b08ada439e3c7fba3e6d18c19f934e7bbea3f34979f2490074f0623b849e8e4","impliedFormat":1},{"version":"40e9c2028b34c6c1e3281818d062f7008705254ee992d9857d051c603391e0f4","impliedFormat":1},{"version":"07a9aa7f3facdfac577ed4aa0c166295d54637508942a2154566d87804a33ae2","impliedFormat":1},{"version":"4a34de405e3017bf9e153850386aacdf6d26bbcd623073d13ab3c42c2ae7314c","impliedFormat":1},{"version":"993bcd7e2dd9479781f33daab41ec297b8d6e6ccc4c8f9b629a60cc41e07e5c8","impliedFormat":1},{"version":"714a7869be4ff21fa7be0dc183569db5e6818ca22882a79d2bb3a7801f5bfab4","impliedFormat":1},{"version":"dfa99386b9a1c1803eb20df3f6d3adc9e44effc84fa7c2ab6537ed1cb5cc8cfb","impliedFormat":1},{"version":"4cb85ba4cf75f1b950bd228949ae508f229296de60cf999593e4dd776f7e84e8","impliedFormat":1},{"version":"e39730c031200579280cae4ea331ec4e0aa42f8f7ad19c3ec4b0b90414e40113","impliedFormat":1},{"version":"e90bd7922cb6d591efd7330d0ba8247ec3edf4c511b81346fd49fff5184e6935","impliedFormat":1},{"version":"1b581d7fcfacd6bbdabb2ceae32af31e59bf7ef61a2c78de1a69ca879b104168","impliedFormat":1},{"version":"20f7f9e30ac8cbf38189b3adafbd945a755a049b082f27d89d1d5d52f46818fe","impliedFormat":1},{"version":"c749b03596746c41abf1e8ed6b5a6a1bcd316c00dc39a337cc152780efc593bb","impliedFormat":1},{"version":"846953ab15b2bf3a06da6ec485be611455c5c83a02102138dd01102f3e6307de","impliedFormat":1},{"version":"218ed8ccd7078df39a26ccc59a094919d7ed1c0cd0b0182233deffda851ac3c6","impliedFormat":1},{"version":"8422f4ff58293a827a8bf401bb36f7eefbf981ae9aac48643d19c1e5439ee1bc","impliedFormat":1},{"version":"f70ab2e7bd23db437c2d5ed8690c401a921afbd5d3998a6dd2aab90d9efbaf35","impliedFormat":1},{"version":"fdda29d1f7eb83a912e34ae73f4e6e612350a7d1a496d5facc2f75487e2a1601","impliedFormat":1},{"version":"8ec6b7dc9062dd5c3c2fcc54bbf24e1e8a32b29ed902abe9192ddd0fd5f5f2a7","impliedFormat":1},{"version":"52e7386606a26e912bd39cad7752cc33009aefbb062d4a45e557c29095987095","impliedFormat":1},{"version":"3a6ce66cd39bc030697a52508cfda7c248167467848964cc40bd992bd9ce71e0","impliedFormat":1},{"version":"b4ec75c8a71c180e886ffccb4b5391a5217d7e7077038de966e2b79553850412","impliedFormat":1},{"version":"58c7522c1b1c94667777664bb9b26be14159220a28abf43e42807815e3102e14","impliedFormat":1},{"version":"d1666062675fe2f5408bfc458dec90de7279820eea20890b19484250c324b8ea","impliedFormat":1},{"version":"aed88228359e87a1b1a4d3d45f5b6555724c01ac81ecd34aa56d4a0a01ba6910","impliedFormat":1},{"version":"25307c3fd3840b5bd50bf95240a134854e80f736a4666711ea8ea40ba706eaa9","impliedFormat":1},{"version":"4fce1ce36a7f6fa69d3954cd685d27995123b683d31819218d204ca6bdcbfc53","impliedFormat":1},{"version":"a26bd8cdefaaf5a4cfe2c2f9e5b9114072f6d274ed4422eb7dcd1f72705a7eb2","impliedFormat":1},{"version":"5bbcd14f0138f4e65971ed5cb5606e8591ffefe3ac78ac310b164a975ea38f4f","impliedFormat":1},{"version":"0220b23c1c15820dcbb94eb74b8671020b53cd192a708e4d1828f290149e7e89","impliedFormat":1},{"version":"4adc1491e1338de6745d009222786747f50d67ac34d901420fbaefbf1b51b58c","impliedFormat":1},{"version":"ab0926fedbd1f97ec02ed906cf4b1cf74093ab7458a835c3617dba60f1950ba3","impliedFormat":1},{"version":"f1a661906cd0e7fa5b049b15bdef4b20a99abca08faac457eeb2b6407f30d12f","impliedFormat":1},{"version":"7f5a6eac3d3d334e2f2eba41f659e9618c06361958762869055e22219f341554","impliedFormat":1},{"version":"626291e7b45a4b6871649c908fbbc5ac98009a5182e2594fbfe80b860f513c77","impliedFormat":1},{"version":"4093c47f69ea7acf0931095d5e01bfe1a0fa78586dbf13f4ae1142f190d82cc4","impliedFormat":1},{"version":"4fc9939c86a7d80ab6a361264e5666336d37e080a00d831d9358ad83575267da","impliedFormat":1},{"version":"f4ba385eedea4d7be1feeeac05aaa05d6741d931251a85ab48e0610271d001ce","impliedFormat":1},{"version":"348d5347f700d1e6000cbdd1198730979e65bfb7d6c12cc1adedf19f0c7f7fca","impliedFormat":1},{"version":"6fa6ceb04be38c932343d6435eb6a4054c3170829993934b013b110273fe40af","impliedFormat":1},{"version":"396e7b817fc4f5461b92f9a03325c2ebb09711aebcee5c41c5fd3e738eb78526","impliedFormat":1},{"version":"4116c4d61baab4676b52f2558f26fe9c9b5ca02c2792f9c36a577e7813029551","impliedFormat":1},{"version":"a294d0b1a9b16f85768553fdbf1d47f360dbff03649a84015c83fd3a582ba527","impliedFormat":1},{"version":"8f2644578a3273f43fd700803b89b842d2cd09c1fba2421db45737357e50f5b1","impliedFormat":1},{"version":"639f94fe145a72ce520d3d7b9b3b6c9049624d90cbf85cff46fb47fb28d1d8fe","impliedFormat":1},{"version":"8327a51d574987a2b0f61ea40df4adddf959f67bc48c303d4b33d47ba3be114a","impliedFormat":1},{"version":"00e1da5fce4ae9975f7b3ca994dcb188cf4c21aee48643e1d6d4b44e72df21ee","impliedFormat":1},{"version":"b991d92a0c3a48764edd073a5d28b6b4591ec9b7d4b2381067a57f36293637d0","impliedFormat":1},{"version":"51b4ab145645785c8ced29238192f870dbb98f1968a7c7ef2580cd40663b2940","impliedFormat":1},{"version":"100802c3378b835a3ce31f5d108de149bd152b45b555f22f50c2cafb3a962ead","impliedFormat":1},{"version":"fd4fef81d1930b60c464872e311f4f2da3586a2a398a1bdf346ffc7b8863150f","impliedFormat":1},{"version":"354f47aa8d895d523ebc47aea561b5fedb44590ac2f0eae94b56839a0f08056a","impliedFormat":1},{"version":"b152c7b474d7e084e78fa5eb610261a0bfe0810e4fd7290e848fdc88812f4504","impliedFormat":1},{"version":"67f2cd6e208e68fdfa366967d1949575df6ccf90c104fc9747b3f1bdb69ad55a","impliedFormat":1},{"version":"603395070ec53375882d53b585430e8f2dc6f77f4b381b22680d26c0a9595edc","impliedFormat":1},{"version":"cef16d87ff9aed3c5b96b47e0ac4277916c1c530f10eedfce4acaeacefddd3bb","impliedFormat":1},{"version":"fab33f402019d670257c8c833ffd78a7c9a99b4f7c23271e656cdbea1e89571f","impliedFormat":1},{"version":"976d20bb5533077a2135f456a2b48b7adb7149e78832b182066930bad94f053a","impliedFormat":1},{"version":"589713fefe7282fd008a2672c5fbacc4a94f31138bae6a03db2c7b5453dc8788","impliedFormat":1},{"version":"26f7f55345682291a8280c99bb672e386722961063c890c77120aaca462ac2f9","impliedFormat":1},{"version":"bdc2312da906d4129217238545d7e01e1d00b191beea1a9529b660de8b78834f","impliedFormat":1},{"version":"62b753ed351fba7e0f6b57103529ce90f2e11b949b8fc69c39464fe958535c25","impliedFormat":1},{"version":"514321f6616d04f0c879ac9f06374ed9cb8eac63e57147ac954e8c0e7440ce00","impliedFormat":1},{"version":"3c583256798adf31ef79fd5e51cd28a6fc764db87c105b0270214642cf1988aa","impliedFormat":1},{"version":"abdb70e24d3b39bf89aa07e769b33667c2d6f4ddcb4724735d72a941de6d4631","impliedFormat":1},{"version":"151aa7caace0a8e58772bff6e3505d06191508692d8638cd93e7ca5ecfa8cd1b","impliedFormat":1},{"version":"82f75b2de1456b0be46945c6c68547f032f11238d07db45bbc9c93fca6abfe41","impliedFormat":1},{"version":"812e55580eb591f3c04245345be8c9dce378b26238fb59d704e54a61e6e37c83","impliedFormat":1},{"version":"1de7ee494c7ac185e6abf94428afe270e98a59f1bb4768e4bea7804645a0d57d","impliedFormat":1},{"version":"2c12f912bab4b1eb797b2fded3f295fee98736f8053a08d0032becbecb4b34b1","impliedFormat":1},{"version":"2b234fce994b272403881b675d6ae2e2afb2a8be8bdec71002ff8ff2d5b59bd0","impliedFormat":1},{"version":"97ba9ccb439e5269a46562c6201063fbf6310922012fd58172304670958c21f6","impliedFormat":1},{"version":"50edac457bdc21b0c2f56e539b62b768f81b36c6199a87fbb63a89865b2348f0","impliedFormat":1},{"version":"d090654a3a57a76b5988f15b7bb7edc2cdc9c056a00985c7edd1c47a13881680","impliedFormat":1},{"version":"e0f69e399a1c3b367058918f3a6f9e5f29a0db26330c28208173556a8c48a0a3","impliedFormat":1},{"version":"6fe7b261e7242f5fb09166aa561110a9e356faf0452cc1828e18f17396dd7eb4","impliedFormat":1},{"version":"c18f1b2b05af7add7728cb54f0fa186e09bd33ae9b9c2a57e75bd45458db1b43","impliedFormat":1},{"version":"97776c2abc3ad9cdf00f858980a12e4b8033de0f6f7df62caa32e930726d665d","impliedFormat":1},{"version":"73bf1badc6969cf5eba2018b20dce1cc0dc00b056e66870abdca1f7b1275a648","impliedFormat":1},{"version":"8c8ee967649fed1f3bf84b3eea1928a2148e7320e5e04bdaa7b0f73946131156","impliedFormat":1},{"version":"638969ce0b4ed7122f2891bd934bb76718e1b3e10663651dbc230669939a297f","impliedFormat":1},{"version":"bb832613b3a3246183c65561fdeb87b6b35324bf46ce407dfa376a49ad259545","impliedFormat":1},{"version":"10291d06c7f3242f3e773262522d2bbaff113109b897b5ef0e7f2cc596408978","impliedFormat":1},{"version":"03239379d84009ee413ab759d85c8a5e254b1a97b175527e8cf946cee7d84621","impliedFormat":1},{"version":"57044825e3199de7211f1cc94a1d8080d5a1b3556f08b1263402014234983d06","impliedFormat":1},{"version":"6b567540de9240ef5b14e86239dd821be6e90274df164ef61a11921344019410","impliedFormat":1},{"version":"c2f426df0b5346a824013c507c611656884f4681e23782ae5bc396cd6ad6fe00","impliedFormat":1},{"version":"f737d4c256faf88b0b84255da24ac95de364db1c5564d67856fc2be2571e477d","impliedFormat":1},{"version":"082be20017cd0774d12bf0a928d0834bcc83454e706dfb3ebf2c314b8065cdc8","impliedFormat":1},{"version":"7e0b933925fa33fca25738c4daabc959f315f1ca3b2431a724069db01ab4e53f","impliedFormat":1},{"version":"34dce1ce616534c11ce825f1acf55cbe45bf7f38b48ddf451fe0ebdb8b04924d","impliedFormat":1},{"version":"561b27749052f76b32ac63a1da5a3f2f87e60fce44189dee4f9a728d4390d144","impliedFormat":1},{"version":"677b3a027ec7df033eb7df33eecfe77e066debd14d983a22916bcc67214b222c","impliedFormat":1},{"version":"547fbf3353abf40a827b1b48b832f9139e41cc600dac954cd4dd1c8faf1df517","impliedFormat":1},{"version":"dc0a609a8ab9995e439c887be9327c272400a3aadd76dae41f6c4eae9af723b8","impliedFormat":1},{"version":"843d9c252981c2cabd6fb28b0e2fc730df962ed2b78d208ca60745d434a2801e","impliedFormat":1},{"version":"c0c5ebf4cc5ef8ed64733c3dea8de54fbb45bb0a40306d26d840e50a954327cd","impliedFormat":1},{"version":"408c753ef793f71dffa590cfc3a95e5246ab67238f71083b7a96056b986e8964","impliedFormat":1},{"version":"6650fd077aa1900d99c2a52af9dd79971322e2991e70e463331d0cd60fe9f914","impliedFormat":1},{"version":"e074ffc31f9e3c284b4220abf9c4082e549842c6af98ae2d80b17071b21a573b","impliedFormat":1},{"version":"e32ceea1db1cf16192a7fd97fdc689dc625ad1c57555a2dc18e7b5c019216482","impliedFormat":1},{"version":"0359c9ffe72d124c521a64dcb4f37a6391e9234e0496a38f62ba9b18be1f1d89","impliedFormat":1},{"version":"5eda29b34e4f3ebed27585c7afb3a23f6017ec83feff42518cb7d4b9eebe7688","impliedFormat":1},{"version":"b1cdc896fa6216d1ff898423395f5ca6bf3b1118411b7f5ccc732c8acbd37e19","impliedFormat":1},{"version":"8d986fc98555e4b017177816f51b86236a357ff9c468f4c86e5b4096775f1639","impliedFormat":1},{"version":"37441514e58c1412999e02a2931b179a4e9330a5b67dd8f75b3dbee19e1b33db","impliedFormat":1},{"version":"a1c9f45912f4ad052795e0ef5743d8e1a8400efb0393a13c47c2b6320bdd9406","impliedFormat":1},{"version":"6402c4b62a62f59d78ad558c0abaa7a8059e966f083c99b89fd4b24302353413","impliedFormat":1},{"version":"a8378c2611bbac166238618fb9bc0af95b77ac59447d9046af1c4188ad758c1b","impliedFormat":1},{"version":"c745295efc9d176836701a358fcab80c77ca7a63f4c8e7bfaf0da3a0910607be","impliedFormat":1},{"version":"9074405ae056ee3066a3cdba3ad29eed41ce32c74838b6ca09e14ba8ed9e3123","impliedFormat":1},{"version":"d486cfb79e7cb30710b2f3cc8c02909fe71b34e26938ba536100ff207e38749f","impliedFormat":1},{"version":"1ec802755650072584f58dd6fc680d1519e790528a4658c17eb12c9a27f16053","impliedFormat":1},{"version":"d668cba43ed8e4364f2625bfc373f51ca7a0ef4333aadee0b953c6a8ce0ad953","impliedFormat":1},{"version":"38acd0b49cda3d9393086f8bd92754eeebaa4bf52b9f3dd0031d464f15d4bddf","impliedFormat":1},{"version":"b80ab976d1968b44e78d7c6312fa7895989e30604712001e504ec74c95d50c0f","impliedFormat":1},{"version":"7cdbc72e654c4cc3a95cf44a0f050a5affa822d626501731ea688deb0662b027","impliedFormat":1},{"version":"af12968b48a6d0d45a4737c6a02cdadae2ace6738b65a1c3425928677154db38","impliedFormat":1},{"version":"8d6283ed4e0f2164daeabb7f26be39785c8366f5deae0f39c1246486ca3df24c","impliedFormat":1},{"version":"ab9ea3afaee16b10a6d7d2fdde5c3c66a154e1f93266832a439ea4ae8f92ed3c","impliedFormat":1},{"version":"94ef22b94d7cafa97366308aba95caee00b8c740567c518dae3dcfd09c58a4d1","impliedFormat":1},{"version":"66332888aba679ae44ff8405d814a794a8328730d008ab5ca2c3b0629a1acc38","impliedFormat":1},{"version":"de9d02955efdcff476bd95184b75e321d962b8fc22f51b2033da443ffa33c029","impliedFormat":1},{"version":"8134618a47361a523fc5daea803db42a8f0751944e8a314128cdb4d51a4d0359","impliedFormat":1},{"version":"3cac1d701152a399417ca60ec47f9ca83fe96a4c519bfc4b9ceb79578a894bc3","impliedFormat":1},{"version":"d0a951a1c211da46f226452a1b758b046ff797e5c1a4d285ccec361749aa937b","impliedFormat":1},{"version":"dba5745334b583e1b8f7ef055939cc750c546c250a74b95e0ede1ad5b863400a","impliedFormat":1},{"version":"801f5c06b1bdb1f520fa9d12ad1cb1f0f5a60233ede2d6313a1894f4c48208af","impliedFormat":1},{"version":"afe7e63907fb014390c8b2fc1bea86232a17431faccfbd8248e9ecf848c698e4","impliedFormat":1},{"version":"6511247b2b1388ebd4e4d7131c3d1f89633b41fe5bf44bca8f90ac7fcaf1a6b2","impliedFormat":1},{"version":"894b86b56bc595744893ed21374abd71aebd7a346b711dbf4c1a39f714edcda2","impliedFormat":1},{"version":"c92258027ef4f99109f802bd82af908afa731ccc379373b5ee69cf420b40f060","impliedFormat":1},{"version":"ba19ddf03bdbd63a4a44ebb761439016507911f153b1b618b7aaca18801b1e1d","impliedFormat":1},{"version":"73f6fd0ee5da5198efd378a07383da0337b8e73a30a53420a46a4a098ea34f26","impliedFormat":1},{"version":"1b5962b26e6698523cbd71a11281f59f6c4f57099fdebb11cf68fe77abac8c1c","impliedFormat":1},{"version":"0f55b6a44c5b1571426a2b4d261c8eb69452079aeefbbd43eacfdc202cdf415f","impliedFormat":1},{"version":"afda2e399d17b76e199f56029d053308fa3d93a90b9565b06a81be01fb86865c","impliedFormat":1},{"version":"dc995c080b26c63ddf09c5a0e949fbbbd562a081fbb3004d25db42a82fec3b58","impliedFormat":1},{"version":"1ac440b352b34743ec4ee150e2e4c4d3cbe20795c803651e9a976fd0cd124bb8","impliedFormat":1},{"version":"6141b4f6daf9d8770f32c174fb185714e97851a345b5734256b2f615f6bfcbb5","impliedFormat":1},{"version":"cd454337b92454283ac6fa6a271d8688a9a875b32386a33c747fd0cb67407843","impliedFormat":1},{"version":"bff7736d65b7f6d60604986ad2a213f19c7828fbcea2fb8a559c0933f3b9ad07","impliedFormat":1},{"version":"2bb117bc6d5073d2562fdc273ffd5264f7b0a92e7b82b55001095873a202ea81","impliedFormat":1},{"version":"1749042b52c8e5fe166dd24318d18ab56db865cca56829a69ac42e6bb55b001f","impliedFormat":1},{"version":"5f8eaa67e3d8b40ab42257d96a714fafc809ab526142230a09f58635c192cd22","impliedFormat":1},{"version":"c1a706fc9524ed51280a131c9b7a4123f9617d18f9b84cc278eb010e695e477d","impliedFormat":1},{"version":"e76a5c0be446202df4edfbc86834735c54010f5241709daac7096683fc18658e","impliedFormat":1},{"version":"a1a60e9c4aa722efc1c0402f7819c4fac46514a2004dabebacbfce89c524c5c3","impliedFormat":1},{"version":"71b7ae47fcf519a301fd2cd1067e3934380054e8fb3adf3d8d53ab74e080cd7a","impliedFormat":1},{"version":"59291388675b917b92ac856ccb5059201cbaa757a358c4a3c50b2dd63a4bee79","impliedFormat":1},{"version":"7ebf776831a1bcf30321f35361a30984036808cd74cf5ff8af65d9bad4c6ea0b","impliedFormat":1},{"version":"9810d5e27b657a2e13ba091c899460c92aeeb2230bd83bd68a57936d8004cb0a","impliedFormat":1},{"version":"77fd2f9105721dee1b0680be07e3dbd7080719f3b7d4fa34ee2f84b5fe301bd1","impliedFormat":1},{"version":"f43678d8b19945df13814b6838120b7c7b8dfdce7569296b1b1976eaf085c705","impliedFormat":1},{"version":"07c21b91f2117a8bd633177327a3b86856b36c387d2f2e045073072fb8a37714","impliedFormat":1},{"version":"b5cf7048adaa2cae79df7f1a0774e2894714c254f56feb89b8a4e07705191c87","impliedFormat":1},{"version":"13baee3bc1768b9923c49f08193495c4e6781262a64d59a2e63c141b771a9fcb","impliedFormat":1},{"version":"fdc91fa6795007d22e50ba7ce3dc88c2fe6499899ab3e89f09aa5cff98e0bbb6","impliedFormat":1},{"version":"0638bd7f74c6483ef226245204f32be39d0fc8e7c4c6c8a5b1577f8cacb70410","impliedFormat":1},{"version":"bef9d33e702185e457db9520d32df7e7e72bd33ed7b9596b211d9c8819e5402b","impliedFormat":1},{"version":"7ff5b052778be5d4b85df000ee9efe6c75e6e4f87861f2a1313987f5476d9ef5","impliedFormat":1},{"version":"b6914c5199969e2aa490bd347dfc5e289813b260dba1331d06c55b3cbcab9b6f","impliedFormat":1},{"version":"f65070ba38c06935f1ef31a318e5e9812363ec7791d556055cd025a882ee8e55","impliedFormat":1},{"version":"93e07e1a200d32977d8e2fce3de3db49fb9fb482b392611ca0cd3d66d0397444","impliedFormat":1},{"version":"21f2230f4af0aebfcfbadd985578bc55e5b8f43fd44bed0bf218823376033394","impliedFormat":1},{"version":"ac214369128f0b6a614bfa33ab522967abe8e970d2dfc7d523f5a3882d72c863","impliedFormat":1},{"version":"3caef1cb673fde4b700242696f2ae18aa9fe35bb93f2c3e6b5e6faf9fb34190a","impliedFormat":1},{"version":"16ae8a29108474734e2f5ec9806bf7a1d596e03265399bacb6cb0c6a122b7c9e","impliedFormat":1},{"version":"c20d73302303381a3b3f510e79576ca38d073294eb9ffcb3858d678b10e511aa","impliedFormat":1},{"version":"c94d37997cae72d328e2492bd572151ab72398daf771a3651672bbde53cede17","impliedFormat":1},{"version":"0a6c7438c05d0c1243030cb34884c927f3e393590242eef0b60f216df11248b9","impliedFormat":1},{"version":"2a198644dda57c757234a8d26b6e466f1fb610515a584478459a2d39c14900aa","impliedFormat":1},{"version":"2491ea15b97d8f9e6602d0dec94e57943be6150adaf6fccf8b77a5830c300743","impliedFormat":1},{"version":"2723a90d915173b65458fe6bad29131622704c0f1ab37b8913342461f61bd496","impliedFormat":1},{"version":"ebcfb573a78eedc5a034d975ef663e77eb2a81ec14bedcb55fc519d6699090f3","impliedFormat":1},{"version":"1fc17ce97c5ac21cb923f0f8eca6e3d6d8a514b6d66da902587cbb9d799f5c87","impliedFormat":1},{"version":"be93596d4f187f67d7845f895d10073ccbac95214f7ebf8671a438249b6e9787","impliedFormat":1},{"version":"13860f2e3e36443d23ad7241926a974077098f400da4cca6db3acf31d24dac81","impliedFormat":1},{"version":"fd0efac55de17fd65bbaa67b9586b6ff148cdda9b7161c77a0ac7b20b79c5645","impliedFormat":1},{"version":"9d74f18aad3b7c8a0336eacd427c659bd8bf57444426f6f0e41f8fca1ceee97a","impliedFormat":1},{"version":"f3c4bf50cd6721e827571cb942ebde7d12764b6f742a58b6c10e579127927ab1","impliedFormat":1},{"version":"297511a767d55924440a6014baf09e42509ce46c63c881380947b642d3d46a2d","impliedFormat":1},{"version":"33073bdc3cfd190a9b1c15a0abdcf611eeb924de4faa190e42227f631ca0ed70","impliedFormat":1},{"version":"83a6c79b5bc0c8b167ff323e2289c716346e4cb86008d2dd25b97c6454a47e35","impliedFormat":1},{"version":"4dd7bc31a2e62befc4dd7411c393bb6890d9fe38cc72fb813644936fca82aa0d","impliedFormat":1},{"version":"adc2bfa76a57b734c8a30695b2538c60e9352d4e354a68d51829ea43b768fed9","impliedFormat":1},{"version":"8c15f3c23f926d273dd648d74539d50e8e23400539116b77ea72d3d17fc8f294","impliedFormat":1},{"version":"980f3c0431cc8689000e35012e23a248dd781e3e5282057126624a7137a3bf42","impliedFormat":1},{"version":"9b8202cbccd1249e8640ea844c0dabbb99a604a6ba52feee963e557680ed285c","impliedFormat":1},{"version":"983f0fa9ecfa26398247e83ab68f60aa5cfe967905736047cad0864762d9eeea","impliedFormat":1},{"version":"473f53747832bc2588d9e9e0347d3fbcc8aa8e61124b4b4ed54185f930e4f80e","impliedFormat":1},{"version":"bf96e903108160a97d684bb1d0991faad9a0c9a209759a7338ea22fbd4510f75","impliedFormat":1},{"version":"ea99aa2e537966df22f8192e99929ee81719c1cf0b9d9d83d0c6fed53325ccc6","impliedFormat":1},{"version":"d8feeb2d114397bfaa224f022701560ede9ed33299d2ab26a5979b5be2bf6b49","impliedFormat":1},{"version":"ee0ff5abedfc526345058f7d0bdb94f038db1d2b85e527a72f8b1d50fd8585d1","impliedFormat":1},{"version":"61029abbe36480fb1b2ab39f2ae333b909e9e25ccf3e192b50909cf26140bcbe","impliedFormat":1},{"version":"d0060157c0e676a2da04b924645eba5892a702e9640c27b9d0aa84e8aa036421","impliedFormat":1},{"version":"2542bb610f66daadbbe7abd6cd6b61413e1b763aac9a599f765a0ec3bde1c370","impliedFormat":1},{"version":"7d0d9e7638a6800b06045dcaa53bae64dea4ec3027736ee26f7c1cff50d671bd","impliedFormat":1},{"version":"c8df12e4975f0e4bfe405a9a9319cd2fda5bd83ed3907c214b37ba77778ef2d2","impliedFormat":1},{"version":"1b601a6492f374b24723cc645a6731ca97209025edd688aed45fe6e6217c100a","impliedFormat":1},{"version":"74a907fa14655328575b29e4dbdf58440dd07c081d9d245f785c4143d10510c8","impliedFormat":1},{"version":"197da644ac572eae35c0e3cf933d24b120bdf09db3ea7e98965cc09eb1e9fb64","impliedFormat":1},{"version":"76d7adf42ea58b58559c7b85a37d84a3ea163c215f12c812df49b0ddec849ea6","impliedFormat":1},{"version":"ae375d815b60f98942bd1af5f865d4e9aa7977ac10e7fed8464b0dc42291b718","impliedFormat":1},{"version":"712b3998796d62fffe46f5cbb47453a141549c0cde81e04da8464243ec3e60e2","impliedFormat":1},{"version":"2d9f4d58554a246616eeaa090a2fb0dddccf412e88617975138389fb15770ca9","impliedFormat":1},{"version":"500561aaec426c1d47bcb10a78c275b7b59cb489c19eb72e1daf4d7d241e4f4a","impliedFormat":1},{"version":"c1bcf1e07567521e99973bdb3a2a7c6508c013356af71bdca9cc1be05560b0d6","impliedFormat":1},{"version":"1ad980595342aa0d8a6711e5c10c18107fcfd11d41cc72d8c1c2160ad11df9d3","impliedFormat":1},"48ce30d04853735210d962e8f0a3639d6a0326779749e8fdbe99e4aa25c5b6f4","cc7c6ef233ce7c1668df9b8b0243df3cee2804a9adb12e8e4f7c6fd8a281c4f4","8e5edeb002097c3a52a07967d7063ee812eeab79f7efbe8011086d0bdb4ec10c",{"version":"1d879125d1ec570bf04bc1f362fdbe0cb538315c7ac4bcfcdf0c1e9670846aa6","impliedFormat":1},"43b6f91b7ef873bedff5e224204d9512cb8c91742527e78d7afcc951aa44e1eb","dd33a13f3884d1cf0d1c3e57a595d0bd85a6a9217e37d75f6d23385c07238a9d","2594a4fb89fbdd01ea83c5cff26367b2ef5aa241e1a59c2871799f00f5c3069e","a2890c3a0f79089d03c2f5659e02802effbe89ddad41aef65ed67098ad34861c","18a28c1bded9978424c5218e10a01d29d2f4a701fb3ff78dcfa47f3425ef4484",{"version":"6ada175c0c585e89569e8feb8ff6fc9fc443d7f9ca6340b456e0f94cbef559bf","impliedFormat":99},{"version":"e56e4d95fad615c97eb0ae39c329a4cda9c0af178273a9173676cc9b14b58520","impliedFormat":99},{"version":"bd821b87e2c0fb5f509cedf47da465c447451835ce0fe2a752c4fc53a9f95a5b","impliedFormat":99},{"version":"f1d7352c0f7041abb43e1054abb14fb8c53a13dd54bcc1d67b97d2c02bb5028c","impliedFormat":99},{"version":"fc820b2f0c21501f51f79b58a21d3fa7ae5659fc1812784dbfbb72af147659ee","impliedFormat":99},{"version":"08f88f75fc2f516413477606a4122b6d3f6eb6680e8eb79f3fda5a5d2ed306df","impliedFormat":99},{"version":"6ab9821afd2a06879620eb4e041b9492a90f294e9b733ae5eb022edaa3964a45","impliedFormat":99},{"version":"003533cc3fa10cc457668d4256d21a65706a67a04251962cfe85d240502f8d67","impliedFormat":99},{"version":"6c00cb8a4b187505dfe21aff242b07f69f84f5c832e8ab4357af69daaee1b0df","impliedFormat":99},{"version":"de14ddf9d780367c6a117bd8a1718d491aff66094186523b3eea680ea7035a7c","impliedFormat":99},{"version":"ee06b94d0521cfaf91e4b003518eeefc45bbd594b0c22955fe35be282958252b","impliedFormat":99},{"version":"9e5f8fdaeb03f1699392b4724a58ca7b47c5cbb6762920d2bfc722c265495ede","impliedFormat":99},{"version":"d05bd4d28c12545827349b0ac3a79c50658d68147dad38d13e97e22353544496","impliedFormat":99},{"version":"a1597b0039f39e9f3eeaf120f02d0c94a826fad30b027a2abfdb8d580c89be70","impliedFormat":99},{"version":"04ace6bedd6f59c30ea6df1f0f8d432c728c8bc5c5fd0c5c1c80242d3ab51977","impliedFormat":99},{"version":"57a8a7772769c35ba7b4b1ba125f0812deec5c7102a0d04d9e15b1d22880c9e8","impliedFormat":99},{"version":"1de82ba3718b2b3bc5333c5bc35da5cfc46d1b654edc012de46bbce48126fcce","impliedFormat":99},"0468b21cc2c1fdc056390c61c0a81cdf2ce3b26842c0fe0392e4487873c53746","944ef944a30ca3879c1f9c4e50c69eec0b9466b75404e2013203d4fa060eb87c","a8316acb6339b488881d0e782bc638e535113cd9a7071dfd32e3fceba4a89ce1","26f7177e6174e43b0db8fdf8b6082155da16dbf7fabf9d895780c3ea69de87da",{"version":"64089edc83778914d8a71dc3033792254d0c6b290d724070cef8aa90f1abf4d2","impliedFormat":1},"807006956c54425431d0b97aa08fddd15d81f766d1d4d8a8f48751ba0c7797ec","695d1a368ea1988ded27b32a10e912427c3d6d98fffcd9b2ca587180d53cc807","d856394df3bf501622abd825a1da7e7f8104fb3c3514f4f0113663aa038d30ca","b4aa3af82053f9ad495041f781e0595645c5b8684d4da2cba902ba84eebda6c9","d193d47d9c2d83b7f6e8d37e8ca73b69ead56ebee2dc5cd7564af6ca02963922","b610d99af393536484cc7da57316288ed2c2c448555e108a3e4fc24483b58a28","2f4977ee001f27906934338ae3a8a9eb0b3ee2a66d400102ce6a0d336defcdfe","82c24effe7255fd940987b21ddb42737a73172983722b657a7a3ba2826962e4f",{"version":"cb5eaaa2a079305b1c5344af739b29c479746f7a7aefffc7175d23d8b7c8dbb0","impliedFormat":1},{"version":"bd324dccada40f2c94aaa1ebc82b11ce3927b7a2fe74a5ab92b431d495a86e6f","impliedFormat":1},{"version":"56749bf8b557c4c76181b2fd87e41bde2b67843303ae2eabb299623897d704d6","impliedFormat":1},{"version":"5a6fbec8c8e62c37e9685a91a6ef0f6ecaddb1ee90f7b2c2b71b454b40a0d9a6","impliedFormat":1},{"version":"e7435f2f56c50688250f3b6ef99d8f3a1443f4e3d65b4526dfb31dfd4ba532f8","impliedFormat":1},{"version":"6fc56a681a637069675b2e11b4aa105efe146f7a88876f23537e9ea139297cf9","impliedFormat":1},{"version":"33b7f4106cf45ae7ccbb95acd551e9a5cd3c27f598d48216bda84213b8ae0c7e","impliedFormat":1},{"version":"176d6f604b228f727afb8e96fd6ff78c7ca38102e07acfb86a0034d8f8a2064a","impliedFormat":1},{"version":"1b1a02c54361b8c222392054648a2137fc5983ad5680134a653b1d9f655fe43d","impliedFormat":1},{"version":"8bcb884d06860a129dbffa3500d51116d9d1040bb3bf1c9762eb2f1e7fd5c85c","impliedFormat":1},{"version":"e55c0f31407e1e4eee10994001a4f570e1817897a707655f0bbe4d4a66920e9e","impliedFormat":1},{"version":"a37c2194c586faa8979f50a5c5ca165b0903d31ee62a9fe65e4494aa099712c0","impliedFormat":1},{"version":"6602339ddc9cd7e54261bda0e70fb356d9cdc10e3ec7feb5fa28982f8a4d9e34","impliedFormat":1},{"version":"7ffaa736b8a04b0b8af66092da536f71ef13a5ef0428c7711f32b94b68f7c8c8","impliedFormat":1},{"version":"7b4930d666bbe5d10a19fcc8f60cfa392d3ad3383b7f61e979881d2c251bc895","impliedFormat":1},{"version":"46342f04405a2be3fbfb5e38fe3411325769f14482b8cd48077f2d14b64abcfb","impliedFormat":1},{"version":"8fa675c4f44e6020328cf85fdf25419300f35d591b4f56f56e00f9d52b6fbb3b","impliedFormat":1},{"version":"ba98f23160cfa6b47ee8072b8f54201f21a1ee9addc2ef461ebadf559fe5c43a","impliedFormat":1},{"version":"45a4591b53459e21217dc9803367a651e5a1c30358a015f27de0b3e719db816b","impliedFormat":1},{"version":"9ef22bee37885193b9fae7f4cad9502542c12c7fe16afe61e826cdd822643d84","impliedFormat":1},{"version":"b0451895b894c102eed19d50bd5fcb3afd116097f77a7d83625624fafcca8939","impliedFormat":1},{"version":"bce17120b679ff4f1be70f5fe5c56044e07ed45f1e555db6486c6ded8e1da1c8","impliedFormat":1},{"version":"7590477bfa2e309e677ff7f31cb466f377fcd0e10a72950439c3203175309958","impliedFormat":1},{"version":"3f9ebd554335d2c4c4e7dc67af342d37dc8f2938afa64605d8a93236022cc8a5","impliedFormat":1},{"version":"1c077c9f6c0bc02a36207994a6e92a8fbf72d017c4567f640b52bf32984d2392","impliedFormat":1},{"version":"600b42323925b32902b17563654405968aa12ee39e665f83987b7759224cc317","impliedFormat":1},{"version":"32c8f85f6b4e145537dfe61b94ddd98b47dbdd1d37dc4b7042a8d969cd63a1aa","impliedFormat":1},{"version":"2426ed0e9982c3d734a6896b697adf5ae93d634b73eb15b48da8106634f6d911","impliedFormat":1},{"version":"057431f69d565fb44c246f9f64eac09cf309a9af7afb97e588ebef19cc33c779","impliedFormat":1},{"version":"960d026ca8bf27a8f7a3920ee50438b50ec913d635aa92542ca07558f9c59eca","impliedFormat":1},{"version":"71f5d895cc1a8a935c40c070d3d0fade53ae7e303fd76f443b8b541dee19a90c","impliedFormat":1},{"version":"252eb4750d0439d1674ad0dc30d2a2a3e4655e08ad9e58a7e236b21e78d1d540","impliedFormat":1},{"version":"e344b4a389bb2dfa98f144f3f195387a02b6bdb69deed4a96d16cc283c567778","impliedFormat":1},{"version":"c6cdcd12d577032b84eed1de4d2de2ae343463701a25961b202cff93989439fb","impliedFormat":1},{"version":"3dc633586d48fcd04a4f8acdbf7631b8e4a334632f252d5707e04b299069721e","impliedFormat":1},{"version":"3322858f01c0349ee7968a5ce93a1ca0c154c4692aa8f1721dc5192a9191a168","impliedFormat":1},{"version":"6dde0a77adad4173a49e6de4edd6ef70f5598cbebb5c80d76c111943854636ca","impliedFormat":1},{"version":"09acacae732e3cc67a6415026cfae979ebe900905500147a629837b790a366b3","impliedFormat":1},{"version":"f7b622759e094a3c2e19640e0cb233b21810d2762b3e894ef7f415334125eb22","impliedFormat":1},{"version":"99236ea5c4c583082975823fd19bcce6a44963c5c894e20384bc72e7eccf9b03","impliedFormat":1},{"version":"f6688a02946a3f7490aa9e26d76d1c97a388e42e77388cbab010b69982c86e9e","impliedFormat":1},{"version":"9f642953aba68babd23de41de85d4e97f0c39ef074cb8ab8aa7d55237f62aff6","impliedFormat":1},{"version":"159d95163a0ed369175ae7838fa21a9e9e703de5fdb0f978721293dd403d9f4a","impliedFormat":1},{"version":"2d2ec3235e01474f45a68f28cf826c2f5228b79f7d474d12ca3604cdcfdac80c","impliedFormat":1},{"version":"6dd249868034c0434e170ba6e0451d67a0c98e5a74fd57a7999174ee22a0fa7b","impliedFormat":1},{"version":"9716553c72caf4ff992be810e650707924ec6962f6812bd3fbdb9ac3544fd38f","impliedFormat":1},{"version":"506bc8f4d2d639bebb120e18d3752ddeee11321fd1070ad2ce05612753c628d6","impliedFormat":1},{"version":"053c51bbc32db54be396654ab5ecd03a66118d64102ac9e22e950059bc862a5e","impliedFormat":1},{"version":"1977f62a560f3b0fc824281fd027a97ce06c4b2d47b408f3a439c29f1e9f7e10","impliedFormat":1},{"version":"627570f2487bd8d899dd4f36ecb20fe0eb2f8c379eff297e24caba0c985a6c43","impliedFormat":1},{"version":"0f6e0b1a1deb1ab297103955c8cd3797d18f0f7f7d30048ae73ba7c9fb5a1d89","impliedFormat":1},{"version":"0a051f254f9a16cdde942571baab358018386830fed9bdfff42478e38ba641ce","impliedFormat":1},{"version":"17269f8dfc30c4846ab7d8b5d3c97ac76f50f33de96f996b9bf974d817ed025b","impliedFormat":1},{"version":"9e82194af3a7d314ccbc64bb94bfb62f4bfea047db3422a7f6c5caf2d06540a9","impliedFormat":1},{"version":"083d6f3547ccbf25dfa37b950c50bee6691ed5c42107f038cc324dbca1e173ae","impliedFormat":1},{"version":"952a9eab21103b79b7a6cca8ad970c3872883aa71273f540285cad360c35da40","impliedFormat":1},{"version":"8ba48776335db39e0329018c04486907069f3d7ee06ce8b1a6134b7d745271cc","impliedFormat":1},{"version":"e6d5809e52ed7ef1860d1c483e005d1f71bab36772ef0fd80d5df6db1da0e815","impliedFormat":1},{"version":"893e5cfbae9ed690b75b8b2118b140665e08d182ed8531e1363ec050905e6cb2","impliedFormat":1},{"version":"6ae7c7ada66314a0c3acfbf6f6edf379a12106d8d6a1a15bd35bd803908f2c31","impliedFormat":1},{"version":"e4b1e912737472765e6d2264b8721995f86a463a1225f5e2a27f783ecc013a7b","impliedFormat":1},{"version":"97146bbe9e6b1aab070510a45976faaf37724c747a42d08563aeae7ba0334b4f","impliedFormat":1},{"version":"c40d552bd2a4644b0617ec2f0f1c58618a25d098d2d4aa7c65fb446f3c305b54","impliedFormat":1},{"version":"09e64dea2925f3a0ef972d7c11e7fa75fec4c0824e9383db23eacf17b368532f","impliedFormat":1},{"version":"424ddba00938bb9ae68138f1d03c669f43556fc3e9448ed676866c864ca3f1d6","impliedFormat":1},{"version":"a0fe12181346c8404aab9d9a938360133b770a0c08b75a2fce967d77ca4b543f","impliedFormat":1},{"version":"3cc6eb7935ff45d7628b93bb6aaf1a32e8cb3b24287f9e75694b607484b377b3","impliedFormat":1},{"version":"ced02e78a2e10f89f4d70440d0a8de952a5946623519c54747bc84214d644bac","impliedFormat":1},{"version":"efd463021ccc91579ed8ae62584176baab2cd407c555c69214152480531a2072","impliedFormat":1},{"version":"29647c3b79320cfeecb5862e1f79220e059b26db2be52ea256df9cf9203fb401","impliedFormat":1},{"version":"e8cdefd2dc293cb4866ee8f04368e7001884650bb0f43357c4fe044cc2e1674f","impliedFormat":1},{"version":"582a3578ebba9238eb0c5d30b4d231356d3e8116fea497119920208fb48ccf85","impliedFormat":1},{"version":"185eae4a1e8a54e38f36cd6681cfa54c975a2fc3bc2ba6a39bf8163fac85188d","impliedFormat":1},{"version":"0c0a02625cf59a0c7be595ccc270904042bea523518299b754c705f76d2a6919","impliedFormat":1},{"version":"c44fc1bbdb5d1c8025073cb7c5eab553aa02c069235a1fc4613cd096d578ab80","impliedFormat":1},{"version":"cee72255e129896f0240ceb58c22e207b83d2cc81d8446190d1b4ef9b507ccd6","impliedFormat":1},{"version":"3b54670e11a8d3512f87e46645aa9c83ae93afead4a302299a192ac5458aa586","impliedFormat":1},{"version":"c2fc4d3a130e9dc0e40f7e7d192ef2494a39c37da88b5454c8adf143623e5979","impliedFormat":1},{"version":"2e693158fc1eedba3a5766e032d3620c0e9c8ad0418e4769be8a0f103fdb52cd","impliedFormat":1},{"version":"516275ccf3e66dc391533afd4d326c44dd750345b68bb573fc592e4e4b74545f","impliedFormat":1},{"version":"07c342622568693847f6cb898679402dd19740f815fd43bec996daf24a1e2b85","impliedFormat":1},{"version":"fa40d705f9813843d47f19321591499f14d1a18fa5e8ca9beaee5aac633c3d0d","impliedFormat":1},{"version":"a7a6330fb015f72d821e23004e63a3827e0c632b614ef3a310b3c81b66de61fd","impliedFormat":1},{"version":"89968316b7069339433bd42d53fe56df98b6990783dfe00c9513fb4bd01c2a1c","impliedFormat":1},{"version":"a4096686f982f6977433ee9759ecbef49da29d7e6a5d8278f0fbc7b9f70fce12","impliedFormat":1},{"version":"62e62a477c56cda719013606616dd856cfdc37c60448d0feb53654860d3113bb","impliedFormat":1},{"version":"207c107dd2bd23fa9febac2fe05c7c72cdac02c3f57003ab2e1c6794a6db0c05","impliedFormat":1},{"version":"55133e906c4ddabecdfcbc6a2efd4536a3ac47a8fa0a3fe6d0b918cac882e0d4","impliedFormat":1},{"version":"2147f8d114cf58c05106c3dccea9924d069c69508b5980ed4011d2b648af2ffe","impliedFormat":1},{"version":"2eb4012a758b9a7ba9121951d7c4b9f103fe2fc626f13bec3e29037bb9420dc6","impliedFormat":1},{"version":"fe61f001bd4bd0a374daa75a2ba6d1bb12c849060a607593a3d9a44e6b1df590","impliedFormat":1},{"version":"cfe8221c909ad721b3da6080570553dea2f0e729afbdbcf2c141252cf22f39b5","impliedFormat":1},{"version":"34e89249b6d840032b9acdec61d136877f84f2cd3e3980355b8a18f119809956","impliedFormat":1},{"version":"6f36ff8f8a898184277e7c6e3bf6126f91c7a8b6a841f5b5e6cb415cfc34820e","impliedFormat":1},{"version":"4b6378c9b1b3a2521316c96f5c777e32a1b14d05b034ccd223499e26de8a379c","impliedFormat":1},{"version":"07be5ae9bf5a51f3d98ffcfacf7de2fe4842a7e5016f741e9fad165bb929be93","impliedFormat":1},{"version":"cb1b37eda1afc730d2909a0f62cac4a256276d5e62fea36db1473981a5a65ab1","impliedFormat":1},{"version":"195f855b39c8a6e50eb1f37d8f794fbd98e41199dffbc98bf629506b6def73d7","impliedFormat":1},{"version":"471386a0a7e4eb88c260bdde4c627e634a772bf22f830c4ec1dad823154fd6f5","impliedFormat":1},{"version":"108314a60f3cb2454f2d889c1fb8b3826795399e5d92e87b2918f14d70c01e69","impliedFormat":1},{"version":"d75cc838286d6b1260f0968557cd5f28495d7341c02ac93989fb5096deddfb47","impliedFormat":1},{"version":"d531dc11bb3a8a577bd9ff83e12638098bfc9e0856b25852b91aac70b0887f2a","impliedFormat":1},{"version":"19968b998a2ab7dfd39de0c942fc738b2b610895843fec25477bc393687babd8","impliedFormat":1},{"version":"c0e6319f0839d76beed6e37b45ec4bb80b394d836db308ae9db4dea0fe8a9297","impliedFormat":1},{"version":"1a7b11be5c442dab3f4af9faf20402798fddf1d3c904f7b310f05d91423ba870","impliedFormat":1},{"version":"079d3f1ddcaf6c0ff28cfc7851b0ce79fcd694b3590afa6b8efa6d1656216924","impliedFormat":1},{"version":"2c817fa37b3d2aa72f01ce4d3f93413a7fbdecafe1b9fb7bd7baaa1bbd46eb08","impliedFormat":1},{"version":"682203aed293a0986cc2fccc6321d862742b48d7359118ac8f36b290d28920d2","impliedFormat":1},{"version":"7406d75a4761b34ce126f099eafe6643b929522e9696e5db5043f4e5c74a9e40","impliedFormat":1},{"version":"7e9c4e62351e3af1e5e49e88ebb1384467c9cd7a03c132a3b96842ccdc8045c4","impliedFormat":1},{"version":"ea1f9c60a912065c08e0876bd9500e8fa194738855effb4c7962f1bfb9b1da86","impliedFormat":1},{"version":"903f34c920e699dacbc483780b45d1f1edcb1ebf4b585a999ece78e403bb2db3","impliedFormat":1},{"version":"100ebfd0470433805c43be5ae377b7a15f56b5d7181c314c21789c4fe9789595","impliedFormat":1},{"version":"12533f60d36d03d3cf48d91dc0b1d585f530e4c9818a4d695f672f2901a74a86","impliedFormat":1},{"version":"21d9968dad7a7f021080167d874b718197a60535418e240389d0b651dd8110e7","impliedFormat":1},{"version":"2ef7349b243bce723d67901991d5ad0dfc534da994af61c7c172a99ff599e135","impliedFormat":1},{"version":"fa103f65225a4b42576ae02d17604b02330aea35b8aaf889a8423d38c18fa253","impliedFormat":1},{"version":"1b9173f64a1eaee88fa0c66ab4af8474e3c9741e0b0bd1d83bfca6f0574b6025","impliedFormat":1},{"version":"1b212f0159d984162b3e567678e377f522d7bee4d02ada1cc770549c51087170","impliedFormat":1},{"version":"46bd71615bdf9bfa8499b9cfce52da03507f7140c93866805d04155fa19caa1b","impliedFormat":1},{"version":"86cb49eb242fe19c5572f58624354ffb8743ff0f4522428ebcabc9d54a837c73","impliedFormat":1},{"version":"fc2fb9f11e930479d03430ee5b6588c3788695372b0ab42599f3ec7e78c0f6d5","impliedFormat":1},{"version":"bb1e5cf70d99c277c9f1fe7a216b527dd6bd2f26b307a8ab65d24248fb3319f5","impliedFormat":1},{"version":"817547eacf93922e22570ba411f23e9164544dead83e379c7ae9c1cfc700c2cf","impliedFormat":1},{"version":"a728478cb11ab09a46e664c0782610d7dd5c9db3f9a249f002c92918ca0308f7","impliedFormat":1},{"version":"9e91ef9c3e057d6d9df8bcbfbba0207e83ef9ab98aa302cf9223e81e32fdfe8d","impliedFormat":1},{"version":"66d30ef7f307f95b3f9c4f97e6c1a5e4c462703de03f2f81aca8a1a2f8739dbd","impliedFormat":1},{"version":"293ca178fd6c23ed33050052c6544c9d630f9d3b11d42c36aa86218472129243","impliedFormat":1},{"version":"90a4be0e17ba5824558c38c93894e7f480b3adf5edd1fe04877ab56c56111595","impliedFormat":1},{"version":"fadd55cddab059940934df39ce2689d37110cfe37cc6775f06b0e8decf3092d7","impliedFormat":1},{"version":"91324fe0902334523537221b6c0bef83901761cfd3bd1f140c9036fa6710fa2b","impliedFormat":1},{"version":"b4f3b4e20e2193179481ab325b8bd0871b986e1e8a8ed2961ce020c2dba7c02d","impliedFormat":1},{"version":"41744c67366a0482db029a21f0df4b52cd6f1c85cbc426b981b83b378ccb6e65","impliedFormat":1},{"version":"c3f3cf7561dd31867635c22f3c47c8491af4cfa3758c53e822a136828fc24e5d","impliedFormat":1},{"version":"a88ddea30fae38aa071a43b43205312dc5ff86f9e21d85ba26b14690dc19d95e","impliedFormat":1},{"version":"b5b2d0510e5455234016bbbaba3839ca21adbc715d1b9c3d6dede7d411a28545","impliedFormat":1},{"version":"5515f17f45c6aafe6459afa3318bba040cb466a8d91617041566808a5fd77a44","impliedFormat":1},{"version":"4df1f0c17953b0450aa988c9930061f8861b114e1649e1a16cfd70c5cbdf8d83","impliedFormat":1},{"version":"441104b363d80fe57eb79a50d495e0b7e3ebeb45a5f0d1a4067d71ef75e8fbfa","impliedFormat":1},"c4d28262126a059a8b1f41d033ba2d52dc3ad8b7abab4dcac5baff70a2a35e6f","f76e6e1051b6def981d26f145454724a906d92608c29b2b93a474b6ca9ed0efd","c0a92feef3ed5b2ab38b0825b8bc9f591394d15ca5f1556853d53b79d78fad71",{"version":"c7074b8eae9a4471112877efb9fccc4ff50d400637b54034f4c697369b8e0c2f","impliedFormat":1},{"version":"000862b3dbaa3ba8112440086eebcf916f578f48f918882f545596c687272177","impliedFormat":1},{"version":"20c1111aa9534c7b7df1ddee02011a3d7067a791c4d5901dced9830fd8d486f3","impliedFormat":1},{"version":"7d3ecc412ca6fb2140ee8718a3d0f15c2cf2990b5891e6e1a40205ef7a4ec96a","impliedFormat":1},{"version":"07ca9984d5e03a45cffbb4c6a06178180667216450d8348a307323d0998d4bf4","impliedFormat":1},{"version":"1910301ed7c3c39bc34fc9715414b5355e4fee0401ae65e3e43ebf012bb1a63b","impliedFormat":1},{"version":"d477e2b38d400c14b60c07c0994601e129c5f17c7f958117601b63efa8e8017b","impliedFormat":1},{"version":"4182411ed303d53edbfe1bb80781f90af30c666fd957989037141bea45431668","impliedFormat":1},{"version":"9995b9787d6034dfe7d1bc6f271dd8bb21fd1d1903a08a62434f161682dcb457","impliedFormat":1},{"version":"66971fbc40a80f916b8ca2e94d5d6c3e35994d11065d0eb03ec41b370adfed97","impliedFormat":1},{"version":"3bc0e49ff8fdad5206ee15a435a7adc73bd293e2097ecc676fd4e0f5eafeee81","impliedFormat":1},{"version":"72f8a4bdb822b85e3cb6397971c39f277c70132b161bb12bebd94b163e2f73ae","impliedFormat":1},{"version":"ab75746ac80c9a08def860982af72a6f1038351901ea8603e71b3e8707668ad1","impliedFormat":1},{"version":"fec0d866732b6c774a60394b5a6d33dac1af6daa3fea1e53ae5975d599180a85","impliedFormat":1},{"version":"b9c5772b895e937641d51d4c2110751a60f12b760f67bcf4f36a8a2477aa833a","impliedFormat":1},{"version":"01799fc79b10be8bafc36e1c7b15f4bae98bb2da8100c040e83e1471ef37fa2e","impliedFormat":1},{"version":"ff49baf4336cacc5ea8646e723c8e4a3bdfbb9ed2f34c20e8c2de1d05b670b20","impliedFormat":1},{"version":"bcabb3091911455814cece1c2817e30f89cf4946f2da15b43472b9a885a56d55","impliedFormat":1},{"version":"4151431363ca6cca001810e3d416b4bee0a562330d26499f802764b2b51aca9e","impliedFormat":1},{"version":"e1ea7b24ce79efe8d19c880a1b4e004a06d6292f459ac153654e24d1ae87d9fc","impliedFormat":1},{"version":"825689d3a15da56a7f985aee87810c3392fb598646ba45cfdb0cf5cb58e9d130","impliedFormat":1},{"version":"49c0281e4c32f99df95722a8c10ba2703631c696c3b3330fc5feb9a9bd0f4eaa","impliedFormat":1},{"version":"b1f64c6aad9676f268b9114c4cf4a5cf4bb29abe936ab73f1c43f5f7dda28f2c","impliedFormat":1},{"version":"ad40c811757b8870312ce3da99ca4b06c6604c0d8139a3189f68c5314aff5fc4","impliedFormat":1},{"version":"e5c3362ea58b105f4468e63f9b3f746fcdb5714024931231ce9b1d59b74c6d67","impliedFormat":1},{"version":"2775b4acaa4ff841a96eeb3e477cb82c84b4c7e3c157acc18901cbe4edd06b77","impliedFormat":1},{"version":"bc0b40ef267631af97290073624c27a6722124d50d9699e29c1f8507ecb6a282","impliedFormat":1},{"version":"1ae6757ae8f0331b54b66381fe7e7cba2837f3681afb9278bf9f163bd5039466","impliedFormat":1},{"version":"b1229442187bea202ab606cd64cf583692e0c57581784dcb21667c7b1122eea1","impliedFormat":1},{"version":"092042bc75b55f727ed62a4e2579c1f010e9d404d6fcf1954473a01d9e94484b","impliedFormat":1},{"version":"38c77a69740b7655457c36f6d115182693f3ab677949682de677fa2dfb2599bb","impliedFormat":1},{"version":"f48592a06d340ee95784a3242b75710000d5e4a467958ed3b2ee8f2ece278b44","impliedFormat":1},{"version":"6ed0949ed569a5b26fe0b988267a6b93be0e525a2ab6098ce8fee2436806c918","impliedFormat":1},{"version":"91305a029dfab85b854e6d31697dc43b0657eb7536aacaa65c6d74da5b437857","impliedFormat":1},{"version":"cfa2d15f468650ea455799c4ebb4d0698fa8fe4665bc73a46d0b0c443e5bc5e3","impliedFormat":1},{"version":"64dec872f5228ebc07812a65af8b1269a145aa6d1fa6f47546abce9a3a15fd3c","impliedFormat":1},{"version":"940c0de32ddda6212e5543177fd063498f6cf422911c6bff0a2086cbb11f5ecf","impliedFormat":1},{"version":"8f13981832313611bfcf90fd96de1218744f759a0175914eaf406ffd4919cbaa","impliedFormat":1},{"version":"dcbc04cc196ef0ed45d4fb8925e867cb463d4db787d9039ecabb737fcf12ab79","impliedFormat":1},{"version":"7e61f20558c56cf82160c3edacc548312a35b1c7f11704f766746ef64f0e4c74","impliedFormat":1},{"version":"c3256abdaaf40e7e73c7c24ec2fba77569a8f03a096b2f9582fe7520d7fff1d1","impliedFormat":1},{"version":"728d7b16a89022b6a86d1ff1e4791c39a4401acc869d22a87baeb37b323036c6","impliedFormat":1},{"version":"f1f6923ec959a00397a2dd72864d9065d823d7aa078d9840af6b78c84b500146","impliedFormat":1},{"version":"f957b495c6b8f1f14a15661e849c02a8def9dc35840e8d2ddc2046bbe14bd547","impliedFormat":1},{"version":"21f98579035b0a0bfbb28996ad331c5aa79302624779c01bc077257c9d91c114","impliedFormat":1},{"version":"7a89725f1817ff1e6cb6b167ad84df1fc851193c66719ce79bc076043eddf118","impliedFormat":1},{"version":"81bb7e34312099a7d367187d7732afb3b21d5210c44a79db857e3920d5e148b6","impliedFormat":1},{"version":"1fe171ffb872e40a0a9a642ba529e87b597b1d67931878c33ac4d1bbb1df58be","impliedFormat":1},{"version":"5b1186921f801ae8e283eb2249ef6e27ae9f8a1834e57f2e16949cd3991846d6","impliedFormat":1},{"version":"b41fd4b7e7414a10054c8a498b0f452c429e63a5cfc621606f58752dccbe85fd","impliedFormat":1},{"version":"37ce26a13df5aede9c6b795ac7a26505ea91874a30b165307e2f264efdfb6511","impliedFormat":1},{"version":"d998bca69f80fc1f04757aa1cfc79ab2e1db7bba0ef4a30869156e3357386fc3","impliedFormat":1},{"version":"1c812a7d6565b034096fa5b5630f3e7159c0c289f525e01b3718f337a2cdeca1","impliedFormat":1},{"version":"c6d2376be1c9805e0ef5de09250553dabbabd67e508d5e922ac5d0902c98aec5","impliedFormat":1},{"version":"e2f44b8f481c2298220d49a59f91a8f33cf0c7075006b401010eb5b73c560453","impliedFormat":1},{"version":"377bb18a79fd85c00b731bb60b33e22a8984b4f97ba6825d5a4564a801038e7a","impliedFormat":1},{"version":"db382b318bc333eb7548cc1f0dcd149794d7e1d4969d54f9586f2533b287b5ac","impliedFormat":1},{"version":"ace39b5aee44e612eadcdf34d3d1f99ae5ff0a175a3fa8903fabb0953570947e","impliedFormat":1},{"version":"570801b10794486f9535c6cb55658aade8a2fa34d6e320062f8c704d49042818","impliedFormat":1},{"version":"0d6fee47fa4c30b31233fd5c6a69b238ee9944a680c69997c7c4093fe3bd6367","impliedFormat":1},{"version":"8bba3c4a509f012d471e1c4973be2b5ddbf81ae6347c99ca0b067fa6e3ba8b17","impliedFormat":1},{"version":"bda138ca76c8291b66ffc74c10f74a8152488da3a61e577bc1d0de329d0502cf","impliedFormat":1},{"version":"de6352a242e56c4df1829349aa0ae1f6e5fafb1326581f7bd1fd9eb3cf5772c2","impliedFormat":1},{"version":"48ff1dd366aef7a6d48201d4e830e99177cf115fa22f5576f0c899fe75280f5a","impliedFormat":1},{"version":"263b4d2898ad1b86569ca63585374d6c42685ae1048c52ffe964d1f62eb9a6f1","impliedFormat":1},{"version":"e21f31c0d0e10034621862f5d37868fc9c1d8e05619b16971f7cd3110f9b1be6","impliedFormat":1},{"version":"025ab7b7c805c3e4a5e4bd3f76cb8c092bec146a3240d7004f78a8b25c467e52","impliedFormat":1},{"version":"41aa1ef57640c8f90093237fff863a702317ef22e962cf2047460d59d3acadd8","impliedFormat":1},{"version":"c1c8a6cde0253f15cfa13429e43f10cc3944428150a308bb40cdb75c9565f784","impliedFormat":1},{"version":"7a1e098bf3c979054761df1cb1e3d09463c734977629576df5ae2be4e4aadb99","impliedFormat":1},{"version":"ebd61ac5f4d954aeb24e1157027a8dd7252cc2f03f733e864f8e38ac5bfac306","impliedFormat":1},{"version":"092c372db6eca705f0c4a3a265e8370e1cb5992c96e4bc8a42e8a71d13ac5d8a","impliedFormat":1},{"version":"e69ef73426bca8394b75d76b7bb582666fbf8fafe2f37d0ee8f95cddbcebca68","impliedFormat":1},{"version":"4a347a72f5f2de8132f13ff85c4207022898783be27e4823c8d811840ae4afe8","impliedFormat":1},{"version":"292c9cd91e57d749d8b83b0002b88fc6e47ae5c706659383ebdd9181ff302abc","impliedFormat":1},{"version":"707e26f766b6f7403f244e7311923a928dde53750a535bdd9b4ea7a9c43cd13b","impliedFormat":1},{"version":"7fe6758ebcb7aa7c040ada95a40149f4f934b5c43b3334058ca7b098b053af04","impliedFormat":1},"075a41e8502d988d0c2288da38385b550feff350999154a62b95c4b7ed01e1dd","3c513bc12d7b887ebb467926b62501ef95677fefe76ab6b03ce0181cd2607e16","74a07acc2cff258305d25e776fac39f232c0cf8560b734e86dad556ccdb23b2d","1e65a94291938e29a024095f77aa7353daac8bbef268d7f4a803768622802b0c","f6254ae3fe78ef0eaf68730ea699f2e42b252667be74cf72c6db25a544d27d93","eb8099e4302072b5a4f719d532d5df0deb2a44b4fdeaad27d558d50e36de2b56","fe97216a67418380ac5672428d091908e6ea7cb00d3decb31d2419a6c628b0ab","c82046f56c04365ac499133006ab3794eacfc6c34ce0c4a26af6d84620b690f1","da284f58e519c1623d91118aa952dda2f0466b87ff4a1ebc50da40be9c8dc8c2","dd759f46502f695295bd1bb9fcc50e922ca644476fcf3b859372f73649502b10","067ca1372640033d90b4616ae0e74d753e3a892acc1d6233b249e2ecf2e060b9","af157ccabf5a595d7b5fd8005112d7519ade5e875f97077db6011c491f54acd3","01ddc178a0411346e77db50aec1f1b75e1f3914bf18ee55591725bfe6bd81f76","ea8269c1195f08a6746af6685d459fa3fbd49c7d7e0606008516fe6b9ba5fd12","f50ebdca4933ff46486f2f25875987b94c3aadbb0b26427ab2f453a71324b321",{"version":"ce7d79da26b858bb29a8c4f2cad8fa64ce123c28a561bd7356bb391dc135b753","impliedFormat":1},{"version":"6aac2c5ca00378e4d1421a03f614643dc1e9fd02279257cbf2e8e2a713b00907","impliedFormat":1},{"version":"254510b0a3c2e04f55e98ae89a6aa42f67852c192c3502b3b8488e578b21c9d6","impliedFormat":1},{"version":"b75be7355591118207e7f24143b27a860da4043a1950c746e034313d9ded4137","impliedFormat":1},{"version":"da15f699f56ab6a37b4eca73eb14a356f5d175d979f0c8197d325d5f23c91bd6","impliedFormat":1},{"version":"9b2fe69aa18037c1f47191b263016cc3edee57afb6660b1ae37c0b491aa4190c","impliedFormat":1},{"version":"d8c3b3c16a4a8656dcdd394df0df07d3149816cb96a89935d62cafe4dd84009a","impliedFormat":1},{"version":"e982879e6ea8ddf8899f637e639bc225996a729e07f068afb120d32fb4feebf2","impliedFormat":1},{"version":"94616e40e31224cb261a78c5cb96fd3f65f9ead7052eac20fc6c975714f3840c","impliedFormat":1},{"version":"931574e125523649902eee2db57c221a1b36417db4f2c4665bf38ce2170ea06e","impliedFormat":1},{"version":"cd0c8c8b5002ec4cac9e8a5e26d853549c5c446a670fb375b9c052b345fb5da1","impliedFormat":1},{"version":"7d27796c034612b6016db97555b84f1005dc3d55e2286379d48ec8db475b6430","impliedFormat":1},{"version":"0d59de214eefc455e13a7f747c011729ee76f1554fdef55554ecf4bfeb20568b","impliedFormat":1},{"version":"e16ecf37f6f2ca79ff19ba2e4c3697ecd9d38b8d01bf6682bc4003d0d5719651","impliedFormat":1},{"version":"845154327584247966f7dea7a3e4960906b7038cbe23ab43fb198539ca12204f","impliedFormat":1},{"version":"cce34c68dd760a55d002eaa02390985f4aeaa39786679f54ade28be6229792e9","impliedFormat":1},{"version":"877388f59a044fc4c4689637425d4f8762662b4c6dc86d55864ca8816382b69e","impliedFormat":1},{"version":"162ffbed80dad8ce0cf81c330c88dccaae85425fb457a6afcae0110419bdedfb","impliedFormat":1},{"version":"a85d6e7924c263fdb7a9e28a578401f2f96950ff9fd0e250c76f25de5ce3b9f2","impliedFormat":1},{"version":"b42b1904665d7ead2610cbadddb80163a637ad2d202ff446af2382bf75b38353","impliedFormat":1},{"version":"57947d16b34a3811f854965fe668e81ccea9dd6321e412ea1a2c75d4fd2619c1","impliedFormat":1},{"version":"e9d4bfe42849ba995ab572beba5f30bd484e88f9441a4eb223a54ddec0c4d490","impliedFormat":1},{"version":"6c36e6a90dd3e1f5e488c56aede7168ea429589d0095285701b7b56f47262bbd","impliedFormat":1},{"version":"4bc4c7612f5cc6298b01f76f7a21674181ae6e199a0b07c518107c15bde32344","impliedFormat":1},"5bb7f656feaa86665b60ad8fb1c9b6523a48b322926016b794b4b194272251d8",{"version":"569e762cf47aafdad508360a443c6c757e56c61db3b652b65458a7d168d139c4","impliedFormat":1},{"version":"02ed2766d79a00719ac3cc77851d54bd7197c1b12085ea12126bc2a65068223e","impliedFormat":1},{"version":"4b84373e192b7e0f8569b65eb16857098a6ee279b75d49223db2a751fdd7efde","impliedFormat":1},{"version":"5aeea312cd1d3cc5d72fc8a9c964439d771bdf41d9cce46667471b896b997473","impliedFormat":1},{"version":"cfa7bf135cafc5aad7cf544bc1cebf65a1fdb4373223cc85ea7d7196e18be151","impliedFormat":1},{"version":"1ba59c8bbeed2cb75b239bb12041582fa3e8ef32f8d0bd0ec802e38442d3f317","impliedFormat":1},{"version":"f2c4a36eb216aadb0d9c79862a31b922ccfa1eaaa38d2124cc9192d40eda4779","impliedFormat":1},{"version":"cb5bb1db16ff4b534f56f7741e7ffd0a007ce36d387a377d4c196036e0932423","impliedFormat":1},{"version":"25be1eb939c9c63242c7a45446edb20c40541da967f43f1aa6a00ed53c0552db","impliedFormat":1},{"version":"08c2bb524b8ed271f194e1c7cc6ad0bcc773f596c41f68a207d0ec02c9727060","impliedFormat":1},{"version":"012b69bc8a16a21aa0863502339c49258c579723f9e7a54faa5f0d5c2b1ae1b7","impliedFormat":1},{"version":"29ad73d9e365d7b046f3168c6a510477bfe30d84a71cd7eb2f0e555b1d63f5f6","impliedFormat":1},{"version":"d99e9f5aa43397599fe824e38c33d13d3a9e19198806a4363114bd7ac58b29cc","impliedFormat":1},{"version":"440099416057789b14f85af057d4924915f27043399c10d4ca67409d94b963cf","impliedFormat":1},{"version":"ac44995fc7d0781d77927bae7dd41a31f0309e695fd2694b175c0ce3bc4b3b50","impliedFormat":1},{"version":"0c1f802f7a60ca8084e5188ac7952accdfc00f39ded3ebbbd3cdcc9da51b9a7b","impliedFormat":1},{"version":"a32e3fc530d8d1a18bf54678d8d55714827a50c9fabdd4ede7155a56be7d1dcb","impliedFormat":1},{"version":"14ecfc29e0c44ad4c5e50f9b597492cd8f45a2a635db8b5fe911a5da83e26cf8","impliedFormat":1},{"version":"ba0badf6398010ef0ba77947d4b52078fc84cb4f209f6b70194b3b2cfb1823d8","impliedFormat":1},{"version":"c2f041fe0e7ae2d5a19c477d19e8ec13de3d65ef45e442fa081cf6098cdcbe2d","impliedFormat":1},{"version":"0cef678147928ef223ff7f2aae3442cc9f4e9996956e9ac92434e626d0e147f8","impliedFormat":1},{"version":"198ae766bb698feb66d3188cfce59fb33696c951b10f901aa3fc3db4847ce76a","impliedFormat":1},{"version":"6dc488fd3d01e4269f0492b3e0ee7961eec79f4fc3ae997c7d28cde0572dbd91","impliedFormat":1},{"version":"a09b706f16bda9372761bd70cf59814b6f0a0c2970d62a5b2976e2fd157b920f","impliedFormat":1},{"version":"70da4bfde55d1ec74e3aa7635eae741f81ced44d3c344e2d299e677404570ca9","impliedFormat":1},{"version":"bf4f6b0d2ae8d11dc940c20891f9a4a558be906a530b9d9a8ff1032afa1962cd","impliedFormat":1},{"version":"9975431639f84750a914333bd3bfa9af47f86f54edbaa975617f196482cfee31","impliedFormat":1},{"version":"70a5cb56f988602271e772c65cb6735039148d5e90a4c270e5806f59fc51d3a0","impliedFormat":1},{"version":"e083384623f90cfa7e8d2aa7efe78c51095a04ad51d1f82c3e4052689666895d","impliedFormat":1},"9133a4808e90caf54141a8b269a876a7c0034b6b357bbe9eeb0a7b897c7302a8","af8faa264a9ff666504d9013a839a2a428bbdc5a492802ba25f6e5f9ef37bd4a","aa6dfd7e3afccd316b259d32a8271ee1c6ef4fa35e8c5f08c5be3ae5420b45d7","c3a33ba6ff053857255c6841ede6c5e4bb329b9ceb03ff170fb3087a5afb2cf6","bfdfb65fc7c386922dac918571c5c58e68329766af83e864f5f0f2fc98575763","33d16f873662f479e8a881e96bdaba54e4fb8dd1f9f668f10552426d0a9a5c7f","ca5a7a6860efea3d406104be2f4114fed70ce8ead702618cb5d8a81959eca20b","73588b5d96a4da41d70532502a974de135087f8465e5b5b6fd55c7f676ad3cda","9006e088df0f3fb0acf62bb23be994dcd5cb57c78703a8c6715d8aa21f1ea5d5","774cb5e4ef0d74c55788fd5aa8a886ed0f8c545cbf8cfdadfb36fdda44910530","7c34770056f0e8ca28210395c8e2d50c459fcec7f8f794f826b42bb05be1c762","84d5b427012a58c51dfed10c1dee80801b0b1ae9f24b605d75f4a2db38593f80","9ab65cf0db9e9746ceb4f9babe2506a6641090940b809193382b33d2f1a316f9","0e90823cd087ce76b514e43e3416a7a23bb7515fab75c186a847c0391d272cd7","87b6e7f5c39c67815349e6a074a1ae9f6737ff59e4aad8aa6481a43a139d9e91","1954d1f845dc4e1d0e34778252f665e07669b626166e5f6b0eccca2b9fed5925","030a24976b35ddbe5072fc68a708b263946c27d2e61da4a37f747ebfc86d0dc8","693591e80abbc59c20721f97c4db91a796e03db6ef9cf96e20ebd2ee26fdb498","aa9b77aef4886c7a4dbaac70a9a0033657c4426c79351220f76ef3bffd9ea0b3","d27b907520a3569aa4902f8d314a390ebf7d89dafc8863846fcf85b3883f38ed","bece646109b5c925f75b7f9768c79dedce4325f15ff35aafa37439ac48c02376","03deefbd958ef0360d922c649920f7af09923fd8464a2199106d49fd8fb7459d","a16b055c730b743691d096ca5f0715c592020f6dddbcf7f49e3d13e49fae3979","0f28aa562b4d0cee9b6a415ff45e5423f784164cc7430bebb7bfe2d85be481c8","9906064dc56c62a1715226f9cdfff219c9457ec0f21a9d97179ad602a8208de5","b1023a0eaf9965e3dd788950b1990f0b5a806d842d1fe5e5c3ed24708173c473","1a0be0a23add3b5c53b1dfb72b06f1415e1c4b135a4c529ac89b376ea5e34f65",{"version":"b6e995b5ef6661f5636ff738e67e4ec90150768ef119ad74b473c404304408a1","impliedFormat":1},{"version":"5d470930bf6142d7cbda81c157869024527dc7911ba55d90b8387ef6e1585aa1","impliedFormat":1},{"version":"074483fdbf20b30bd450e54e6892e96ea093430c313e61be5fdfe51588baa2d6","impliedFormat":1},{"version":"b7e6a6a3495301360edb9e1474702db73d18be7803b3f5c6c05571212acccd16","impliedFormat":1},{"version":"aa7527285c94043f21baf6e337bc60a92c20b6efaa90859473f6476954ac5f79","impliedFormat":1},{"version":"dd3be6d9dcd79e46d192175a756546630f2dc89dab28073823c936557b977f26","impliedFormat":1},{"version":"8d0566152618a1da6536c75a5659c139522d67c63a9ae27e8228d76ab0420584","impliedFormat":1},{"version":"ba06bf784edafe0db0e2bd1f6ecf3465b81f6b1819871bf190a0e0137b5b7f18","impliedFormat":1},{"version":"a0500233cb989bcb78f5f1a81f51eabc06b5c39e3042c560a7489f022f1f55a3","impliedFormat":1},{"version":"220508b3fb6b773f49d8fb0765b04f90ef15caacf0f3d260e3412ed38f71ef09","impliedFormat":1},{"version":"1ad113089ad5c188fec4c9a339cb53d1bcbb65682407d6937557bb23a6e1d4e5","impliedFormat":1},{"version":"e56427c055602078cbf0e58e815960541136388f4fc62554813575508def98b6","impliedFormat":1},{"version":"1f58b0676a80db38df1ce19d15360c20ce9e983b35298a5d0b4aa4eb4fb67e0f","impliedFormat":1},{"version":"3d67e7eb73c6955ee27f1d845cae88923f75c8b0830d4b5440eea2339958e8ec","impliedFormat":1},{"version":"11fec302d58b56033ab07290a3abc29e9908e29d504db9468544b15c4cd7670d","impliedFormat":1},{"version":"c66d6817c931633650edf19a8644eea61aeeb84190c7219911cefa8ddea8bd9a","impliedFormat":1},{"version":"ab1359707e4fc610c5f37f1488063af65cda3badca6b692d44b95e8380e0f6c2","impliedFormat":1},{"version":"37deda160549729287645b3769cf126b0a17e7e2218737352676705a01d5957e","impliedFormat":1},{"version":"d80ffdd55e7f4bc69cde66933582b8592d3736d3b0d1d8cc63995a7b2bcca579","impliedFormat":1},{"version":"c9b71952b2178e8737b63079dba30e1b29872240b122905cbaba756cb60b32f5","impliedFormat":1},{"version":"b596585338b0d870f0e19e6b6bcbf024f76328f2c4f4e59745714e38ee9b0582","impliedFormat":1},{"version":"e6717fc103dfa1635947bf2b41161b5e4f2fabbcaf555754cc1b4340ec4ca587","impliedFormat":1},{"version":"c36186d7bdf1f525b7685ee5bf639e4b157b1e803a70c25f234d4762496f771f","impliedFormat":1},{"version":"026726932a4964341ab8544f12b912c8dfaa388d2936b71cc3eca0cffb49cc1d","impliedFormat":1},{"version":"83188d037c81bd27076218934ba9e1742ddb69cd8cc64cdb8a554078de38eb12","impliedFormat":1},{"version":"7d82f2d6a89f07c46c7e3e9071ab890124f95931d9c999ba8f865fa6ef6cbf72","impliedFormat":1},{"version":"4fc523037d14d9bb6ddb586621a93dd05b6c6d8d59919a40c436ca3ac29d9716","impliedFormat":1},"9d68d227f7a7e0ed3a06486d8ab6283ab503281257f340b8ffe865ddca037689","52a54148c8255bacdf9b78bb93d407ff48a28ae457379414f37ffc2a4fe5d31c","8c0904f48cc294408fc64a16b6aa4236031a5cd1a6ee396ee98e38c1f10da4de","4f92c127bfef75d7246edfeaa251df206ec867eb204ecc84f0c39c6f05971dd1","857c4d08a528c371a9bba8c71e28642e16ae6a7eaa79b156ec4d5528ac411d01","de964154c78f62cd0fefaa7ecfe7e17676ce32ff4a411a74c5be1d61717229a8",{"version":"9b50ee091a0f3aa69aa51ab09233b0e834772a5e537dd1c85fc9963256d82d17","impliedFormat":1},{"version":"46acc28f4b194c3cc7d1a7d79310ea91925c449cb272aa820628a8609dd0a447","impliedFormat":1},{"version":"78647004e18e4c16b8a2e8345fca9267573d1c5a29e11ddfee71858fd077ef6e","impliedFormat":1},{"version":"0804044cd0488cb7212ddbc1d0f8e1a5bd32970335dbfc613052304a1b0318f9","impliedFormat":1},{"version":"b725acb041d2a18fde8f46c48a1408418489c4aa222f559b1ef47bf267cb4be0","impliedFormat":1},{"version":"85084ae98c1d319e38ef99b1216d3372a9afd7a368022c01c3351b339d52cb58","impliedFormat":1},{"version":"898ec2410fae172e0a9416448b0838bed286322a5c0c8959e8e39400cd4c5697","impliedFormat":1},{"version":"692345a43bac37c507fa7065c554258435ab821bbe4fb44b513a70063e932b45","impliedFormat":1},{"version":"7283aaea55499e5a9760db0ef199e3c12a35ff5391733fe1495b2d3add0c58ab","impliedFormat":1},{"version":"c1c8ccb14c76efb31ff84038ec7833a5715ba23e681b158b3c83cc012b8c3cfa","impliedFormat":1},{"version":"abb0a41fa0432cdec9595dbe84bd3efc5b59f04aa85dcd5cec7b453908fc85c3","impliedFormat":1},{"version":"522edc786ed48304671b935cf7d3ed63acc6636ab9888c6e130b97a6aea92b46","impliedFormat":1},{"version":"a9607a8f1ce7582dbeebc0816897925bf9b307cc05235e582b272a48364f8aa0","impliedFormat":1},{"version":"de21641eb8edcbc08dd0db4ee70eea907cd07fe72267340b5571c92647f10a77","impliedFormat":1},{"version":"48af3609dc95fa62c22c8ec047530daf1776504524d284d2c3f9c163725bdbd4","impliedFormat":1},{"version":"6758f7b72fa4d38f4f4b865516d3d031795c947a45cc24f2cfba43c91446d678","impliedFormat":1},{"version":"1fefab6dc739d33b7cb3fd08cd9d35dd279fcd7746965e200500b1a44d32db9e","impliedFormat":1},{"version":"997b94a03707d35abe497e427bc26b403a538838c3a82f2be71d85109b88e32b","impliedFormat":1},{"version":"bdf7abbd7df4f29b3e0728684c790e80590b69d92ed8d3bf8e66d4bd713941fe","impliedFormat":1},{"version":"8decb32fc5d44b403b46c3bb4741188df4fbc3c66d6c65669000c5c9cd506523","impliedFormat":1},{"version":"4beaf337ee755b8c6115ff8a17e22ceab986b588722a52c776b8834af64e0f38","impliedFormat":1},{"version":"c26dd198f2793bbdcc55103823a2767d6223a7fdb92486c18b86deaf63208354","impliedFormat":1},{"version":"93551b302a808f226f0846ad8012354f2d53d6dedc33b540d6ca69836781a574","impliedFormat":1},{"version":"040cb635dff5fc934413fa211d3a982122bf0e46acae9f7a369c61811f277047","impliedFormat":1},{"version":"778b684ebc6b006fcffeab77d25b34bf6e400100e0ec0c76056e165c6399ab05","impliedFormat":1},{"version":"285d50a08440314f7aea3246a5e15acbc38e2867ff07d21ef457ae8cb4e8a31f","impliedFormat":1},{"version":"672701f824ff6b9dab50514c6c4db711fb7ec5e7c2f0f6bc27b2dbe2e445a52a","impliedFormat":1},{"version":"be8f369f8d7e887eab87a3e4e41f1afcf61bf06056801383152aa83bda1f6a72","impliedFormat":1},{"version":"352bfb5f3a9d8a9c2464ad2dc0b2dc56a8212650a541fb550739c286dd341de1","impliedFormat":1},{"version":"ebef680e3597d7b3c8a9fc9e5581eb078461fa1406ded8d9859353dd6286eff2","impliedFormat":1},{"version":"dc923d45c03261241cefb03ea952883784e28fbc1d089af1a5a553a7b0f5e2b5","impliedFormat":1},{"version":"764150c107451d2fd5b6de305cff0a9dcecf799e08e6f14b5a6748724db46d8a","impliedFormat":1},{"version":"b04cf223c338c09285010f5308b980ee6d8bfa203824ed2537516f15e92e8c43","impliedFormat":1},{"version":"4b387f208d1e468193a45a51005b1ed5b666010fc22a15dc1baf4234078b636e","impliedFormat":1},{"version":"70441eda704feffd132be0c1541f2c7f6bbaafce25cb9b54b181e26af3068e79","impliedFormat":1},{"version":"d1addb12403afea87a1603121396261a45190886c486c88e1a5d456be17c2049","impliedFormat":1},{"version":"1e50bda67542964dbb2cfb21809f9976be97b2f79a4b6f8124463d42c95a704c","impliedFormat":1},{"version":"ea4b5d319625203a5a96897b057fddf6017d0f9a902c16060466fe69cc007243","impliedFormat":1},{"version":"a186fde3b1dde9642dda936e23a21cb73428340eb817e62f4442bb0fca6fa351","impliedFormat":1},{"version":"985ac70f005fb77a2bc0ed4f2c80d55919ded6a9b03d00d94aab75205b0778ec","impliedFormat":1},{"version":"ab01d8fcb89fae8eda22075153053fefac69f7d9571a389632099e7a53f1922d","impliedFormat":1},{"version":"bac0ec1f4c61abc7c54ccebb0f739acb0cdbc22b1b19c91854dc142019492961","impliedFormat":1},{"version":"566b0806f9016fa067b7fecf3951fcc295c30127e5141223393bde16ad04aa4a","impliedFormat":1},{"version":"8e801abfeda45b1b93e599750a0a8d25074d30d4cc01e3563e56c0ff70edeb68","impliedFormat":1},{"version":"902997f91b09620835afd88e292eb217fbd55d01706b82b9a014ff408f357559","impliedFormat":1},{"version":"a3727a926e697919fb59407938bd8573964b3bf543413b685996a47df5645863","impliedFormat":1},{"version":"83f36c0792d352f641a213ee547d21ea02084a148355aa26b6ef82c4f61c1280","impliedFormat":1},{"version":"dce7d69c17a438554c11bbf930dec2bee5b62184c0494d74da336daee088ab69","impliedFormat":1},{"version":"1e8f2cda9735002728017933c54ccea7ebee94b9c68a59a4aac1c9a58aa7da7d","impliedFormat":1},{"version":"e327a2b222cf9e5c93d7c1ed6468ece2e7b9d738e5da04897f1a99f49d42cca1","impliedFormat":1},{"version":"65165246b59654ec4e1501dd87927a0ef95d57359709e00e95d1154ad8443bc7","impliedFormat":1},{"version":"f1bacba19e2fa2eb26c499e36b5ab93d6764f2dba44be3816f12d2bc9ac9a35b","impliedFormat":1},{"version":"bce38da5fd851520d0cb4d1e6c3c04968cec2faa674ed321c118e97e59872edc","impliedFormat":1},{"version":"3398f46037f21fb6c33560ceca257259bd6d2ea03737179b61ea9e17cbe07455","impliedFormat":1},{"version":"6e14fc6c27cb2cb203fe1727bb3a923588f0be8c2604673ad9f879182548daca","impliedFormat":1},{"version":"12b9bcf8395d33837f301a8e6d545a24dfff80db9e32f8e8e6cf4b11671bb442","impliedFormat":1},{"version":"04295cc38689e32a4ea194c954ea6604e6afb6f1c102104f74737cb8cf744422","impliedFormat":1},{"version":"7418f434c136734b23f634e711cf44613ca4c74e63a5ae7429acaee46c7024c8","impliedFormat":1},{"version":"27d40290b7caba1c04468f2b53cf7112f247f8acdd7c20589cd7decf9f762ad0","impliedFormat":1},{"version":"2608b8b83639baf3f07316df29202eead703102f1a7e32f74a1b18cf1eee54b5","impliedFormat":1},{"version":"c93657567a39bd589effe89e863aaadbc339675fca6805ae4d97eafbcce0a05d","impliedFormat":1},{"version":"909d5db5b3b19f03dfb4a8f1d00cf41d2f679857c28775faf1f10794cbbe9db9","impliedFormat":1},{"version":"e4504bffce13574bab83ab900b843590d85a0fd38faab7eff83d84ec55de4aff","impliedFormat":1},{"version":"8ab707f3c833fc1e8a51106b8746c8bc0ce125083ea6200ad881625ae35ce11e","impliedFormat":1},{"version":"730ddc2386276ac66312edbcc60853fedbb1608a99cb0b1ff82ebf26911dba1f","impliedFormat":1},{"version":"c1b3fa201aa037110c43c05ea97800eb66fea3f2ecc5f07c6fd47f2b6b5b21d2","impliedFormat":1},{"version":"636b44188dc6eb326fd566085e6c1c6035b71f839d62c343c299a35888c6f0a9","impliedFormat":1},{"version":"3b2105bf9823b53c269cabb38011c5a71360c8daabc618fec03102c9514d230c","impliedFormat":1},{"version":"f96e63eb56e736304c3aef6c745b9fe93db235ddd1fec10b45319c479de1a432","impliedFormat":1},{"version":"acb4f3cee79f38ceba975e7ee3114eb5cd96ccc02742b0a4c7478b4619f87cd6","impliedFormat":1},{"version":"cfc85d17c1493b6217bad9052a8edc332d1fde81a919228edab33c14aa762939","impliedFormat":1},{"version":"eebda441c4486c26de7a8a7343ebbc361d2b0109abff34c2471e45e34a93020a","impliedFormat":1},{"version":"727b4b8eb62dd98fa4e3a0937172c1a0041eb715b9071c3de96dad597deddcab","impliedFormat":1},{"version":"708e2a347a1b9868ccdb48f3e43647c6eccec47b8591b220afcafc9e7eeb3784","impliedFormat":1},{"version":"6bb598e2d45a170f302f113a5b68e518c8d7661ae3b59baf076be9120afa4813","impliedFormat":1},{"version":"c28e058db8fed2c81d324546f53d2a7aaefff380cbe70f924276dbad89acd7d1","impliedFormat":1},{"version":"89d029475445d677c18cf9a8c75751325616d353925681385da49aeef9260ab7","impliedFormat":1},{"version":"826a98cb79deab45ccc4e5a8b90fa64510b2169781a7cbb83c4a0a8867f4cc58","impliedFormat":1},{"version":"618189f94a473b7fdc5cb5ba8b94d146a0d58834cd77cd24d56995f41643ccd5","impliedFormat":1},{"version":"1645dc6f3dd9a3af97eb5a6a4c794f5b1404cab015832eba67e3882a8198ec27","impliedFormat":1},{"version":"b5267af8d0a1e00092cceed845f69f5c44264cb770befc57d48dcf6a098cb731","impliedFormat":1},{"version":"91b0965538a5eaafa8c09cf9f62b46d6125aa1b3c0e0629dce871f5f41413f90","impliedFormat":1},{"version":"2978e33a00b4b5fb98337c5e473ab7337030b2f69d1480eccef0290814af0d51","impliedFormat":1},{"version":"ba71e9777cb5460e3278f0934fd6354041cb25853feca542312807ce1f18e611","impliedFormat":1},{"version":"608dbaf8c8bb64f4024013e73d7107c16dba4664999a8c6e58f3e71545e48f66","impliedFormat":1},{"version":"61937cefd7f4d6fa76013d33d5a3c5f9b0fc382e90da34790764a0d17d6277fb","impliedFormat":1},{"version":"af7db74826f455bfef6a55a188eb6659fd85fdc16f720a89a515c48724ee4c42","impliedFormat":1},{"version":"d6ce98a960f1b99a72de771fb0ba773cb202c656b8483f22d47d01d68f59ea86","impliedFormat":1},{"version":"2a47dc4a362214f31689870f809c7d62024afb4297a37b22cb86f679c4d04088","impliedFormat":1},{"version":"42d907ac511459d7c4828ee4f3f81cc331a08dc98d7b3cb98e3ff5797c095d2e","impliedFormat":1},{"version":"63d010bff70619e0cdf7900e954a7e188d3175461182f887b869c312a77ecfbd","impliedFormat":1},{"version":"1452816d619e636de512ca98546aafb9a48382d570af1473f0432a9178c4b1ff","impliedFormat":1},{"version":"9e3e3932fe16b9288ec8c948048aef4edf1295b09a5412630d63f4a42265370e","impliedFormat":1},{"version":"8bdba132259883bac06056f7bacd29a4dcf07e3f14ce89edb022fe9b78dcf9b3","impliedFormat":1},{"version":"5a5406107d9949d83e1225273bcee1f559bb5588942907d923165d83251a0e37","impliedFormat":1},{"version":"ca0ca4ca5ad4772161ee2a99741d616fea780d777549ba9f05f4a24493ab44e1","impliedFormat":1},{"version":"e7ee7be996db0d7cce41a85e4cae3a5fc86cf26501ad94e0a20f8b6c1c55b2d4","impliedFormat":1},{"version":"72263ae386d6a49392a03bde2f88660625da1eca5df8d95120d8ccf507483d20","impliedFormat":1},{"version":"b498375d015f01585269588b6221008aae6f0c0dc53ead8796ace64bdfcf62ea","impliedFormat":1},{"version":"c37aa3657fa4d1e7d22565ae609b1370c6b92bafb8c92b914403d45f0e610ddc","impliedFormat":1},{"version":"34534c0ead52cc753bdfdd486430ef67f615ace54a4c0e5a3652b4116af84d6d","impliedFormat":1},{"version":"a848339c272ab23e686b5d0b81297e3a7116ba7d27589c66ca1f4ebcd65e7f19","impliedFormat":1},{"version":"566315d39e476ca9e7fd0b1908074cb2a5ff9246cc3ed7da64cde5ad30f7e0b1","impliedFormat":1},{"version":"5f7ed82e82aa5017fe830ac40105321c32e3bf45d3cfcd557601f2ed58ff6ca4","impliedFormat":1},{"version":"9c8125fc43f5fc74a40240438d849d56ec7e5eb68961ce8af70a930ffb0580b3","impliedFormat":1},{"version":"d8d07d4c2cb69335afe919f64e519bd3972d8265ba1a073e4e7a2f1a0ddbe2af","impliedFormat":1},{"version":"fd3d0e2bc2829d94b6ea717f0217cc1fbe7f7e5c3e6dc20554d8682d3850ad72","impliedFormat":1},{"version":"e71863e8db54c3584405caa0331efbf08ab6db455b192e95ceb44a2905eb9124","impliedFormat":1},{"version":"a229c67e3306551dbd0310b21712247ffed4e881c7a834a19d62a149c8cbd3d1","impliedFormat":1},{"version":"83b5f5f5bdbf7f37b8ffc003abf6afee35a318871c990ad4d69d822f38d77840","impliedFormat":1},{"version":"634e56b085407249a5c67e6520fd7de77060f28c61c901e2e4d23784c204596f","impliedFormat":1},{"version":"a9caafa49a3cff37544feff78ea8a6d3bb49f7adce053a6eef7c8ed4ea87928d","impliedFormat":1},{"version":"b86ef9f4a38b5f28352d7b2a9f9a64eb0097cb01dadb9f6e57843b143c2e04fe","impliedFormat":1},{"version":"23ad184b6ec52e8c1eeee56ffb3ee922481330716025ab133fe1f0425bddcd78","impliedFormat":1},{"version":"d0bdbe18ff5e0ce26b0208dac7fb18f989547ae67661fab18eaf7d9c2a43eead","impliedFormat":1},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":1},{"version":"6ac5de574ea90ae410c8aa8528dd4e6ae38b6abd2161f4efd9b6571212c14d4d","impliedFormat":1},{"version":"da6a8af46de75441eb9eec84b82cb7a2a8fe7385ffdf29dff17a8f92e4e50b0b","impliedFormat":1},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":1},{"version":"50635666275413d74bdcec40ebff603c3e4b36cf62376c8781911951d55e1f01","impliedFormat":1},{"version":"fd77d9bad26c739ff2d8e9535f2bf2773bc340eb2e70c76a8d59e1b10d6543be","impliedFormat":1},{"version":"e6a42337746abfde6cf108ff93c5e5144f105fa8dc8f4bd6eb4b9f412c49d234","impliedFormat":1},{"version":"ab4dbe62366e31f41e3337c829f806c395a028aad37669ad60e45c58e34f8dc6","impliedFormat":1},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":1},{"version":"33942d18807cbf38c440d00701b9c83adf4bf5d66ba5dddbf01843384a24046f","impliedFormat":1},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":1},{"version":"267725112f5f9cf7235a8c1b1f995ad446e287dd50eeb088e6b6c4ae184aab92","impliedFormat":1},{"version":"110e37c80a94ae1f0bf01a12985d12fa9ea08e4f5226a5bfd7bb702e94da721f","impliedFormat":1},{"version":"103658c98e0af6a41556f845bb35886db9c95f9e40b6c345f476a85afc72add3","impliedFormat":1},{"version":"039f3db1416b09d58d86ffd54cbc08f975cbfc75e3857b5634ac461ca2123609","impliedFormat":1},{"version":"c7f6351ac45abfc84332fd2255e4fc9f40ab81be67418f95653c5b635f06489c","impliedFormat":1},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":1},{"version":"5db7bf2b215e4ad7db9356f7fefaf579ebc3669393c81addc94c815f7c56883a","impliedFormat":1},{"version":"1c465dcd7e295ca87621cfc722410abc34d2fb38133cc4d39a88182e7c1776f4","impliedFormat":1},{"version":"1ff6207c7c85da59a11b2a1ef4cfa88347b52f117faa4bdbd6e6bdb60d634719","impliedFormat":1},{"version":"89fff07e4f4a77d0973bde15c5652dcc6c2a5e67e3460a3ab14b520e8ee0dfec","impliedFormat":1},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":1},{"version":"ac2b859d346b9c79548810c0b5821b05a6a766db90bed7416f7ec0cc6bbbd3bc","impliedFormat":1},{"version":"68408a0a4000e2d3da6984c995252646d3ce12a0d593e97c12b7f4fd0ee22c86","impliedFormat":1},{"version":"4e0576acedaa905e811e3df0cc488d8e18c14925f14ab2a8e53935a4c0e4cd03","impliedFormat":1},{"version":"0459d2cac302ab51fa7f27504cc70bb0ecf36eef8cf23221effd4e16946ed89a","impliedFormat":1},{"version":"6d861e053bfac8f22f7612f6dbf0e1f4e8b2687fe93bda154b8dc34c643c26ad","impliedFormat":1},{"version":"eab879e68089c36bb373977a6e9338fa19a607f5581d30f2e5252d9333590922","impliedFormat":1},{"version":"cf06fe0d1db6452d4823c347e22edaf92d719d51d17faeb6a5c17f539b4a2dfa","impliedFormat":1},{"version":"ef0644180b71552749ee1bfa27286360d31a73de7c1fd9e43cbb70a04e2199db","impliedFormat":1},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":1},{"version":"74b321f970095426033b06dd2a8bf029b09688ec3747196d9455c526d0327de9","impliedFormat":1},{"version":"b8bbadecf2b1ca66f8ab691aed9910b37b3d3532ac3de360ea0141630d7701a2","impliedFormat":1},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":1},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":1},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":1},{"version":"faf4a3ee383cc6bb81207d4f8730e6d90ac38a010ded7583e4ba1bab1cf57b5e","impliedFormat":1},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":1},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":1},{"version":"4f52c5d04464feecaf4e55db0a0cc42d38b84a502afb54082ed6c2c8352c90d5","impliedFormat":1},{"version":"3a2a7e7343d20346af5b944a8d39d1756809c86f05bd95c4f62d53fb27a14a73","impliedFormat":1},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":1},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":1},{"version":"f22b102137264aa6bbe59e60da29993215477fbdf063109e36448c31c3ce369e","impliedFormat":1},{"version":"13137e2629cc5684fdb9dff20ef99674ac13609fb7bf0d52b6d1ee7bfb743e5a","impliedFormat":1},{"version":"f02a49f495484978dbc62b9dd7bc63904a4853754b789876f4f107aef19585ee","impliedFormat":1},{"version":"4e0b3f6636723fbfb0d0b2587aae5135b4d142aceaa185adbbf8260313e5526a","impliedFormat":1},{"version":"e9fceb52882eddb54d6010a966c1ff3b3056d3cc652ab020d32be7022030fbee","impliedFormat":1},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":1},{"version":"9acc62b08ad2fa64f71f557b849430ee8c96300af72448b86104ad5e3eca8bb0","impliedFormat":1},{"version":"56bac357cefcfd19e72e66bd6984bb39adeef3d513f6c5f396d97040b5a5dd4b","impliedFormat":1},{"version":"f81515e6391f020a4b8f346a2af47a48a85a12491feba9166ce71c692e45441f","impliedFormat":1},{"version":"702caf4b27b5454a3305f20cea27aaae0c5673b91df4936a8559f3112e4d68b2","impliedFormat":1},{"version":"0456f6abe968e44aa231527842e90fc493ccf0086c044685cb66fc9d307d5767","impliedFormat":1},{"version":"14b62d4b47d0c2c832ded6390e33303be8be08b21c1ae0d478a46f0f49854e78","impliedFormat":1},{"version":"5881743bcc06d8169dfbbb238422a9d75d3930b09b59e099d0aa4ebdfee7dc0c","impliedFormat":1},{"version":"d962ff332884aa5af93c4601189c35747b6724765a3cd697242b5ef1e02cef70","impliedFormat":1},{"version":"e5af1a573638d8532157d7c69bffe9aa3551ae84db39d02d255efbce2207b342","impliedFormat":1},{"version":"e3d196421e621fa84174dd79502e01d2e00d67e23362a8c530f7e05cd68e8ea1","impliedFormat":1},{"version":"f5e8dd756948f1c077b3ecccbdc1f95aa5a5edf4f58dd079667d4611814666e0","impliedFormat":1},{"version":"214cbcbd70d482acbe40ed45aaa8383e98c86a86706afa85cdddc9719ac548ab","impliedFormat":1},{"version":"73f84a43613929bfe3efdbc61d2dc1ae39e5a32c35795f7806cf0a60c83e60a0","impliedFormat":1},{"version":"6957a2d31554536d37e96402c117b2429f2e9baee89f26b87caace936ca2ac37","impliedFormat":1},{"version":"bad9e7bf88505357ad4e64ec0a87b7abfdc783fbba6d3c257d2eee2493703304","impliedFormat":1},{"version":"c90c20f613309279aa05bcb314e75d762538bdb1e5bb1ace75d1c1ef2a979637","impliedFormat":1},{"version":"923b19f9e0d134113ed5b15f48a046db1afbab4e34abad9993ba873b9e18dc7e","impliedFormat":1},{"version":"3c4ab379d2e80517f92e24479d0161f58fab9ec7b2b508d2f243ca765aca0050","impliedFormat":1},{"version":"e18dd77323af9b0e4f7b8a4de60688b08c157814a59383dda5b9dacce2230f46","impliedFormat":1},{"version":"ae76106be2fe3281cd7e96b9dc9e12b4583e61e31bae624656ec0feeaf75371f","impliedFormat":1},{"version":"afd70a57b376a4e926abd4c1c8e9310fe96c969d5a0197ffcb565d001676a9f6","impliedFormat":1},{"version":"d7538da5cadf8bd654a7725666b4382a9ae6f9aed039098a36ee878ca6a3bec8","impliedFormat":1},{"version":"8d48652a8cb3ab8370fb264ba855d9f5f232553a3d9f5bd25a88b290e3e23c10","impliedFormat":1},{"version":"6cd4ec90f2ed6e15bd4e940d5ca7aa38cd570d1cb1d0b9952624e5c8ddb4dba4","impliedFormat":1},{"version":"a7f1293a7400026dc420559629b54c8493343200ff36d92a8d78502a9282a35b","impliedFormat":1},{"version":"d2f3f85583a57ad1987ee9f6b8b174499e9c5d7115e37dc9a62a2dcb9b054d1e","impliedFormat":1},{"version":"7ff52e014e5f1f653a7c3d0f5f323e8e7daf5db30abfee1380b78dada4ec2da9","impliedFormat":1},{"version":"be6b8d983bd7d37162db454ea038196367a799156c3caa33927494bc99d885d0","impliedFormat":1},{"version":"b343cbbbeae17e5c0ac05bab9dd4e08c57a1559cd31659d7e152bd122ae646d7","impliedFormat":1},{"version":"42c67685ee5027789a51538b046b3a7a11a2b19705ebc63ce3f0404b8e9fb0f9","impliedFormat":1},{"version":"693104e41fb5dc31f325c518cce5eef5513d91a650148fcdecc064d137f8581d","impliedFormat":1},{"version":"c54d981103b6a51e2e7f52821795ea2f8a2e08093cbeeec3016613697df11d87","impliedFormat":1},{"version":"9f1a99a5145d55e9543b58d51eef81ed14318575355f554c76c97ec043d31131","impliedFormat":1},{"version":"2829c78953b67be242428630b35cbae50af7dd9c9b24ef5c467986a1f14d94cb","impliedFormat":1},{"version":"da3eab33856ccf1f35e8e9ded34994f2b4a23422f1e0e99f38805f66d4231a3b","impliedFormat":1},{"version":"43276dfec18eb7175615c6327a4ee01a116de68e37cebb56da1dd742225d3ac9","impliedFormat":1},{"version":"bc6e29688d6a2cae05887ddbb04aca69aff1e5102ed1074671445bcca1c881d3","impliedFormat":1},{"version":"51718633ad06a6d05d68a9ac009d49e97b84d980ed06b1cd04f0b40398310d43","impliedFormat":1},{"version":"ff90a6bc7812f609903f41b98c60f3edc2d593483fdeb9bed20cb93e6527a777","impliedFormat":1},{"version":"7da854941074e76cd1ed6f23c7ae615e931589f9cd3ef919ce832f47d666ab6d","impliedFormat":1},{"version":"2c6dafeffbb2afc2c681099fea24af5b76c43553d40867e25efe097ed4c78965","impliedFormat":1},{"version":"64135cfd2a693174828c8e842198f5e6854e6862df1ea685d62bc1a20fde9006","impliedFormat":1},{"version":"8c525341425df5d0863a79895b37ec46d306f98d856f6094b52c87c55a77fd31","impliedFormat":1},{"version":"dc8a332007798357766fb7131b180bbcd917b5cd98916d827d9a05edb9260e0b","impliedFormat":1},{"version":"831444604ca9fbb1f144fb399b92e3de5ce9d6c4c651f061fa5e34f72e29197f","impliedFormat":1},{"version":"f5831fcfbcf7f09591af1e5dec357cacf57b7e1259267a4ae5769b7f83f8a441","impliedFormat":1},{"version":"aae56a55145c92171dbe7280fd6c0ae4c286b2933b4b0ea56d398f6abd82f454","impliedFormat":1},{"version":"2b8d26d51897913d32cca6dfcbf2c509e35f77415e50a93466d560cf42ef703e","impliedFormat":1},{"version":"4fb248f0a9fed6d8658e6bdd6c1422b1a7fd9b27cf30bd3b1a5a26fe4d7d8963","impliedFormat":1},{"version":"d51176c3c6362a0f9a59184c71f3b8d8471b6a6a4060258c4264722fc5a11463","impliedFormat":1},{"version":"a2e88c1cb313192e2e5142e8898dd35b39a4f30d272cb07577787510df606bda","impliedFormat":1},{"version":"32b457a43b19f02c0fa6b92ed3171e2052cdd0eb2819fddb60b7324d4bc3b406","impliedFormat":1},{"version":"e172920ce3b5f5d4ba43ae4a4a2c6a61ea5960f267e5d25cc84dc12527005f6b","impliedFormat":1},{"version":"0e8785bc79cbfc14a2c4a001e272ff0ec909ec94564705e85664db9492265e1b","impliedFormat":1},{"version":"20c8eca485f3f73c9d5855a1c99029f2907846b88d0ec81dcc11d6abc20f5653","impliedFormat":1},{"version":"25c8897df13b2f74c1c3e68c3e8d1f22bd7adadbb0ffa6e48e14e09045694ff5","impliedFormat":1},{"version":"253db8a1162220c88e504d2e31af9a9afe802a498a8b4920ae5b8751bbbc7bbb","impliedFormat":1},{"version":"df35bc4ff5f2fa4cddd5d499477c595ea76644bd03150922e0c20184ce1f76ec","impliedFormat":1},{"version":"d8c33684d5af091b42e5e4fac2654ae0e4fb707ecd56d2b5ea954f1754dbff36","impliedFormat":1},{"version":"07d5c61850d955ff344ccacc4c35a1cc1b534ef92201da4d555e3cae26ca994c","impliedFormat":1},{"version":"19ba4067fc331691fc5af2aff7dfcf39a0b6d50b5bda255e3c6682b32983e5d5","impliedFormat":1},{"version":"a22fb21723983b4e2edf3d34893256c8b6075f77254f394048541f5a4eb25d15","impliedFormat":1},{"version":"969948f990cbb4f0b594d8b3e66bc37d04f4896314afb888e507ae0fb9aaac51","impliedFormat":1},{"version":"8b9782193fd21acd035ca67a18e607ca68e8345d5931962ff5862d89fae1965e","impliedFormat":1},{"version":"107c2243004cd47d8a63b15b42644343db310383b8008237f7563710116589e2","impliedFormat":1},{"version":"9a3a28ed970a073f6f87f9827839c2d06ecdd05f45e07ce30899f72ca968b46a","impliedFormat":1},{"version":"08107d403a7a4235fd239bd1185800d10f646ea07a71b119c2252713d466920e","impliedFormat":1},{"version":"175707c3c7618f8e3ea64636dc591ed6892328fa430149d3ad414018751da8f6","impliedFormat":1},{"version":"4b086cd2bf1f7fdac4fbbe9acb863b29040fd8ac4188c5d7a5b3c95bafa1b380","impliedFormat":1},{"version":"2a7ef8d34c40308dc2a2b05a78b8ee602f205e82e4eac3f44f1959e95bece679","impliedFormat":1},{"version":"022d05125afe3135d923892f13d1b176003560edd270900f52957a07e1efeab2","impliedFormat":1},{"version":"e0fa1fa96fdf10e88c8a23aa4eb2566232ac5f8d93961815158a7c6b22d7efaa","impliedFormat":1},{"version":"0a6a304a71bc56611b60ad013e583564b6056b8265961123d77fd65fd8b74061","impliedFormat":1},{"version":"63bba6da188f796caf21284a73dab06f85bd17042bd5ad49c0ec81451fdb0f5a","impliedFormat":1},{"version":"5f2a79b58c58371b68d6f3a3a225e0804c6ce517c423c8a1efec234765de7586","impliedFormat":1},{"version":"358a84f9e1f6680ffddb329a580be5f932c7ff10ced8d60f43904f66dddebaa3","impliedFormat":1},{"version":"b60efbac98231283107121b5b3327f56a6632c2d14d7616920bc309a4f6d4bc3","impliedFormat":1},{"version":"0e58e6f3fa554921c7950ff344d6c299caf9260e4c78bf7c699a6b5a6d02e6bc","impliedFormat":1},{"version":"3eb80f1addaca2a298abd6186a2cfe98567d88635f334a0f2415438ec5f2d3b7","impliedFormat":1},{"version":"8d5af927098c40f41f315552e7ff9ee9376b26fc03e83421a0cf2d93907ea346","impliedFormat":1},{"version":"c993b44ec48e09bf9e9b512941379656f9090ddf81f7ab61b4d8a6cdbe7ec409","impliedFormat":1},{"version":"54f323b0c25677fcd7dbb6541667f131e17bf49d48b6efdfed72ae71722fe8f5","impliedFormat":1},{"version":"7668c31fc1a0d6c5ee0ae1048d41f7232a56fbe18367929f78bd0c77303af11e","impliedFormat":1},{"version":"8b41773894ca3ba064857d72a6cbd669b299e47915c3b97cbc2a613735cbf35b","impliedFormat":1},{"version":"badddb55fb1a8186abb7d4b972820f9e5763916e59e9567a847d0237ba0f72d7","impliedFormat":1},{"version":"74689440172e6a26a40b93a21ca3f263e2d06bada97b270a884737f921e7818f","impliedFormat":1},{"version":"9c3ded425a22114083d56daa858d27b46bc1b059aeb023f504574312ab1439ac","impliedFormat":1},{"version":"08f50b290537a8bea3a96920b5d5664d4cd23472161af28c8bcdc5091250c3ce","impliedFormat":1},{"version":"c4d0d823f114af573cdd62f5724648cb9df7a7ca1f8473ebe65b7d7df1789422","impliedFormat":1},{"version":"e16aa5f3e598ad86a044934071f16729c0f95fd77794f0ada7a88faa2f66c185","impliedFormat":1},{"version":"098148c34c5cef91a12c622fadf8d19a7f513206d3dc61fc31af13fb361d99e9","impliedFormat":1},{"version":"4130eea8635f6d6bc82a8a9560b8064c163b1029d3efa39815fb53c4aa51c145","impliedFormat":1},{"version":"f1c957e436f37c6bd81fd6bc6a13eb1bf7a9ad5f297a167db0e96415f621ed66","impliedFormat":1},{"version":"98144631dc436418a7b927607618136353a32f4ccc420b76358a730310bbcc8a","impliedFormat":1},{"version":"026447d4bf29241ac992589ec620a86b13c76bdfcb1ff8dcc7e26f0eb2d0d210","impliedFormat":1},{"version":"12f79c131043198b4d0f789df3cc4b90d5cc00dc0c64afbe9e6965f4a55b3d61","impliedFormat":1},{"version":"fa890a742e523ead1ac2d8738c29c843d2a1acaa98da02a7667fe00d177aa196","impliedFormat":1},{"version":"b99faf232d2c47ddfdfa086a4bb0665bcb25e3a3989498d467caaa79200afb06","impliedFormat":1},{"version":"21e4a665ab9901d7a9f42aa585fc3bfba8ef4d090640a1e412669a0bb392edb2","impliedFormat":1},{"version":"e7d94b0ae7d41c1bd5f1aa4c2bd62676af83e1fe743316bf82bb32ec1be11421","impliedFormat":1},{"version":"aaf88ec377baa9cf35177eab96b5db57bcfdc5bbe34bf38b1805d883f6b2cfa4","impliedFormat":1},{"version":"d4b211bb230daef2a02fb8952c1b21730d4d14d70baba4f04c5efce000205ea7","impliedFormat":1},{"version":"8eeb941ef7939f9f0180fafe779c7fa9e1049b5716a654fc25463fbf472d3dc9","impliedFormat":1},{"version":"80e2b75b44778105663dee124d241ba133250df92d3b5760784ef9683c622c1f","impliedFormat":1},{"version":"e23514abb70d5803377e5367af5a9554b15529d97b658930335b195f9d5753b2","impliedFormat":1},{"version":"b5af0716932f268f2a4a41420d7ba9fdbc037e1bb406aa57caa7616b173422c6","impliedFormat":1},{"version":"af67cf7922d64c7e1cc0a0c327191d97ef6e1d54f7f1661a06e7225fa8b35e48","impliedFormat":1},{"version":"67ae5eaf9ef6ed32a30aced05943e9f83df215d62f80076f7cce3a55d08c8722","impliedFormat":1},{"version":"8bf4808d0cbdfee342649aaa6744ccdb7f3b98c127985024474f961e3a96d038","impliedFormat":1},{"version":"27e56c281e88ef3107c9ce67f02bdcfba297804d3d14006a3e3d59f45a3f1d9a","impliedFormat":1},{"version":"42d00c41e9cffbb3cfbad77417055030f952fe8d7dbd8f646fd0005153b6e821","impliedFormat":1},{"version":"ecebc4355edf1384d191afa1c0c4ccacadb199ab55c90c9c450720425e975fc5","impliedFormat":1},{"version":"77ff7b7d3bef88309b2c6b48e2fcdb7db8000b57f7f627b9481b014ef2db7581","impliedFormat":1},{"version":"b8d5fc4baf94f4aaf437c2505b751083c58983a126fa712d34ac5e4e7d064ee1","impliedFormat":1},{"version":"8f3a98972a1f230e69a9c11e2b78ead1761bcba0e6cd7ba029e1e57cb5f89eb8","impliedFormat":1},{"version":"f681b47b5e0275d8a2fe219e40d2c80fdac5c6f865af6fc61df0f15c59c6c9ee","impliedFormat":1},{"version":"17bec14562208b93665ecee566ecb99baf6ca82eeb92ab1eb9e3442aafb26a99","impliedFormat":1},{"version":"fb00be4532eaf1800d8a2a625a8843f5d8f122990d2bedd72ebeb90a555f8cd8","impliedFormat":1},{"version":"374ddae0cfbf334836cfbaf71ec0ab9c16e677306d31f6e843e428119a90dce7","impliedFormat":1},{"version":"688e6406967d02af975bd78a3015d9ea0d1d3bad93d62df0329bab69cd278f97","impliedFormat":1},{"version":"d8fd376b0555bd256ee497d88cfad88d6edce66b0136c57ac4e06c2c1226b48f","impliedFormat":1},{"version":"c9df649292e24f2cf451d81faaf1a5dedde2fa1bf043b3f74a647506cdc83ef4","impliedFormat":1},{"version":"0fe0ef68e8dd9341ac44174b965cad70c377d078c86085b92123ca8f1a6c5c56","impliedFormat":1},{"version":"75543c4a7a4755138a91f7260d663c393b700d3ed5fec65538851286f427c234","impliedFormat":1},{"version":"98d24457c0cc7837ec1e984430abb00a1b3556cb84134c00712cc08a6376b2a5","impliedFormat":1},{"version":"69fad132b41289d40e3d373ad0ef1065cbae26c78f11d2f1aa574a3eb9b7cb7e","impliedFormat":1},{"version":"eb2d716abdab98d94a6be94686ee0ed7506ba70dde267cbd6c24a1e94cfbda60","impliedFormat":1},{"version":"58d9aa65560fbd9932063c2be022c9346ab7c79ccd10522c61b60cc112ac4341","impliedFormat":1},{"version":"c6f19eb7c75210f37cbe83efae91b7df76a7999dc40fd7b79c7363b67a8790f5","impliedFormat":1},{"version":"68ad94cf9d4b8f9affd8682dce00daf93cd8c5e3072237c22dc6138b997e0bb6","impliedFormat":1},{"version":"42c93fb5d5b061f7f95a3ea59c2f2b31256939e97f5183c827234d32160901ea","impliedFormat":1},{"version":"0c27495545631e99af91cebdb55d81320cb7785e05e86a3745fac2949564a12f","impliedFormat":1},{"version":"248127a1de2be832dd6b9a4c2589ea241f3caa9bde05ee788dfa2128c6ee3218","impliedFormat":1},{"version":"b5929098539f514464ac3c468bc3c93ea40996de9eb5e770ecaefcfbb2677758","impliedFormat":1},{"version":"b9a6aadc8283344ec53b55c8774fd92ddb473b3702dc846afaaee80c49b50131","impliedFormat":1},{"version":"2f5685827f73fd674138b81ca48275458be188b01af8a00315231efe74397980","impliedFormat":1},{"version":"6097f1055816adfa4bfc72e65ee4b6abd46d1fbd65bff4c7c7180a031408321c","impliedFormat":1},{"version":"053b26604faa65cae78a2981e04dd917fb068fd367ac39203f02d64ed0ff498e","impliedFormat":1},{"version":"cd1876d05dc46e55c625362f53068c9d4e7307c7e74a590b2975c657a1d3b625","impliedFormat":1},{"version":"d0d95a2893c600f181619be0c35d30845d93ae669e0aee435f9c95f6237c6872","impliedFormat":1},{"version":"1d11106d40ce32966017ecc516704eef10e57f72e376b1ae3d66c9f9cc6dc14b","impliedFormat":1},{"version":"249470a524039cb8a01af509318599f57b0808a602617eb7a655e2ec584446a0","impliedFormat":1},{"version":"07c06586fbe383765f7a404c9ef9b96ce61aeb6849f375ba5fad13139051f326","impliedFormat":1},{"version":"a54442465e1152334910e6533d10e92fbb8b2d36ab0aeadcaeb3f3f151a8a825","impliedFormat":1},{"version":"49f3ec14c8cc3e911e40adc42a42f40c109d8a62c655cf905d15e235883b369c","impliedFormat":1},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":1},{"version":"6ac5de574ea90ae410c8aa8528dd4e6ae38b6abd2161f4efd9b6571212c14d4d","impliedFormat":1},{"version":"5e1dff61bf656dc543938e285ba67f84ba0416959a7c10792cdf19ae3b50672e","impliedFormat":1},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":1},{"version":"171c708e6354a6bea00590af5b2e4c67ac5b6db17f5c6610074aa5f89df16e48","impliedFormat":1},{"version":"e742691b51bb384ebe58d525ed1b2029521a52dc63e35d445208a1efffb4089b","impliedFormat":1},{"version":"e6a42337746abfde6cf108ff93c5e5144f105fa8dc8f4bd6eb4b9f412c49d234","impliedFormat":1},{"version":"ce8044e0d8a197824255a3738253d7045666f9ac7ede8ee9ce79da095e4aa99c","impliedFormat":1},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":1},{"version":"89c5aa237d783c2938e56bf33611ec6b54c1df76c62d474c2475b6120f185a7f","impliedFormat":1},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":1},{"version":"267725112f5f9cf7235a8c1b1f995ad446e287dd50eeb088e6b6c4ae184aab92","impliedFormat":1},{"version":"110e37c80a94ae1f0bf01a12985d12fa9ea08e4f5226a5bfd7bb702e94da721f","impliedFormat":1},{"version":"103658c98e0af6a41556f845bb35886db9c95f9e40b6c345f476a85afc72add3","impliedFormat":1},{"version":"62f0cde62b87e93aa0ca1a00e24eb7fb4196d2eb8d69fe27064c96ae92ae3527","impliedFormat":1},{"version":"2a527df5c4828328fa6b35cf8b8f5bf0640933a4602c517faace7a1c3af0d446","impliedFormat":1},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":1},{"version":"5db7bf2b215e4ad7db9356f7fefaf579ebc3669393c81addc94c815f7c56883a","impliedFormat":1},{"version":"f6694bea88421c6d7342b5381b1a49fc823ae746680ca9ee16d518c9c16118e8","impliedFormat":1},{"version":"b8a25d32e4a2a187e2169f0936416cfcac8926f56166f3895fb5f82942f3150e","impliedFormat":1},{"version":"89fff07e4f4a77d0973bde15c5652dcc6c2a5e67e3460a3ab14b520e8ee0dfec","impliedFormat":1},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":1},{"version":"dec52a42c912503c35463f974fb86cb1a772cab001c2c9ed413093845be2f677","impliedFormat":1},{"version":"d2a2a7be324ab271073676edb22f5de259d4baf5bad32bd2e5545f957f503ac4","impliedFormat":1},{"version":"4e0576acedaa905e811e3df0cc488d8e18c14925f14ab2a8e53935a4c0e4cd03","impliedFormat":1},{"version":"22f897e17f18b702f8aa1c6e6412fcd33d180f8ef61297fec6c395a2b18d9908","impliedFormat":1},{"version":"9b48fb7d6521c10569a09921fea776719fab153e4b24d6bf4290fe6fab9be6d3","impliedFormat":1},{"version":"1850f2a9f8d382ef6f4cfea284a6994a1a636095d24f610ddbcd0dd74eef387a","impliedFormat":1},{"version":"ef0644180b71552749ee1bfa27286360d31a73de7c1fd9e43cbb70a04e2199db","impliedFormat":1},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":1},{"version":"608389d7f7df2c187768c89d6c4edc5ea43b1720203b4d212b85233ddb951ae3","impliedFormat":1},{"version":"2873b8fe4083b54fb60dd1d03ee8b22496e41f96a4e536e06cd59a481aba01de","impliedFormat":1},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":1},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":1},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":1},{"version":"e0f1bf295d165e3e7fdb6bbd9910888e9c5645e19cb4ae4b86303ee5ba2c951d","impliedFormat":1},{"version":"59879794c2339dd47bec19378cbd9c53cf7f6f851ccba99eed6eb1e0bd3d9519","impliedFormat":1},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":1},{"version":"2d99e3afe124a3c40300762492a49425df4b8090f771cac8034e233eed31bdb8","impliedFormat":1},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":1},{"version":"88f162f40062f4d9db248fed81d8d9258b2d0846ab8640904e220d69e4a040b8","impliedFormat":1},{"version":"df35eb1e5ccd6b597d18655f69dbbe24e8cca39ffe13822158c9756c528faacd","impliedFormat":1},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":1},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":1},{"version":"73b9a46d4b4135cf3e6d2c298027c1df258e4cbdc6ab9774b957ca5e207e6271","impliedFormat":1},{"version":"13137e2629cc5684fdb9dff20ef99674ac13609fb7bf0d52b6d1ee7bfb743e5a","impliedFormat":1},{"version":"7f7ff540a5107b5ff4b47191cd6e16a1be6d1ece99fc18b40a791dac9e3fb369","impliedFormat":1},{"version":"4e0b3f6636723fbfb0d0b2587aae5135b4d142aceaa185adbbf8260313e5526a","impliedFormat":1},{"version":"13060c51147d2e618fd5def7eacffb38fe536f2fd5ce23cdd0642f7e1e29202d","impliedFormat":1},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":1},{"version":"503b2da756cf3ab36e9708b43379353ca1c6aeeadfde37d757923b81a97f93e3","impliedFormat":1},{"version":"91ec9831ad545c826a697fa50de41d6fff9d927b505aa66ed46d4a9263a9ce0b","impliedFormat":1},{"version":"ee76bd42d21021012bd1f11c216c7e710ff11ebf8bf7ad134ebe1ed484ba85df","impliedFormat":1},{"version":"bd9d4c79b48d0f7ed5d5d3f7d06c029599023546fbe37fa33a95b525bb6febd7","impliedFormat":1},{"version":"e3abd62334ac65b6ae754c0950e014c8869c10e280da768c9c66aeafa1fcd9fe","impliedFormat":1},{"version":"fbb2889b4f092db88dc16994888ca046365ad0cd2232d1bdab0ae23db74a8a97","impliedFormat":1},{"version":"b32293aada046eb0690590b723571aba0c34e57566f1067d837277ac5ea5a58f","impliedFormat":1},{"version":"9255c726528f439160ea4d10ed4a091652db99f259acc2d9edc40c467fe91070","impliedFormat":1},{"version":"f66e1aa906b1de4070ba4dc224201a81dd36a30f9ae27cd9a20f9fbce57d98a2","impliedFormat":1},{"version":"f83adc50f31f8c405bfcc1e537f511df92b683cf1c6daf2236ca1551ba576918","impliedFormat":1},{"version":"84ca3c4c8f8b4b3328afa2c57555cfb71deb39053a1b783b81a44c0d183bbe1c","impliedFormat":1},{"version":"cde540ab3aa5b98e2555a87f25316ffdaef51dc2fefa113ed96c47e17e06efee","impliedFormat":1},{"version":"6a0d02ea3868cbbecffc40e314a8cf76ae69b14102b14b007b6d43edd7d4b078","impliedFormat":1},{"version":"76ea95684460d48dbf18e249903aaca73e8b011a5a6bc6e3071e75b3cfda1dc3","impliedFormat":1},{"version":"8fe2df5f35b364703a41744b0263abab1d0c94ec4da0e97e3fb09e7f1f49d1ca","impliedFormat":1},{"version":"cd39f4dc66037ba2987de3a740b709cbb19370fd564d7e8a1082806e52fc7da1","impliedFormat":1},{"version":"f1bad28391c2827fffcd7210a91babccfe7a097ee5c33bb20566d777e4525aec","impliedFormat":1},{"version":"84827ed75fe6e06f34cff9752e36084d9a0d316372b3862d59c487f3608b9014","impliedFormat":1},{"version":"fdc17b998a4934f09616fafd7920df0c0d120e22a805c12e8170f9ac473d3ac3","impliedFormat":1},{"version":"4815c2f8effb9c688066003158a8cc0079ab50d4bfa3ce90ec43c1399ed40d51","impliedFormat":1},{"version":"6669c0c622aee9286e3b8f8f0bf70b329ee2a45cf7d98ad1b7a250a43e7e3374","impliedFormat":1},{"version":"0c27495545631e99af91cebdb55d81320cb7785e05e86a3745fac2949564a12f","impliedFormat":99},{"version":"b5929098539f514464ac3c468bc3c93ea40996de9eb5e770ecaefcfbb2677758","impliedFormat":99},{"version":"2f5685827f73fd674138b81ca48275458be188b01af8a00315231efe74397980","impliedFormat":99},{"version":"6097f1055816adfa4bfc72e65ee4b6abd46d1fbd65bff4c7c7180a031408321c","impliedFormat":99},{"version":"053b26604faa65cae78a2981e04dd917fb068fd367ac39203f02d64ed0ff498e","impliedFormat":99},{"version":"cd1876d05dc46e55c625362f53068c9d4e7307c7e74a590b2975c657a1d3b625","impliedFormat":99},{"version":"d0d95a2893c600f181619be0c35d30845d93ae669e0aee435f9c95f6237c6872","impliedFormat":99},{"version":"1d11106d40ce32966017ecc516704eef10e57f72e376b1ae3d66c9f9cc6dc14b","impliedFormat":99},{"version":"249470a524039cb8a01af509318599f57b0808a602617eb7a655e2ec584446a0","impliedFormat":99},{"version":"b0221dea05831243b8f2fc4da23079128a812a5a5a9908327725c92049c39bbc","impliedFormat":99},{"version":"a54442465e1152334910e6533d10e92fbb8b2d36ab0aeadcaeb3f3f151a8a825","impliedFormat":99},{"version":"49f3ec14c8cc3e911e40adc42a42f40c109d8a62c655cf905d15e235883b369c","impliedFormat":99},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":99},{"version":"f753928cdc4391702905204cb54c5335545c786311c5f52ed9dade3f55040faf","impliedFormat":99},{"version":"d7bb71b8da554340046cb2986dea2f6802436149fb40fa949167756f00a51f18","impliedFormat":99},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":99},{"version":"ed54678c383848e12de6b9621648908d63703be33d3d542b125dd4cceafa99b1","impliedFormat":99},{"version":"e742691b51bb384ebe58d525ed1b2029521a52dc63e35d445208a1efffb4089b","impliedFormat":99},{"version":"37dfcf681f7dfa16001120800a5559d418c84bba05f74806169a953930ca1108","impliedFormat":99},{"version":"64415fcb1c664e0a60f10696d10027d96c9810e3412af9972e6a4dc2c2e726ae","impliedFormat":99},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":99},{"version":"a81510a532b797072381fc8b72287d54595f8b2d25691f793f5d114875282b23","impliedFormat":99},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":99},{"version":"7b8f4bcf71399d7bbad22014a4eeb382841c61ad3aa079943ed287598e70485e","impliedFormat":99},{"version":"fc5115956fdfddcf86a30a1ba0cc02927cf7035a2bdc3adbc8766b79242e0eb4","impliedFormat":99},{"version":"6bc0e969085d2ad0696627de23af748de2afae059856a22fa0465036bcf2b6c9","impliedFormat":99},{"version":"dc147a0ab89bc4abf1913f699a9335e98a889f00cda6f07a5b133c5cc3112622","impliedFormat":99},{"version":"2a527df5c4828328fa6b35cf8b8f5bf0640933a4602c517faace7a1c3af0d446","impliedFormat":99},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":99},{"version":"b203573913f773b35d92a3a499a7873038149a35e0b23c7e189d7590b27f6da0","impliedFormat":99},{"version":"f6694bea88421c6d7342b5381b1a49fc823ae746680ca9ee16d518c9c16118e8","impliedFormat":99},{"version":"b8a25d32e4a2a187e2169f0936416cfcac8926f56166f3895fb5f82942f3150e","impliedFormat":99},{"version":"74f9f15dd600e9737bffdc26343d74b2d17adb91536fe4e29a9d110295136334","impliedFormat":99},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":99},{"version":"dec52a42c912503c35463f974fb86cb1a772cab001c2c9ed413093845be2f677","impliedFormat":99},{"version":"d2a2a7be324ab271073676edb22f5de259d4baf5bad32bd2e5545f957f503ac4","impliedFormat":99},{"version":"8da99e8ca9c8fced530f92f1f2faba413b961735ef92da80c577f981b767e9a6","impliedFormat":99},{"version":"22f897e17f18b702f8aa1c6e6412fcd33d180f8ef61297fec6c395a2b18d9908","impliedFormat":99},{"version":"9b48fb7d6521c10569a09921fea776719fab153e4b24d6bf4290fe6fab9be6d3","impliedFormat":99},{"version":"6908cf62ad2018d33473007b4f5f6c5f097aa0d28505e694aa7646291136dc67","impliedFormat":99},{"version":"2fac6a45f688a1be6081e321a9ca7886923ecfc3a9083959485567ffc38b1dea","impliedFormat":99},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":99},{"version":"19c7b443e13c14613f6cfe274d924597e3bea64375699b98603c40c4c4f3dfb8","impliedFormat":99},{"version":"2873b8fe4083b54fb60dd1d03ee8b22496e41f96a4e536e06cd59a481aba01de","impliedFormat":99},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":99},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":99},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":99},{"version":"e0f1bf295d165e3e7fdb6bbd9910888e9c5645e19cb4ae4b86303ee5ba2c951d","impliedFormat":99},{"version":"116418e8039e72fc6a67f90222c77ed8daa944be04eceb86bcf08e721e291ec8","impliedFormat":99},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":99},{"version":"2d99e3afe124a3c40300762492a49425df4b8090f771cac8034e233eed31bdb8","impliedFormat":99},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":99},{"version":"88f162f40062f4d9db248fed81d8d9258b2d0846ab8640904e220d69e4a040b8","impliedFormat":99},{"version":"df35eb1e5ccd6b597d18655f69dbbe24e8cca39ffe13822158c9756c528faacd","impliedFormat":99},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":99},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":99},{"version":"d366ccfb8cb87789f1592b0be7430df7ce17fca8be422033bdf0a8d0e06b7336","impliedFormat":99},{"version":"e243dd83e46a4fd3614589b4589042576f86d4748866b9423c77dee1318847c0","impliedFormat":99},{"version":"6ffba5563c43d43c1ea6b051421d59af7f6d23cc1baa9fd18a99a39d060c1cdb","impliedFormat":99},{"version":"bceb3703983ccb7177c4f8f21ed775c0ae7672559c90059a7814b04065ae04bc","impliedFormat":99},{"version":"138b012318f035855153d24cfd0a266d0aa30cef0565d56b40cb6057324ff8c7","impliedFormat":99},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":99},{"version":"7924c9999e6db6eb085f843000443a40efd7d30474fd038fff5fa0609994d766","impliedFormat":99},{"version":"91ec9831ad545c826a697fa50de41d6fff9d927b505aa66ed46d4a9263a9ce0b","impliedFormat":99},{"version":"ee76bd42d21021012bd1f11c216c7e710ff11ebf8bf7ad134ebe1ed484ba85df","impliedFormat":99},{"version":"bd9d4c79b48d0f7ed5d5d3f7d06c029599023546fbe37fa33a95b525bb6febd7","impliedFormat":99},{"version":"e3abd62334ac65b6ae754c0950e014c8869c10e280da768c9c66aeafa1fcd9fe","impliedFormat":99},{"version":"fbb2889b4f092db88dc16994888ca046365ad0cd2232d1bdab0ae23db74a8a97","impliedFormat":99},{"version":"b32293aada046eb0690590b723571aba0c34e57566f1067d837277ac5ea5a58f","impliedFormat":99},{"version":"0e3439dca47087f3f3685efd6b47efe55fba142bfba722f84dbd90101ba00f44","impliedFormat":99},{"version":"5482a5b0d560b5f1c22d894a1b30ca37b5a8b509668276d4f2699c097684c5d7","impliedFormat":99},{"version":"7163fb6c706557131bec0a259fa99ef690b9ea74e2c8e22448d53ee767dc0003","impliedFormat":99},{"version":"84ca3c4c8f8b4b3328afa2c57555cfb71deb39053a1b783b81a44c0d183bbe1c","impliedFormat":99},{"version":"cde540ab3aa5b98e2555a87f25316ffdaef51dc2fefa113ed96c47e17e06efee","impliedFormat":99},{"version":"6a0d02ea3868cbbecffc40e314a8cf76ae69b14102b14b007b6d43edd7d4b078","impliedFormat":99},{"version":"76ea95684460d48dbf18e249903aaca73e8b011a5a6bc6e3071e75b3cfda1dc3","impliedFormat":99},{"version":"8fe2df5f35b364703a41744b0263abab1d0c94ec4da0e97e3fb09e7f1f49d1ca","impliedFormat":99},{"version":"cd39f4dc66037ba2987de3a740b709cbb19370fd564d7e8a1082806e52fc7da1","impliedFormat":99},{"version":"f1bad28391c2827fffcd7210a91babccfe7a097ee5c33bb20566d777e4525aec","impliedFormat":99},{"version":"84827ed75fe6e06f34cff9752e36084d9a0d316372b3862d59c487f3608b9014","impliedFormat":99},{"version":"fdc17b998a4934f09616fafd7920df0c0d120e22a805c12e8170f9ac473d3ac3","impliedFormat":99},{"version":"a1b9afa158ba3d9e2ca524a899818a65f4b46c0b7aea5549ac161960d57e33e0","impliedFormat":99},{"version":"3525e2af342b254e263cec0aa825c7bc9fd77de4954d4cd32b0431b0e8fc4fb5","impliedFormat":1},{"version":"8a88236860c9107900f479f1f27927648652b67d6c8a3cf9d3b2e3d863e0130b","impliedFormat":1},{"version":"04123655088b1a0117eeee5a63e4c433c76bfdf157fd26d1a977323df2279ded","impliedFormat":1},{"version":"918fcf5a4ea36dc085da4f323c7ab62d73cf45f8ff472b8bea7db96234709db5","impliedFormat":1},{"version":"8d1a17b7fd8103f40cd859fa317d90182508957fa9eea299300ba8bcbef971fc","impliedFormat":1},{"version":"9f8fdc648eaf1a9e2db3e04e28a7c931f7fd4b343c9a24e35a1c90c083beb876","impliedFormat":1},{"version":"4bdc1052e65bae069341d76188efb10fe7fb5a4a740b5808ad2c5a1afa6227e6","impliedFormat":1},{"version":"02936427b136646c3576ad4a5edafe0246aa07af9fe7abf336fd530af54cb2bd","impliedFormat":1},{"version":"ce527d4ed3419efe88dc25bf76202ef4393424a8ada90fe84bed472bd94d8b1b","impliedFormat":1},{"version":"dbe33e437acf117b99ae63d7fb5ba9a0fd5ea42fd5521358f68a3c512a44492c","impliedFormat":1},{"version":"c26c342dac3d4a22334724962700cb4e12eb3030dfa6edff4f4df9ec5d645605","impliedFormat":1},{"version":"d364e530078696b3c5427f3d2e8dab96f148320c0f504566cd8e113738a0be28","impliedFormat":1},{"version":"a220c7653ec20a4b7706d467d277c16d71738974b5eaf4c78eb7dd056fa49f05","impliedFormat":1},{"version":"ee1495aad782d603e433e8b9dfaaa9b9ddd07851466df6ff6f21ca806071b2a0","impliedFormat":1},{"version":"789289e8b082a79a660dd568a95f91c33f27e89f9c25a9b7ea6bcf687a074006","impliedFormat":1},{"version":"181e3d1a29b3894478a22651d36b8b9e185f677168ddd30abc6bdd2f35f94789","impliedFormat":1},{"version":"a61c701659297fa42871310e54d4b3c081e362f498c97f64abe36190146b8885","impliedFormat":1},{"version":"2ca7817866541fe3f0c54ff923c1f7091d288880be9f5c07fb1d6edb49551f11","impliedFormat":1},{"version":"9464e8b0705443f5e1e2caa3a54a11b6f473c0836284f8cf8516ecb1ec697321","impliedFormat":1},{"version":"eb76861bb142f7c50a48d0dbac8de02f7dfbe2a0c70120eb9e4bfc323416587a","impliedFormat":1},{"version":"39d2b1cb54393286008cc6e0abfef5355a90db0c82699e0539940c3f8520a571","impliedFormat":1},{"version":"b3fa01d3b343a9030cb5db64e92ae8f7c080d112ab10df2fd74749f807860941","impliedFormat":1},{"version":"dcf1aeb81aa85d7a1d8e18258951f8932ca5054b8328a14d57468bb1dc1f5761","impliedFormat":1},{"version":"2705ae1316f3fc38cc78fe19804febc29cd0bdfcf9f2f8793f2e0b5027a03ab7","impliedFormat":1},"551950f22e0dc4bfe7f5d651243e7a26eaffafa20b8583e1c1b4f06099489261","c4c1eb37af6ee48a25f2fe84fc6a98b623233166ba1e756541c5dc38af5c65b2","7ad9ba79134152ebecc67104f6da6f276bb11991094bb821657c6cd5024b10eb",{"version":"07fcc9be98e12bd2f0f71a501a9bfbe2e53d38c50e8a5e84223fdd05bd8749c5","impliedFormat":1},{"version":"8c724ec27deda8f7ecdca0f22d9bb8c97a4507500ec645f49dea9c73184c2512","impliedFormat":1},"23838336110b26545582dc730fe931d6ed597f2ba193441f164cce2f3349fbed","792b548ea833ac5b87800a87d1882a069df9105270d9ec7ef0f5b34ebc88ae5f","4dbffee15ef008f912df66ab4a38a5b96891fc651b77674c249ae0c89058dcb7","e70894fcf240aa82b7e4c08cb2cd5566fc9c8e077dc1863b203479222fac6c61","1cc4cf97a28e0346ce52f055c52c42cdd6f94ac55176329eeaad0826567113ab","e2b4d810c380679688e57238d9061b787d888d652b4bb3a168af7a1e0307d082","c5010cd069c5d05df2afd21e0f1085c47b16a22e036d20f17b8d084c87ff15a8","1859a8d2385788c166b7ef46e5f39a2e7d92a2dd482f2a32019bf4aa01b8efe8","08ce4f71c19c1c1fd9fdd3eddb1be912d0f77603c43e8f8abf88e5ac75b21d62","af5b838915c37f67e93e72d6b98b7c54844205395988034b47ba304dcd145cf0","d8bc2969fa1c19fff19e140aa07459b7617cf86a2f3de153f33515ecdafc08cb","1407458277a3f512cdd69b8ac9558709786620ce92d4b5ff785061dd79acaae2","9807aaac38406d739f31bb5946b4164eecd58c4300bf844ae76732c065235d90","e1a5d4164bc157459e299ee581d99148cca369d8e5cca60bd66cb9204f4222b6","734a54e72cd06db546eee80e8bbc2ded76ae0c1e3ff8a27473e9c594e3c0a71d","1714fdc2502caff21e239996081a3d1a87218d6a601e79ca48553316db294eb8","a60b111084d3ced80fe0676dda72cc6a4eb17762f694b7ce9f3af18169d3afd3","c872011e0884132a55787d0e1feae7562ad766df40a005b907c23de5485b7bc1","0521990e5717feed1dccd938e1ee41d99a23076a676f1678b25301e61481cbb8","b30e3a3f45ebc4c42cde9454fa7e5e134737f01dc3bf3f0605793a106568aee5","377d2a6b55eb10150d8396074d38445df006f0724353dcd46d82982ccb71c118","ebb002635987c43e7aea3fb773d02b54940958b64c9e8292c15b86170361c886","9bfacd83eefaba8bdccb43a19b3a9ff10530afe2603862ae76bee6c0f25254ec","498f2e065e37434b37b8760afe9c13609748d099127070a0e52865fe04d61fb2","120ee5c0f935ca302cda6397313a3182d662da7e5fca428254d4ec86383a6621","a9383208a44d1dff113594af013496f5d9d58428c4fa2ecbab8bbd42d9793931","50b7c6beef0a7989970c57e326c1d91265b85683528ec44361f732738cc0e5a8","a418db45f871ff9e0bc4e1e0ba28f0126018a06d4ed552843dcb3dff86e5a8ee","a146215884b674be0294ce929519bac61603425d511dbf32815a16361fa6ca2e","3b228f8fdc6de6f99a74492ecc93b7aec0ce4ca6140698635b06318554190c50","7d0d45c660fa6183ddd61202092dcbb2ef7023376e7d72c0bcf9dd2916eb9d47","7d0960bd4162f8f67dec11c30f74520c1acd68b22d62ffc584f151f7b1d3526a","d098e3dfb7a0b4429335d31932e42365565f667224812a8bc292c0960d7440c5","915e2e7fbbef78fab362f7f52811db170687b4f946c238de854a7297e9bbddc6","6017de9c122882cf259d307c1698e2c61946db211095fb9da2d3082c014e820b","a2c6755facc8cdb7f2f3e6b0f5e7aaf30b9ab8d8bcbe1802081f837aaf9a2eb2","77d4165e5f860e90473890224e9c415ad419bd9e4a6b0f14ef8729af5d4ec588","2bfbd431dd8ec18ea6bc6505ad90718980890c839b101e6436e4cb8608bd6de0",{"version":"fb893a0dfc3c9fb0f9ca93d0648694dd95f33cbad2c0f2c629f842981dfd4e2e","impliedFormat":1},{"version":"95da3c365e3d45709ad6e0b4daa5cdaf05e9076ba3c201e8f8081dd282c02f57","impliedFormat":1},{"version":"7245fa556a63e4c5997642422fe4250df16a56694cc1a1f50108dc80fe3018ba","impliedFormat":1},{"version":"7c14e702387296711c1a829bc95052ff02f533d4aa27d53cc0186c795094a3a9","impliedFormat":1},{"version":"4c72d080623b3dcd8ebd41f38f7ac7804475510449d074ca9044a1cbe95517ae","impliedFormat":1},{"version":"579f8828da42ae02db6915a0223d23b0da07157ff484fecdbf8a96fffa0fa4df","impliedFormat":1},{"version":"5f6beb8c43a52e9119cf31a7583ceca88dc910d7de29caad1f81e98497f808ba","impliedFormat":1},{"version":"3ae3b86c48ae3b092e5d5548acbf4416b427fed498730c227180b5b1a8aa86e3","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},"a2e451a27090b30cf55326370f3eb0584d581254280734f08df70104946e0313","2ed94498405232b1e387b432ccf0ee818a27938c376df8ba829f20a076db60e6","a492d2197c75ee9a32351ff04e15f8a56da56ffc07c4bde9ecbe012c7f086cbe","5fd4848b10f0b06096f1cc9bad9e7e928317061d4ca0032ddb95ade7b3b6fe75","a20be30deb698087b64c5d692f4d56973b6a6cb982bc4d3962ad8e0d09312fda","85f46df8ad233b30d0ad0bce7f6dc032ed01ef86b77e807812a06480e5804b1a","431b14ac05b1622e5da7ff1e7e342fd53f247d09340bb0fd9b502e8c6504aaed","647a0df829fa0224dabb4e781b054b1eb2f64218c7fb97d87e6245d64b65cf02",{"version":"6825eb4d1c8beb77e9ed6681c830326a15ebf52b171f83ffbca1b1574c90a3b0","impliedFormat":1},{"version":"1741975791f9be7f803a826457273094096e8bba7a50f8fa960d5ed2328cdbcc","impliedFormat":1},{"version":"6ec0d1c15d14d63d08ccb10d09d839bf8a724f6b4b9ed134a3ab5042c54a7721","impliedFormat":1},{"version":"ac393d11e2c585763ce7a8b9118ba4a809cc19f9bf6d647657d38268ed5d3b56","impliedFormat":1},{"version":"b61028c5e29a0691e91a03fa2c4501ea7ed27f8fa536286dc2887a39a38b6c44","impliedFormat":1},{"version":"a4bf154e0f9d56112713c3a7d2d60c85d667cae17e69f7869a32578881b652a8","impliedFormat":1},{"version":"d5f65e3a5277cbd0b2c89da26703c5879cc428da7ca816d1d1fcdfd7c0a2500e","impliedFormat":1},{"version":"c784a9f75a6f27cf8c43cc9a12c66d68d3beb2e7376e1babfae5ae4998ffbc4a","impliedFormat":1},{"version":"feb4c51948d875fdbbaa402dad77ee40cf1752b179574094b613d8ad98921ce1","impliedFormat":1},{"version":"51d4fca2239d818a6254ba46be06e4def3be685ec034e9255cba403d3b27a07c","impliedFormat":1},{"version":"b457d606cabde6ea3b0bc32c23dc0de1c84bb5cb06d9e101f7076440fc244727","impliedFormat":1},{"version":"859cf43771b68e589bb12c6e5cde3edcde4b530c7d324f455af2b9e61d4f4768","impliedFormat":1},{"version":"9faa2661daa32d2369ec31e583df91fd556f74bcbd036dab54184303dee4f311","impliedFormat":1},{"version":"ba2e5b6da441b8cf9baddc30520c59dc3ab47ad3674f6cb51f64e7e1f662df12","impliedFormat":1},{"version":"8ae5f77a947f3af79012421d027e8ccfdfc4b004eb753e77c26845d4a1907de4","affectsGlobalScope":true,"impliedFormat":1},"a3d03eaa0a0ebb416fec5fb58846db9476231a4aaafc4537b5d862e1b4df6f65","e25fc248031c55bb0a396467a9a5685ad7017e0d7d3abbca1785883c2085951d","08926621914f9b733452b608b4ddf3197a03de210ccaf422d207e8e188524a6a","f1f757eaf46314f40810f12cf94631b0979259d6d12fc5c73fe000db85adf488","cf56b0ee8565bb6676368cc74787f4cf808f0564dd1f688075c75716688ef893","b48af39224c64e04653774309772f5a2c2846592cd27fac75c18d2e88ca954e1","95fd5e797b07a37b9102e1d36f056a5afb7b97ee62b344742ef643521f2f36d8",{"version":"03c92769f389dbd9e45232f7eb01c3e0f482b62555aaf2029dcbf380d5cee9e4","impliedFormat":1},{"version":"32d7f70fd3498bc76a46dab8b03af4215f445f490f8e213c80cf06b636a4e413","impliedFormat":1},"5fb5d82f60a4dcf56508fe206528e269fe7aa775cb5b3909847169b1fe9be19e","592cb7e766025994ab43bae6aa49fb7479c575397e4ce1dd0134dfdc60ea2f19","504c5248dddd1d135c84d14460fd79c4fcddffd79087a966cedea6c38fbb97f4","a72faf29eb3544cad5b5d9616a1627818a8f09b474ef5315b6044132eeac184b","303d6e71bbb57dff9b67d7afc1cddacb4f9b3122347a7db48eb0a61f5aced7ea","61b0f6b4b25f73d0a42cc3ea87bf4cde4bb752495e81738ca80f37fb8ae05494","44b3771f2819daa3866a6cf1028acded0c14ea286dedb50c705e63fdf4259f77","2a59335990a6fb1083e584237aea888636c59c86ce620e3420bd5a42a810530e","516d8ff265a4e1d784eb4298e9c2814d4185257a6475a84b48465efa18721b0f","e353bb86259d150270020df068559cf76a2d4396d34b3ee6318e6bdce546774d","8bacb3bbc7ee97503d7e40d33b44c8a92f805747d139371782b88869726fcf56","ceb9d0795729cce116b0d0f74c60d6ea40e4c367f6f6b405dea215013ce50e91","a8e5774347061cebfbf8adf18417d481a1f6df0128ec25487d080787c66681a9","f4dd0f4d22383c234bc3fb827cec2b7f1c72fc87f76186531157cf3bd320bdb8","fb00a58b9d3265890055f88b076f492c6e4c8a8c03abe0ddf7be94addba86618","d7c1056653649ba0c4c423e6be916625cf7fdafc41990164ca090bd33ce9868b","8d4c36e5656f203818348d3700b94b3a9150a1e719bd35a48e510aa36b8b59ed","9722518162156eb97dcbd76fce8e067f61c793cd21046f34e5c98fac291fbbc2","518c669ac515be16647aabad86f8261dad20a67791ab3718d766c5c5456125e9","eba094b4455a83f2c09e5bb3c3396fc188330229f88f7a46c63b44aac31070e0","04f6e687102da8e679be1d54d8bcb4a104a86c2ec6208d0e8bdc957986d0c967",{"version":"6f69c10b1f3f0ac3157f88719a16edd13d718b2c2aa44d59a9483e04b05e113d","impliedFormat":1},"a52d02261b86e1b08a9a3a5f06835d5efdf75c0fcb447bd4ff8e079f2bd6365c","bb94f7eaad89cb5a1c9563d302c50ee2cd2e6c155692ff122297cd518f33e960","7222b47bd5d2493f0e85eee0874cd24b2dd4f382045b24a0f9cbdad2fccff6b9","162efbbd7bde3f162df942698df1efadb4055e99db856c7d7f96d61a60ffe07a","92e6ba8654688bde151eb28c776453d8335f1602b190026927c9564e32fe8475","4c4c1a5586190c4d7270716db10d772e8b0a9ea7fa312cd5435e9101fc3c7b4e","65b9cbd28a126e85e8dfde684579aa2bf7891310d868ad6fff81633ff5883d61","36bd93d1e6e26b990d91d28ddf74deca9814a4646b0862c856ed4955d4782579","4aae9247f5795f93f36d2b2f770034327cf4c7411f2de58c1b36e03b25963562","66e4317f81ab7c92f3eb916f1a8ee979631c8aa44827ac556449777be339c6d3","32af45e284edaf581e19e701d6990a446359ba4068c7875fa489466f2e65f83c","3600faf34b0f437987d43b99f109a064aace4235fd6b4f52d1d89b794078fdde","5859ff5c1cc373a0c682748878dfad93bcdaeebd6e5d4594a92d86096262318b","be886f87b28bf92d83e596c33d56431c2b9c3a9032f9f0adfa86c9df8b0aa7ef","dc1e0fc94f629579540ad9328763d21fa146eb6b2258c452ff4c3f6c7cc6b025","ed328b37414e5d587a249d5cae79e4d20389accbcccaffdc6a906788294f0226","13fed56e8d15b5c2ed6a8f9906583e8a260f6b199e133d63d169560c9cde1c1d","cac9878fcd0f6ace9eab1bdbf1ed17635be9b4249ee673fbd0b9bb102314434c","bcd3634fa0313844490ebe516531e6ff4182105062f206b885833ad4d5cc74ad","583616787db2314508e212b79100b77691bad017d45cb4eed8c58b9fa2cea4e6","d7438baff14a2c3d19115a46c7d259bfa14143eec1d4581632769155fed52ce3","313cd73dc9c6bb793191ad708b88ffdb33a1e638dfced584263154364f06edb5","6616aba3d4bcd9f1c243a4fed9b4b34c7408a82df80d898e3a28c40778d68813","273c02eb4847ad71d52551871aec3f5fdd17adf6dc7af95e192f4e59dff4aad9","1ba95148c9f40109ffed374a99ec36f8a2525678d03e33e7167aaa1b8b08a968","64e22d8179a0d05fd10848c7fe8a7b9e3316b75b0b2fa0a79c81f1afa60804c8","7720d67b9b578a07785dedafbe0a875cf387eb0d5015ee8676200b393666d831","f669f08e0c0988239e9d8d71211beecb4414b192836350cc02569a51e8acb372","2ef2843aba77a5d09d3e18a50b998193ee41aa60089e0b804bc4fcb3714c021f","94755fe2b911054d0ac211844632f7edeac296117f9c123e3154664e4fa52f83","d6ed51ddd37183fbb2937e6cd3db9e8c03c34b609c5afa27f101687de66d4220","a84b72c08d04ae9054b2d7d1a9f6285ec7a0f22d6e559b085521dca2d4844af0","3ec95ea6afe5dbb588558a97b98f1a535c8dd5cf2be3f2954957f70fceeb7586",{"version":"a4e9e0d92dcad2cb387a5f1bdffe621569052f2d80186e11973aa7080260d296","impliedFormat":1},{"version":"b2dac7c80e9f6c821024e635ffa39f2ab6def88b2d26072dd2915b29e5802585","impliedFormat":1},{"version":"d0fe3f291ed904d59025ef05bf98f1226b8814f924e6241512e21488b03d4cb7","impliedFormat":1},{"version":"8cb5dceed5b9fb8717b93ece4ea5b2adf3fe317d0f01b7278e5d55f19a7f7e1b","impliedFormat":1},{"version":"01329ca0b974c12cc8a198ee6e0e7c8cc9c305816dfdf0e7d5d08360abc34e13","impliedFormat":1},{"version":"f8b0e609ff71a048d86bf0c22a5852d80a762c1f3073fd87e4e24a748e645d5d","impliedFormat":1},{"version":"e24e97519fb282732d44ac765d18f90c1022927a952bc624d58fb7ab2ea11992","impliedFormat":1},{"version":"1edb00a6d353c9222891ff6cfce4dc70fe7cd5e1820456cc7e5c427507f39ac4","impliedFormat":1},{"version":"fa5d0a3ded577f413e3e1bd04e59e2b1c0acaa826bdeffe138b86b513a5d9de4","impliedFormat":1},{"version":"441bef2be624d8ab826cd8bc5dfa29d389b83bfd6a6f026a9d8cf4c7fc6cb20b","impliedFormat":1},{"version":"d877fe18289a4578fb34ea19013e3ef8fbf0d5c7c91cdef9bcd57e573874612f","impliedFormat":1},{"version":"6aa9c6f506b53c3a2e17006fd9b9d518f75506394daa687a3dd5e48f14fb8c0d","impliedFormat":1},{"version":"2e0b7b4d1db2d8031ef7e7f0bb8caebf0c3a5fb068dc1e79d28ff5c981185450","impliedFormat":1},{"version":"ec3005b118e0ccdd71203d774ac3557ba4692c6d2f4b7be802dfb8832054b743","impliedFormat":1},{"version":"b27065cfce873cd68383d574d64c347f473c25dc4619c5d57428db1758c17fef","impliedFormat":1},{"version":"4423f0146fb37cab9d4a722a3df77d195a81412bd9f2bef0f927469ad3d07e72","impliedFormat":1},{"version":"81656d14d40c277b25b02c7b68826b2978064e9eb7c5288e83d1c1941f118cc0","impliedFormat":1},{"version":"30b93c0cab303910b02829a4c6ec32bd09a622089d5f0cbc07782948ce8954d0","impliedFormat":1},{"version":"c77b7782fdfc83af4fbad71446874183d3a89be9a9d81b8114568b2a3c8e3ff6","impliedFormat":1},{"version":"2526683537bd5270fc8c828283530b971ef20fe613da53f50c7670c8102f8f7b","impliedFormat":1},{"version":"0dfcbfff816ec838d0610e07bd6cf004158cf5e41a5e046d99cfcb70b2bbc684","impliedFormat":1},{"version":"0d0c9f06f0efab0c13c2096aa9717df8a8260e6a5c804d301c605eefcfb193f6","impliedFormat":1},{"version":"11c0df3d2e349a7575427aa1a6e391a5284cad25292e4cc74109a1bfd02765fa","impliedFormat":1},{"version":"9dacf04c9c542d2969038334981d87468b87320e99e8cfc203a7e13fbea48231","impliedFormat":1},{"version":"a0c7e388df0028192f174dab2f074c1cd7c8a79b56205f0c0f91294915d52df9","impliedFormat":1},{"version":"a3bd20d0262d0d4da24b67a38629c92cdf1e0c67d610550fd0c40c23c2c5a331","impliedFormat":1},{"version":"6311c547c0230efd5ede4ece1f4ac5ecd983c3e01073bff7237324c47ce0c3f3","impliedFormat":1},{"version":"32e3f90d661c71be5aa6fac5a6e3b531b2bb49694c724e446694fd13980c9e66","impliedFormat":1},{"version":"986e6dcc124af0bea9d3437b3c59afded8f7a1bed66514e0bb7924464a0fb992","impliedFormat":1},{"version":"40e047c6d798ffbf2b769e01bdcc7a7b8fe8ae49b3260ed5151c623d77c2155d","impliedFormat":1},{"version":"b537f57f873438e6656c7f162939cf116a4fa3575d7a46fb4cb6c0e0fd563b5b","impliedFormat":1},{"version":"7af11204419c230069f9bed9b3388bfe50ac032a91ffa3501f8b96d0593bef8a","impliedFormat":1},{"version":"0f2fb2612fb867967bcdab5ea59bf874e19b3b7a8d31e95ae5c49b16b90ec825","impliedFormat":1},{"version":"4eac8a79f63e27930d509fbaf614cf0c779f3777d23c8a06023867997aac09b6","impliedFormat":1},{"version":"f3ff1cd0b656cf7b78c2c166c9bb7d4d2be0d6509691c64a7ad11cfb9fb65ac4","impliedFormat":1},{"version":"d6a87d509be5c033adcad19dbcabca9fd4ecd0114b3f44e45d8ec75e9f392334","impliedFormat":1},{"version":"6d9fabbf693d36d0281a389a13862ab2b20d2c596292fea0f884dffc8acacea1","impliedFormat":1},{"version":"479a80820456c48c5e7d1b917bffcda72efa4fc93b2157b72d01d53f6e524f8c","impliedFormat":1},{"version":"0ddd21a422292a1433c0acb0953b95dade7945db6ad10f47f66dbc3e9656ce77","impliedFormat":1},{"version":"259c2e20c17b8884c5854ca8e211abc423229dbe3ac4f03fa0c7c29bdd3c5f7f","impliedFormat":1},{"version":"ec47dba399069e35052531e38011dca52fe56de0ed8bd96f255e05b0e02da6d6","impliedFormat":1},{"version":"cf9c2ac8e0974422223b788cdf400e34c7d9b0b2ccbfcceda7084ff0b55e3048","impliedFormat":1},{"version":"86f5468cde4828a20f2dec5bdcb5679105bbbd08c3e49c9f9654af190ffaf32e","impliedFormat":1},{"version":"864a1d1df8b3d7882ec6f7050b26404ebc3e4bc9a4187f39a91aba83a13fee77","impliedFormat":1},{"version":"8fa762cbd08bb96cc522d04427ed1dc8af9d584cdab0eba21f33898bea6af91c","impliedFormat":1},{"version":"05d2e21a179498afa4a9e822884830a93e3e43f5289bc1528a5910f461b765d6","impliedFormat":1},{"version":"ca987d92730519fada583cab88c43f20798223b2bf97b37a687ec56c962c30e4","impliedFormat":1},"f80f3ca762fe52f03397ed40e6ef86b2e1f377f55a03db0631d67e57e8919512","984be85622ac34064ce4a377571c890835f91ce9643f8ad87a934049a56814be","aa887a95718b47bc9f63a4240217ac359155abd466105cef21a36442d9ea41c4","e13eb83fb6e5b7f9fd7258cc68f043030b83e0efe1cfe9dee97680e85da1b09c","399a338c1afd5aa7f14146074652550b50ff9622642a95be2c105a7669675298","4fe31b8e880d7a738ab970f1752ce42f6ea00b887c42782872db6bd689dca835","1585224da6cce970f22fa2a5238631a0624f1b79357121a299bc60965b328092","261e13dc0195dbefc3d631bd1b78e8b1ab2c0cbcdf838f8bbe25172263a74c8e","3a0ce954146dca1c0f4189f0663ef0a57dbd048d8b359f90dda125e799a4cfc8","afcd60d922b354c72480f6fc05c16109384a5414c8dfcd21c1aab25bc8a37983","9237c229c3fd63692f83d4b9371287656b88ce4e6be75f70e3f6d0fcfe79d3aa","bda0c9bd29541ffead71bf3efa857c72a826be3a6dad8bc3e6dcbe69835f2dbf","bae05426d32e386b9bd54c9b69008178af06ee8d3bf56696e3f495f378dfd73c","693624311da01ae399f67652484b5d786fd8784d86b4f2371c1e3ab16d23ac47","e38d8963752deae05d3c5999a9d571e93d44dfa4bac7d814c0900b41a91e5081","5b95da50b972d7be58d00976ba08bb264fb1683c410c6b4a73e7b194d9ebdb28","742fde362887b31a1e6043f77b458117ce074d5315ea8903696340e476a02753","8c131ed99722e5634db9f159cb7fd1b0ed7a1d7a4ff70a8383e9a0d2f8499880","6a801665c87fe41e8ced996a02993b2909adb406cf48646f11e5a26ac5632717","d91473ea675709364125916a29f2e0ab0861821a1625e0f72d7fe13f66a56447","a135dee7713cfb851af9569d5b4d23e8b2c155d63676b87607e15b541e9af3fe","3d25dee7eab48a8068729e476bbedaf4a3ccf66ea75c82ce802a2a305cfdcd60","de65b45bceff60892a57830cd5feb7623c132caa97a8bc90af8d579e6eefc162","0227e452ac7548e118afacffb732e41e960c285132547db7837938bbccc0a444","8719c725ff2944e2b2cd34b485b4b5f6cdf4e491f988bd56b7a09af3f283e76d",{"version":"332717be8a6f579d116f24026966eefdf8e57023e179ae2e6c80c256a970d938","impliedFormat":1},{"version":"8c1c61728ff828d1643a656559d5466e8c9ef9234bce65a21c75d7f4d90b2b9e","impliedFormat":1},{"version":"4d03adbf48a9a0f36d3f9ce33b968ea8e0af07e32333bb5b1dc106b69ed9381a","impliedFormat":1},{"version":"351299cadad07cc40dddcd6bfd60681de6e5ecde9d84e4d2ba2303171f5b706b","impliedFormat":1},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"041fe7f6b53b6cc942dbc60aa9a0e0680bd52519b0a63c01583ef65f4068a3c4","impliedFormat":1},{"version":"a5579f898b9e47423f675bdadf4148222f28c08d2c906c42ecc8b8ef4c897dba","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"4a1d973d2c93128508df27876bcaeafb2d5b2e3cf6e84092b6fc8608939b1287","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"aa5524e0509c2168c9493604acf51ef97d2027f03f3b38da097802d3aa719dc8","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"d072cb61b086eeae042c45d85ea553a03b123d3e27dbac911faa1a695f5d6752","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"299fd0c281633d8dbfbe5f44c5f2850fe37392da6fd3b9cca3cb4e10cda16432","impliedFormat":1},{"version":"6c2af5c8d77956b1c82e11ac2386a3c15be42f758dfb597814d49dfdc446e8b2","impliedFormat":1},{"version":"a6e8cbf36e9d911856980c8efaa2187897919ffe897a7a4030693a2eba992279","impliedFormat":1},{"version":"7ed210605489d74ce93ef8b351a28aebd69409f1e9c3ba225d4fbf8ff0428932","impliedFormat":1},{"version":"fc9d689208e575600e837246841cdacf3812beaac77237475d7016422ba86bf4","impliedFormat":1},{"version":"537387829e8d47f812bac08196bc811c521ca53d28f53ead67c5673bebbf49c2","impliedFormat":1},{"version":"1762ed275a1eec5b7d30e479fd3825f88a27fa906a32ff16c64dc67b681780d6","impliedFormat":1},{"version":"a348f5ea72c33f6d2d7a98522858ed8f70981118000e926f915fa5c4aafbd7db","impliedFormat":1},{"version":"cb849466df885c46e229a616c9c8633537fcb44f2cfc39069d8dc0dfdc31d1bc","impliedFormat":1},{"version":"b04781b92ace25dcd4132687eac5d08c5264a87bea640ee77e89b210383e4193","impliedFormat":1},{"version":"ed3e9856772f055d63b460fbc89a04503def5ea71ee73cec0ba39e262a07ec4b","impliedFormat":1},{"version":"81600e99d5aad2774cb003e062357f2c05fe8cb0a370dee4fd48254c46c763bf","impliedFormat":1},{"version":"01c186e3788bc0bfd4d619555e2e15bddcc0eceb4cd256e476a04d091ba2abbb","impliedFormat":1},{"version":"48b020d8433eb29cc297ec5dab4e6eb62957ccbd6c1ee33d4ddb7f73fe50ec38","impliedFormat":1},{"version":"702a76f2b79cfb45d8a81237603017aa6c70558193325fe7cd6076023b6bdcc4","impliedFormat":1},{"version":"05adb45e3dde250b6ae4a50b9dd26457489cbe6bb5de36468aea83df2324e3b8","impliedFormat":1},{"version":"b71da9f22a46322f03f5900108c7bc23fe598e2dcd3955d94df0bf9b03adc9ef","impliedFormat":1},{"version":"98c95e6474d2924e976d896182bdee5635928e7e4a1d85fcd8722731286581fc","impliedFormat":1},{"version":"d2847d2d380d316031a863390efc06ba34ed055babc99a047c000d3a34e1908b","impliedFormat":1},{"version":"8cb85070594b644a3a07bb99b69a305dad09aa03d6325c50483ab19c86095bd6","impliedFormat":1},{"version":"c1bac9bbf0273a0c4a223f09db8d1e89f9a5223f86ebc8809429b140732110be","impliedFormat":1},{"version":"d7afb3a7abe0ea94ebb1afda11d80deed1a07959550dd48ee13897fb23d39a1b","impliedFormat":1},{"version":"cb31fbd38e527e01368c338daa15c472c9dacb72a0a6e483d7f338d2a2b88d92","impliedFormat":1},{"version":"9a056e9b9acc76b8320976d9fe6cd87c53bc1f4e2919613bcceebcff6b539cfa","impliedFormat":1},{"version":"a8f09ab4bfbaf32009c5ceb09f267c45d3e9fad63a9640c3dfc824480522eb1c","impliedFormat":1},{"version":"8d6da1d6d75e21fc322882a7a6cce39c4a85279582c0652fad76ae658f1fa4d8","impliedFormat":1},{"version":"2dcb4881af7d254e20cef0d36e33ef63025cea48ac9b739e310ac8dfb6a4d4d1","impliedFormat":1},{"version":"0e8c1b7ce40dab54106b02db1b529a9d1d34b0bec61bdd3af9c27dfc5041b8d5","impliedFormat":1},{"version":"fdd8e8b914f1d8c32363f45e37f9577be9f65e9171833a4c8c117afde983df3b","impliedFormat":1},{"version":"9fa2d338f2c6e4fb5a2cf20bc83f24102f177c9996a9550ab9cb295afc443322","impliedFormat":1},{"version":"b6b354bd57b44849015d119134a2adf6136dd459fb38f3625fbb35c844343362","impliedFormat":1},{"version":"831e08bc1e0e0fed9f34180a42bcffa15437283b3a90c453f98fd82f639784c0","impliedFormat":1},{"version":"576d3ac930288e7fe44873f24dd8ba11873ab8c536c5f67464e9acdbdbf5f0be","impliedFormat":1},{"version":"6210058f2ce3e9052681f3d2df475d6bda1cee4584dd3a5ef1ef0e60959522d7","impliedFormat":1},{"version":"7a04ce0e85d6db683f63ec9f2699fac3e2d9fdd6a9313dda42e25761a3c83a2c","impliedFormat":1},{"version":"2b9c4aed45c33a93dc6de1e5894094090363aaee045921a0e6ef245657c5315d","impliedFormat":1},{"version":"b9c7f144f9051934bba76cb051d806377a0676ed488ae5764daa9bf7a198fbad","impliedFormat":1},{"version":"dd36b72841bc2a5edbe39640abc5d0175f06b9de32d6b767615c62c460330382","impliedFormat":1},{"version":"de06c3051539ddd64044947bf5a804005e98b09613699b19de1c09ef8e8df95f","impliedFormat":1},{"version":"681c8a82369365bef1360957b467844e8bb3e9626df2162d904c8bbfc92863f8","impliedFormat":1},{"version":"8585b7a7cc1cb48171fd9e168ca0126a921189c6643cc4dd5dac43de6d3b61e4","impliedFormat":1},{"version":"7eb739af12059b8c368641641776937254b381ba298e43820b78696d4e12d3c9","impliedFormat":1},{"version":"f85ef2b6b6243d398b2146de3186f12c825a18295d3913aee1d7ad237856c6c3","impliedFormat":1},{"version":"e83218020bb0bc9a527cf10bca9f3afe489900c60dee03e8398fe135054c81ae","impliedFormat":1},{"version":"d30f3ae4c835c4006e1676170181461e7e97b4e1d2fa0c96a4d0a355cd97fd8f","impliedFormat":1},{"version":"989b02e98599537eccb0e89c3c737b75159fc64299bcee5ecf495535a4096efd","impliedFormat":1},{"version":"b0cfe92f5a41d98256df17702e2e30afbcbc5d33fcde330b20dcac2df26b393e","impliedFormat":1},{"version":"7de22e2447797056c5bbf57153d8e7d7106bab19b2bb8111cf9c9776935b81e9","impliedFormat":1},{"version":"74ecda5bfdd35b1b365b3069acb0224303c20b458e92dbacf6954eef4d9f30db","impliedFormat":1},{"version":"7e1862fcb5a27f449515e7ad569afb738896183889a3dfbf09f125d2ad1effaa","impliedFormat":1},{"version":"c3bc001ab25d65e716b576b0c607d413802122e85fedf847629227fdbf73298e","impliedFormat":1},{"version":"e0644b2e6e5f5709fd5b0377e801ae12fbd1a9d612312ed86b823159421e29fd","impliedFormat":1},{"version":"1dfa53faf0395d8e6714329b001659009171d5268f7231ad05fefeb19b7dd0a2","impliedFormat":1},{"version":"baf281afcc69097878a9f76190ec1139cdcb6d39adb1e0685644789fe99769ed","impliedFormat":1},{"version":"6d0e0c26cd26694ef8f4776443fdd846add0a327244424b5be1eb3320a770486","impliedFormat":1},{"version":"77df8e8553d35e13728f6d0a37ca982520046645694ec1edab9df2df4e905dc2","impliedFormat":1},{"version":"2996e46b71dadb93d38b47e420444d91ce3685f7ff0f0314bcc6503c6018c00d","impliedFormat":1},{"version":"03c9cee66774b18d3c20119b5fd25b3a94f95757aa82cb5bfe9cf7cb91400bd9","impliedFormat":1},{"version":"0c7aadf8379300a1dba35b087663c682f440aa5923ea96a8ff9ff483b891766c","impliedFormat":1},{"version":"70f8da676faa53028982803fb3b389b98119fb0b41df9020a3130b008ac6dc74","impliedFormat":1},{"version":"2409017257471ec5e3bf053cb4a0e0a5f3a7f11901a247118c1654980d7e1fe7","impliedFormat":1},{"version":"8b729a1d9b4f457b23b790a07111742b5c2714b614da768550e0a7309350e438","impliedFormat":1},{"version":"07ae8276b1ded724db7342f1271258bdf7d897ad47affecde328f89543fbef71","impliedFormat":1},{"version":"959e21dddaa3d50c7a9573c033371f8a8eb75e1da2e4f0d2ffc2e44862a7939f","impliedFormat":1},{"version":"5c5150c7774dcedeaf599b1372b8158b3f8a0e665d602a35e34470822da59f59","impliedFormat":1},{"version":"fbe77f3f07a47c30835cb7b88d1aeaf605052884b819f0669977a9977bbd4a8f","impliedFormat":1},{"version":"07cf01ae7f286d5c1390bec1fc2cad285c2cd96b3778b61eddfadd2305786040","impliedFormat":1},{"version":"d07829c9a6e7288abc6d1df2d0e3ffd31a2418b38e7bf3d374437042844ed17f","impliedFormat":1},{"version":"7299ae6e1cd70af673d26872a2c7616ea6fa287021669473bdd7842c94094f61","impliedFormat":1},{"version":"820567b6f3633584ecd3e57c8cc192a6a18f2803edfe730fd1531d9cb6fed891","impliedFormat":1},{"version":"2ae462dea06b9d0a202c1c034ae686d225169038c33242052f4edf93db00b254","impliedFormat":1},{"version":"5ffe14c99d9130074d6bbc1edeabe4b4ef9168a05986ac7aff84ac5735c4d77b","impliedFormat":1},{"version":"86241fb7f3594bade8e6589a5426d72a23dc6426703d43e1c8dea0570d78fd14","impliedFormat":1},{"version":"97444e4665407602750452c136344a02669cd8c42e07fdd4e1952bdb5df13a90","impliedFormat":1},{"version":"b0765a00e3e002773a6af233b937dfebf23fce20a9a7abcabb44ad6b7532e6ff","impliedFormat":1},{"version":"39ae6f648b10326364bae3e24c5735d12ade5ef4ba6ae6cf17e6b57dfc7d736e","impliedFormat":1},{"version":"fdcd57d2575b4e00c4c92b1a2fa15c791365aa763c4d4c901c3f3a362acd27d5","impliedFormat":1},{"version":"860d8f412e586be2009ba1806760f46f0501aea96880436a53956915295ba847","impliedFormat":1},{"version":"0a02b0f5641d72d688128db3e2608d927612131c11eb4ef6ee28c880b8577019","impliedFormat":1},{"version":"8f1e32247ec8ed20364777038b8d11fef3b6bd73b8596e11005481a3ea2a1bf1","impliedFormat":1},{"version":"9a7e14707830dbb4968c875b9f8ab510f531f11d3162c64d4188fab2ab0b671c","impliedFormat":1},{"version":"fd2d5cc8041746b1cc71ab247163982c68b4fad0522b2a8f555368d57f1aa134","impliedFormat":1},{"version":"7f56883fceba869ca2e3bab049cf33272bac1a6937c235c652e0bbd9aef67624","impliedFormat":1},{"version":"af1f132f95a4a56d9304f8dfe68f116d2324b0368411574932c55cbe2fafa214","impliedFormat":1},{"version":"4e0a5de3811fcb44426b7f6bea3458d1c794591d0b4a715c51c3ad5d70f08ab4","impliedFormat":1},{"version":"68b735874e866f37d072bf674535a9f33965132ed9e1e4164a6fbc494d590604","impliedFormat":1},{"version":"9cec0cc895584e0e42d5c97c4a16ebd25a4fa60730c186edf0d28df0a5bc3702","impliedFormat":1},{"version":"94d41a754d3dda0f2517d49d181f1ba1d812b85e7bc2c822c40be945328edc33","impliedFormat":1},{"version":"842ffda229092b37ce0bc6748b09a38aaedc8490a69b3a10ec48ebf47baa3224","impliedFormat":1},{"version":"0449afb9de90d153878437b4004c6c5ce1e2f8f33e93ace383d387b683bac845","impliedFormat":1},{"version":"358999876ec96fa23597d354ed2fe6ae495d7c654e720ab3be349179133ed14d","impliedFormat":1},{"version":"c5de19123a4ab7d152289dd2d7336ff852e78c134f0443cd1bf40b7cb9cbf0c6","impliedFormat":1},{"version":"4f84447ecedf492742cf058a1dc4a3cba63b932778463c858112e4072c79208c","impliedFormat":1},{"version":"74e3a9065b290394d3ee7fb111bb4523d846663d898aa21bb76c8e9af979ffa2","impliedFormat":1},{"version":"c50e1748196272f55890a55bb1cda5173fa615e4a07b3f741cf4f24eaeef838a","impliedFormat":1},{"version":"e19b2a73885f75f29b2adcf1077c8dde7d69137af24c065b5ae7d7fa9bd3b820","impliedFormat":1},{"version":"03770bdff8a4fb0c206e60d6139aa924b5c0bbf94e924d6625f553f59a6a27fa","impliedFormat":1},{"version":"2e54730070d00c443dbb388a356667bceb0a4c7ac5711c0cfc6355964cc7ab2e","impliedFormat":1},{"version":"09846ba03c1c6081fd8600716d65328421d37356a3b584b898325d5d013ebe77","impliedFormat":1},{"version":"7008aa856c52cc0af5aa6c755abfba94dbc5e0a9dac3f9a985eb5ed52e6d535d","impliedFormat":1},{"version":"27551a71453552cdb14453753b2808fb405e6b1495f53b1de318953ac4ac73b5","impliedFormat":1},{"version":"0bb991b7c106b013ccd1b236bca252a34d8cfd2a61387409c1c407e8e07acada","impliedFormat":1},{"version":"0c9fc9833804b2f941bbaca9c55858b3959b7ab386bef26ffb897361430bbe0e","impliedFormat":1},{"version":"be0b2fd0e29c0ea2153fac7069219f520447b1925eb1e9bb8a26623a0f1883f4","impliedFormat":1},{"version":"8a3ba8887f57d779b22773c0a7d988654bfc4ae2c7c3dfb497b8e7d0414f082e","impliedFormat":1},{"version":"b63cd507f90ae6676915db153a40ce7d6a4c2796d4eb9d607a6739f4017b04e2","impliedFormat":1},{"version":"360816468e738d7f3a96116575347aa1b1d3d470a35be1c3953c50cf6d50268e","impliedFormat":1},{"version":"8e9f6de6a56197fdf0b0a75ae16f95d3b067607ec1ea389b2ed97f4a9d83eeff","impliedFormat":1},{"version":"3e33309e181bcd36a58ddcaa2ccb18426e6497338f5c4d2f723804b2da440834","impliedFormat":1},{"version":"51c9f201ce3da119ca046f465c131ec8bf1e4dba44cb21fc7d3b83f2b75069c0","impliedFormat":1},{"version":"b79bd0d2db49a032b3f2bc488f1ae36ecb03adbcf7d15904cc83b29bdf0b91e6","impliedFormat":1},{"version":"881d0a25d25a83449b54d6b4f963c891e742edcef384bccb66e074561f50be2c","impliedFormat":1},{"version":"e9736dd1f0d9c3f00a7980640b2aeba560146ef7094546549d64ce5d51c34c1d","impliedFormat":1},{"version":"d6bf6f2a8f1bf3fdc6ad05035672d8c38a04f72f39c5a55db9f1844689eec144","impliedFormat":1},{"version":"ec1e72c665317467b85ad4d27f5c106e6a888116f8757d883c8600e5f299192e","impliedFormat":1},{"version":"414e31d3a66f5c79cb49e82c3960a6983f1c031df82de1008bd255bf7aee58ae","impliedFormat":1},{"version":"0b981a84abb59dd25ddc171976274c353809cb6c9384b6a82d1950790a25d0f7","impliedFormat":1},{"version":"0881bbb944fc4d723c4ac7dbd4bccec7a5bad8f1cbcb677d12126e88a92dacaa","impliedFormat":1},{"version":"5b022572fb0abf915827636c7d318a067ccf6d9836f020f2350e7c6b01268695","impliedFormat":1},{"version":"ad967a3bcad41e910ffe99906e78f68dcb5191b752327b0eac5120427212cf5a","impliedFormat":1},{"version":"cd65b4ea38085499721fb1ba4624ab10559d49b4b106f388fb35c23b3855f6c6","impliedFormat":1},{"version":"a6ae4e8d4805e76cff3fe699395c1781537cbdad8dc94417db81ed4f65a699e0","impliedFormat":1},{"version":"7808da70bee3b44c90aced70284302f34cca901e190635c26694ecb5b4ce2d6a","impliedFormat":1},{"version":"16a535be43c022b2b40c0fb4365841633beebf3d6f55f808f4999d830883e9d4","impliedFormat":1},{"version":"87fd703309f6f640f2a0a6ce79c0b14c02cbbfdbd3913d6af601be883ab8cf18","impliedFormat":1},{"version":"9bb021b1303e52cdc159ad2a254e449c68b9b5157ae26b9d918f19e2c8d94223","impliedFormat":1},{"version":"90c36c6e2722eecae74eb676d1d5cd8d6b788ff9d27bc5cb99571fa42fd0ae48","impliedFormat":1},{"version":"f5e0905774ccb56cdaa68627b7cdaa7d59634bdbc1d70954d69a3cd6b0b8099e","impliedFormat":1},{"version":"2b4ccaad2c242381606651a2cede50ec716b83caf254660849c60970b319c324","impliedFormat":1},{"version":"234153bf35fad77184c16cf346ce96b392c291f26d7cfd5f45f868e226ba7fcc","impliedFormat":1},{"version":"b01d9cda99bd4a3363a6605f0e20de677fb6942eadd642991fb05a27abbba73e","impliedFormat":1},{"version":"b69a0e1b607af744043ccf45c4f6ea74580cd60e173380534b77153300bd3f17","impliedFormat":1},{"version":"bce03a3640e013438c4567ec02683ad313d8c9ea64de068e5a508fac67554ac6","impliedFormat":1},{"version":"ce69ef779e504f2f48fc59e25e3b9817a4d813e77c7d3e936f213b8cdec7ac14","impliedFormat":1},{"version":"dcf522a16eb912d546550a5484d74f610ad6b5b156d50694c6c6478c8e865c97","impliedFormat":1},{"version":"01261c650c213c608dae5c038374d796190e03f048167be1f0b8472814caae3a","impliedFormat":1},{"version":"590d1940f1480512ee8de41f980c922393dfb61657ba825c9f73a645754c38d4","impliedFormat":1},{"version":"45fd7479cacb1b2cd1fbbac7091f65fa6f85c04366845dcddcaaf766fe30c7eb","impliedFormat":1},{"version":"7df3db55fcf50063348d0bda64af3f21e4a42eab56f5716c5cf36f2c309716ad","impliedFormat":1},{"version":"eb6d7e86dc52d6594ff5dc7d3d89008ab9f0f7619968ed907856381d75ee9259","impliedFormat":1},{"version":"2775d2fd8e2c6f3192db64cf04f38f44a2cd0ece93fdb6dbc7ee7ea07e7e4a79","impliedFormat":1},{"version":"70369948eff602cc000e532c12174f0745a9a5730617e677eaa75de36fe7f00f","impliedFormat":1},{"version":"c3c95a81bc454902bdffcdf1ecef500d4729aa6a141b5ed38e76bb0e74561c64","impliedFormat":1},{"version":"995e450c7834279b8aa2fa3162e86d3b47b9b7e63c7407ca0e3f01c053372a83","impliedFormat":1},{"version":"dc811d697eee6c86d36f29065e4cb4ea41da5336c3d760b04dc0f8ba61207717","impliedFormat":1},{"version":"66cf9358e50a8b782483f0b8a80d7483d50639642e12a18bcf5ffd0535fd926e","impliedFormat":1},{"version":"1a03a876422fc006fc5ef1fd93113c5c5330db37f2519f8c0976453015a60998","impliedFormat":1},{"version":"d66c7731f11b4eb8dfb79d23d7a3d8771e071a66b75dccedd7c70dbbbf600247","impliedFormat":1},{"version":"6e07d28ddfedcbbfdf773d3a29e3b6bf7b13a9550f2b4dd6c266efdfb3016795","impliedFormat":1},{"version":"aa9ff0cc14bac2b7f0cf2a31f00b0d0f1ac4203cccfdd1d0a188866e2b6cac3c","impliedFormat":1},{"version":"de12aae130d94131ce4fcc10cd578aa7210c55217f51e76a8471472bc155b72b","impliedFormat":1},{"version":"bc06493126b2437e3f85cb788e05d27e0584b994351c162aaf6322f9e51a1196","impliedFormat":1},{"version":"253b6652283133b8fe1c7ba038c7fed7c1b806f60433ae2e41c747f96e5bc9e9","impliedFormat":1},{"version":"02a62b658cac5a76affd34e0da9653fc32d882bb2af956b933fe0959d386c6b2","impliedFormat":1},{"version":"4ff33257d307eb3c960367a4577218e89e375ad5534bfc7c6f88962115ec567b","impliedFormat":1},{"version":"02a4a4666502a566f451841a1d570f615e5b51f9171e99ce0e31fb7b759e7ced","impliedFormat":1},{"version":"d8a2f4cbc18330f8d9cc1b4c9d723d9e77b6003b32355e03ecd1dbd93422450b","impliedFormat":1},{"version":"3a6d7502b7dc76ba0d03b02608e53a5643f4fb8687738940ec59b4e35332bc3d","impliedFormat":1},{"version":"a1feade8dbbffaefe2f6ce05ef37996eac654226a4ed85ce93c2a638a2e78ca5","impliedFormat":1},{"version":"d1aa730c32d9745b332733e65d83867d288bcf89508f51ea58f427df5321420b","impliedFormat":1},{"version":"a75e176ab191f710de50447bd5aded6ad53b9b26c6caee1704613e8100a40f22","impliedFormat":1},{"version":"49af458d05e16e37e5a4b793f65f38f1b739aea41fbf64ac7205b4d02a895550","impliedFormat":1},{"version":"730fddecd22461d03108e74818c2d54070e377cb7547f09b9560405c5d532984","impliedFormat":1},{"version":"443757ad22c5d2103a673b9babadba21930bc98e1c13ff17c57aa26fc470ea7e","impliedFormat":1},{"version":"8b71038e115f0472f4e2ec02bef4a314e1b0bc1d876835ab3e72bb74c4651526","impliedFormat":1},{"version":"00055fce37576a751e128d5c2f5ee5d9a22f2106f8ea21f62a83e6900c92ed73","impliedFormat":1},{"version":"da37d0c35a6f0c058c2d5cc7aee9b975a7d1e6def82e9baed4b1c166447a2724","impliedFormat":1},{"version":"cc6e8f8baffe0c5f6a8c6e73524f385eb6fd95c81bbc349b275d303f00b6d69e","impliedFormat":1},{"version":"58fdd698d0969ea83496975d1fafa3f17565b92eea8c66b794ae734c77887dff","impliedFormat":1},{"version":"04e7c703a42c32a95e5962a06e97adf6abc98975bc19488e6dec4cc03b1daf20","impliedFormat":1},{"version":"914025161b3436cf3bf2a0c1c3cdd05d86f60b4c9d1b06625fc02ffdef8db980","impliedFormat":1},{"version":"682fdf0737120f52e9a6564a99bba087d49dc88c2ee621777e5918b91749d184","impliedFormat":1},{"version":"44208d7a48cca101644c03a154c8544ce8919708e39c060ef1d6526b0672df85","impliedFormat":1},{"version":"f5bf416aa2e3a04499a4dff617266806656a0cf874dcbb3f76d932ea7cb6d6cf","impliedFormat":1},{"version":"24ad82b2c510e55a69dfec91b8f93a6a6bf2a1cce00e7b362398c81b94641645","impliedFormat":1},{"version":"9a90f27e9b4523bdafadd76b6b0b956c29f4ff2167014b9349b458a3be8efc52","impliedFormat":1},{"version":"2e2b206abda66516276ce9f18afd72ef035acb2cba7f911e194a9373ddccc359","impliedFormat":1},{"version":"53175b3b3e8cc44c5a822ab73a9330aef96ae80dfa3b080fff4ff3b4380b6545","impliedFormat":1},{"version":"7d21b5ccd3d4f1b0c538334c4f285195b01421f95c639f164a59fa907f55ddfe","impliedFormat":1},{"version":"1994b6a5d9d2483e6ad2655ea1a28bbe9f157afd93b368956cf1100837196978","impliedFormat":1},{"version":"f930b047fac53fb453faf71202ad06856e435b85878428bb1461d9e69cd513fd","impliedFormat":1},{"version":"efe8071011585e5a4e6f98ba486a3b50a4a12962546b07fcdfd31d7cba3d61d9","impliedFormat":1},{"version":"6340f1ee07636c61d7a54866ca967549453b4b65cc9ce01a930ac7b80352ffb7","impliedFormat":1},{"version":"ace6e06ee0d7abcc780df319cf590f40403bcab66c1f7aa34a23c4e71c8abe9f","impliedFormat":1},{"version":"343f8ca55d746684f18bd765aac60791e271027a57d125001f6f6a4b2c45983c","impliedFormat":1},{"version":"8748bb756dfaeb47f1735753cd8563e06a28fb21f78189cc2ba3252dc7a6f909","impliedFormat":1},{"version":"97a8be08b25f03642fc989afb8328ef7fec3597d5022e69b606c0e97e23c4662","impliedFormat":1},{"version":"8c2a41579a8f79d974c446ad30abc568843d1d71bc9fd5cecabdc7909ac19148","impliedFormat":1},{"version":"770fd130b6d7d71b809e64c706cd88ade602b28645809ab4718a31b8a762464e","impliedFormat":1},{"version":"7e233539bb3a821e4264e9b2788316344c0a6a8f251aa63bbe018981348599fb","impliedFormat":1},{"version":"cd59d351674711db4f0cf99f376dd42ac8ab981d0acfae34d3e31b5e58cc78c5","impliedFormat":1},{"version":"0ffd9f8c9629c76a235b9d6aad4ad683f1104777bc73772905f1d4380160f058","impliedFormat":1},{"version":"f1d3f11dc691e6fb07d4fca0a6dfdf6f256a456f5eec578706fa58d1dd1cd91e","impliedFormat":1},{"version":"f4365217529ac5c0c792444c4c655556dc7a3dc1cebb12c7b8f22bf68688e61b","impliedFormat":1},{"version":"0e141c1f5c6ee7ede3b42444fddd1a7bcce1f02af671ec43c617d4edde68201f","impliedFormat":1},{"version":"5fc1c764111de7f89d2b2e86f8bc22fdb273baa275f5ff8d2e7ea828a05206db","impliedFormat":1},{"version":"d8a142b313b3b38def4b4b89d0c4365e5c7effd1c3a448361042beb0bdcb8a44","impliedFormat":1},{"version":"3b76d6b94e097335fdb71f16db7b0cfa71bf9c126fd6b439032e55d01d2bc374","impliedFormat":1},{"version":"80db7661a743970459b71c85168a92e5a51548b91ef52e485fcc239597f5ada8","impliedFormat":1},{"version":"6afcf19d308f72d41424f2cc3eaab84b522e7f77ffc987f1b28cb9ddbb0bd65c","impliedFormat":1},{"version":"f117450517ebb04a458d699bdd3b03e8f38ae58658e18667e2dc4948684b9433","impliedFormat":1},{"version":"64687220bdb074b06df8bf800d9eaebc7a0fe28c868c80923d56af5d874fe7c4","impliedFormat":1},{"version":"c4b37157d753d187fa5927fa7bdbe8935b68242ea2b1f7a340d9b905793678e0","impliedFormat":1},{"version":"6e9bc9303ef4d9a0ee5a17e9796b4d22168c1935ac336c69884e3acd474ca4cd","impliedFormat":1},{"version":"5d81f18f4180866039e07668a32d898df0a6336a0548998919ec7f497f85ba6c","impliedFormat":1},{"version":"0b4167292c88a75b47853f16665b47018c6695e5cdebd3a7d9cf70e1cd209e71","impliedFormat":1},{"version":"77590aae8633c799521101260537d88360c87e72ea092d66f839ba758679c6e1","impliedFormat":1},{"version":"696eeb83cdcd2961631c0019cddea0e38a16e5e687b54e0cf527461f93e7a010","impliedFormat":1},{"version":"cb17fb0d8ad59206358300181046dae39f66cdad102562cd0088d87ff6182135","impliedFormat":1},{"version":"8437b145e4ad4df671bb99bb79a49ac646495f2c85bd945edecabdb08ceabf6f","impliedFormat":1},{"version":"1c7db5e9ba17e5d4f4267fb1ba6fd4ba398ffbd38c8335422557a7950e995f33","impliedFormat":1},{"version":"647ff75e7b8afe76301ce5b4067fbe87be4f0104d03e627d596520706fc695df","impliedFormat":1},{"version":"f5e604855265b390c923f28fa485942779ca5a685564f9539eac36ae73e1755a","impliedFormat":1},{"version":"b87ae8d3be3434db924b9c8dcb10265cb46de954acbcf6389c44c2d6cfdc3590","impliedFormat":1},{"version":"0cf85a26ab1817238cbf59001093152dfce7d96357d9ff4b026c455f87e20fe9","impliedFormat":1},{"version":"2f02bf91bb2cfcf721f18db1af83ad6c50e079002ec79c5a7ec1bb014137bca4","impliedFormat":1},{"version":"af85028497e2e9902a2142ff8edb93585f453c49b5d679ed1711cc2cafb9db1f","impliedFormat":1},{"version":"8a9235448502135907b4cb3e3256376385455ffc07595310177fb0a617b06171","impliedFormat":1},{"version":"e8881bd2fd230b04d815bd5278b974d9cc6e798cbb44a6e4b671ed1974e1a9cc","impliedFormat":1},{"version":"d806ab6ea9b1099967a11f9e8d939f40785f4482c14178166e9a0386c6df18f0","impliedFormat":1},{"version":"d2d246de7bcc81d4009d4c28e8a8c4920bc34366c63b93845d132deb3539ee72","impliedFormat":1},{"version":"3caf659fd3ff85ea7f962801da208de8d31f575684eade9b103caa3b45d1e3bf","impliedFormat":1},{"version":"97cba43725c4b9dea426e3c31e81de5ad8c5a85e8dcf4b2afa39aef117ee0f1f","impliedFormat":1},{"version":"d4bbaa99c31e834ba8ca2f4aa32d1ce76d2f3d7301d77ef87d6e4c22f0966a86","impliedFormat":1},{"version":"fa866d4dba8ff3030ed22ee15335d5bf5e7c20bd870142574ff96bc42da453ee","impliedFormat":1},{"version":"2732846b3f2c2d4155e7fc57c144805f75d43a16f2ebc610195d7a65737c9c03","impliedFormat":1},{"version":"1dfb40e6629cf803267a65920a3327c3fa6a5e42b4c6fb8865cc503a5b7742a1","impliedFormat":1},{"version":"f35c1a8bca091f454997d35340379aca49d25346e51ab1e15126760ee2e171e4","impliedFormat":1},{"version":"92230275025180a19caae70b82c704d73b2de644c2b4951b72b24101a19093cf","impliedFormat":1},{"version":"a2b176f66f0b708241265fb3b417597c9c9d21912bbb7f5cc00d99af551c2078","impliedFormat":1},"80a0706c702e67bdabac6d4f9cd0dd36d4a71373beaafcd646029d379e0c7a9a","b3f4bff3b10c8b762347376f18743f1c94ac2abbca1aa8f7e4a4ae7a38d14b30","eecaf201c7f14a238bc09f2222cbc5af9895ff69dcd82ad5b793f4bf3bf12add","dd0650c9b2af1036653784f0fe1fec57a500cfbf3faabb815c9915acfd34f22c","455a219f681a48c3766eae3a13c4503bfa3eb189bbe7f960fe0ed224aff66729","4084eba38671008cb762c551910b5a73944ab446f28b7a4431050cd645457b75","d3e6bbfcc2be65d61dfd62fd18aea911ba386f5d581b0072090452a5da047d78","e4d9057a6120fe5c5392015bc9b7a8c09c93627d40805666dcdbdc664689d0ed","a3b7b5a0e7ea2aaa1d2a074d7d4a35861da2dcd555c4d37f4fce6f4b2524a523","d2da14731f879ee5138136da557389a9b3c94e204d07c739432f16ff8f57e7c3","07e56e1fddf7dc6f15f8309b81630eba630322ef5e36b87aedec6dfd1296f851","12ac994c97f5fa99bf9e9db22cb028fc46531b75477647671b6cca8de57281c2","4a1794824021a60470785b28a7da4ebc4aa97235cd274f64463420020cd63727","1fda036d710158d2310b4f878d0e007f72c337183ba5dd3025c0d60f757d0136","5f3c32d66ffadad14fca6dfd6375f1a40348c2a08ee7b39c644032967718e339","9701e293c3ffb14f4af084d6fc5b99ddfa4bd6f37a63ad683658d7a6c23d398f","2ca3c91cb5e02dfb1b3e6b2a96a402728891500673602d0ba0a8416afb6fe57c","8d404ceaa08ff39a50393f6dd2464b9e076811bb84e1c5bb2a4121672fb90c75","7c1a14b10398c5191dedaba1508de0cd41f34778794c0f332e3c900b210cb972","946c765c97d4f158d787c7ebc166246a0cf038c57e0923a49da68405f7917711","16ac41bd48ee615f044782492d04da21e22bfb7006b42e1f887d3d1991cbbcec","7f9cd533a08675900d357871ed4e5b36cc54e272fa85aef06d737d9c51f4dc40","079defbecfa588300d53e7c07d0411f5829bbf22ce789ff4fb92dc261ad09e87","f8592ca7a0888e5a13f6b82bdee9b24ed73c2a6b71719c002a4013464de1e30a","7f713468e067f0da7508624ccb350a65cc07e33eea1b3e1ee15e96694985a265","7ceb58a42ce6bc2c96f8c76601440c0cb94df987214c9121d36cfa7e2ad70bc6","32930fa99ce455c7bebf59b51f09f23eb1952efe2e8490604c957f4272685a47","3c9f5bbb82d1f5b424d295963a9449bde41358dc09bf6351456307b99d515fa6","950a276660221f9741dfce986179bf90600a97f9e67f3fe59261f2cdedb9b3b1","b4c588bb90978824b2776860069ba780eeb782decf7f5504a9529ac222f00231","9be91bd31c0282526d7ebf5dc15b2c6c3c4243ffda63742bd7b65677471a4968","2388cd05d863549953753d2ce67b4339cfeb5f5df865355abbf83913a5e6d54e","a948ea60b5e5fda2d111240fd70eafab67591a41ebd9ab4e9e76c7a277aaa1aa","afc000d32cba6b2ba6670a7709dcbaae9ff4c4b5d3f676dc780f048098f1f3f0","da0d88c6dad6941afcc8a9d0fd7dc2ec8a49469c8e03e067860bb419fc6ae398","ae681633470d01dd934220b319d22b058bbc7be3df41f996a385651806998903","940576fbb6e568f5278e2532252340460a43bf18ed120f871296f90b0e23f4f7","0a8e836bebcaa40af2ee7afc1f09de1addbabc8b21db10b5cedc1a30524eebd2","b1f73749b4cfd2b26284b904b454dc449127327e3fd7aca685be4bd770e7695e","bbb8ea461812ada0c55e597561aca2d4c9e7d66c545fa19c1fcc4285c20f6e0c","0629e131a82e8667c20090778d8026d45cf52ad446405ca57178143cb2c479b6","203a578b2cf0d748905d5604a4b1ce500e84526c4e7f462ace6d47459895b97a","8fa380f206e0dbe2dd7793547f5a3e3a1449143455baea518cecd8b93a23b218","e6cd1577bb5ffa00f6c2a0ca5d8b4f943c81cfb85dbc5467c2f8d57b0f2076dd","c280022427a7b2aceda5672e855a92e5efd88f1a5ef20ff7801e047cab165489","2be53f4834ab83330fe03c9cfa32e2de8052a05a263fee416de11528c47737c2","9bd847c49b88468046dbb856061205554f874690f2bc7c3b3d6cd17bf35126c5","1af913f9b3039e673f45bfa4a9abd4dd6e857c9956c9ca98672c19e3e7b74171","c7b4ba7fd9252fc53b9fcb1afca4af6f38c79c053be0cb7fa2a4974cdad933dd","64068d5659ceb1a98ca8f62b52497548e37ab85dbd2da88e033a1154de33edfe","6e66f03773e60659e35cd256b6a61b2762916aeabd365727c3e66c7d3943a4d1","8e002add5e698c80f942706ab52d724f25a5e529a5f875c18c5246a00e46c4ad","8cb701c409054b3a548596fd912b9de1ea3a4a4b302af7af12108be341fbdef4","2f44fb25871052ce683bc622df4a2db16903ba781a754ca8a74254f2c090e2e3","8c8b630e761d64ad1a7e420bf8b8c96884355cfaa45844ed50e5b1857183e804","9e53957c5e8cb5705ce757021fb23b09071c94f0eec27c38ff1262ce940cffd2","90dae985a82fd065f68b477719961e84802f1d1b6d004f97178597a4eb586eea","47c2bdddbf649b0ef55fae7e0fa2d9ec11e015c1b5c77185617b511733713c3f","1beb461fc44f7e3c7d8a5f9f9b5739a7921f9e151d34a872a9a388716627f66d","8ff255b7c89d7c3a7ae034a96dc7fef5dff0cd401a75363877da4f55d5cb7fd8","303a1db85fbefeae25cd0a093ce762e7e5d787572711a594a774bfb4c5ace0a3","e62b4c1691499f6a33142c1b9ff68a3de65dadc6a7448bbc8473b5b7dcaafef3","bce1968a9114eca2b4bc7207bfd51a76354698c37462c31ec322314a82b4aabd","4a031a415667e0983c2c2e0c3f75fee395ae80f743a34679e28c7977be020db7","a689a7f6e9e023a49f6edecad75d0a4c59fa9a86fce23b35b23189eee33d1b3d","6d0ffed8a0ff347005f102318190349120344483ffd30fcda34f103a4d5c68fd","c8edd12c583984f1cea0f8761bf9367001d424519509292f9239cd1261404e79","1633e1fa7a6b8721874af51d1643f20ce7e380b3e215db83ccac2ad2c426bd06","9d2cc50fa68b5251b71daf6342e39d9c4446d02db1120518834af6b450d7914d","175a359b8215135a013babf24172d3de50eadb41ca2aa6b43bcb2f0b195fbe02","656cda14fb439d08a61600632d4718cbe8d2350aae331b694e0eea1d1f76ca73","e37f7c7ea5b6b7a2e0fd199679c21f8a1d0e918b9f59b8aa8585eab8adb2ae9c","4693d4aaeb46836017122248ebf427df3c99e4786696aa71200a13ee746f12fd","d32b148f5feb97a6d5d8d27ef861c77d97b00a78c7a3ac489241ed81a8b62c1d",{"version":"9a6fa82c8ba263a1ccbfbd0b76be4853e4ce80088c19a05986053d1236d43daa","impliedFormat":1},{"version":"771736b1b19b7ba3ff81131b4e4e3f2602affe9014bc348d055ab1f29fe8e111","impliedFormat":1},{"version":"a0cb8606eb67d99e44ad83f758850f9ec0f93ee41ff1438705d730ab55e2152b","impliedFormat":1},{"version":"10d4be4e93c6aad31c4c0596e8faa91c136fce7f844b9857c43ca2e772f29af6","impliedFormat":1},{"version":"b631b94a548848e50a2d73cb75d665fe501f56ed71492ba35c71588e3ecc3d24","impliedFormat":1},{"version":"82bdf9c5e4707f7e99d4f2aa9d0322cc7763beaa4f4aadb8975a17958a543a53","impliedFormat":1},{"version":"a2e59f767d7d5ba0ad1a3e163959d89329b14cb38cec4fe1a1c3f00855794933","impliedFormat":1},{"version":"debb3ab7b5ac10a524327fcd29a9954f0d4d3bbdb7a139e50a52cd7a2d9714bb","impliedFormat":1},{"version":"905938b5b97268d568b5079729896ab9d85dcc7580a7f15f312669fd324aff34","impliedFormat":1},{"version":"d56beda91955d2405674b64e09fa5d3b201ed523fdaa4f22106f265abe0b52a6","impliedFormat":1},{"version":"8814fa8398cd0c43ba36ab4b3e0cbeb684929c2ee75b6dfc17fca5779a54b312","impliedFormat":1},{"version":"09cfbdc7ecad1afdc03f862f6b055dce06adf42d4751469dde62bbd0bc66e4d2","impliedFormat":1},{"version":"6d1c39508b7cc34f216cbe7fb50f4cab696a701cc5f16f8557fa1dd92dbe22bc","impliedFormat":1},{"version":"0232138c115e763ca41fb2c99345f5d00b38dca6aa318f92496969ed3976ba26","impliedFormat":1},{"version":"7dc421746b898f625a41eef01991552dc2a736743b6896b78b2474dbafadd9e1","impliedFormat":1},{"version":"729f68032b902fce0e4518d639b90b3ae566f653322f6ee134fe074dde5194eb","impliedFormat":1},{"version":"f451bb486dbe9dca748377e54f3c6b265bab7ff92d4ada0807931d9d1eda05b3","impliedFormat":1},{"version":"dabab4ab3b7b0d64307a30a214aa8d84b3218a6c13b17c55adda70f3e55718b3","impliedFormat":1},{"version":"9c5c3add0cd1a4a2c3b148bef1000628975c30f7fecaed66b952635b17921f80","impliedFormat":1},{"version":"fd670008e1c0b86fca825b0870ad25726df1a3fd5e27b57d7c1cb0c30713369e","impliedFormat":1},{"version":"e5eb34a6d6dcbd488012f8f35d7fd747433bbedeadcc5061a7114b54265ef6c3","impliedFormat":1},{"version":"0cfaa3d88b15053e766547c7254bf7ad1925ee306a1e95b2d92784d4e9857e0a","impliedFormat":1},{"version":"bb4ccf042a5fc0a77e455b15a1d6dea6e646811797ba1bb271f45a1814d127d3","impliedFormat":1},{"version":"8805654981638cfceb30d0cb69d9d55e7624fd153a07934660bf39f6fe11fa7a","impliedFormat":1},{"version":"ffe45a0886ac155ac748f57afff2d307a24195d1365a9068805035a42fc36535","impliedFormat":1},{"version":"209b6ad1436a8bf2196ca677e430a8d05534e9fbbc7c049b0408ada826a658ee","impliedFormat":1},{"version":"c5c2d9df867da5ece1608d7e8f339f7af9c6e07b07bdb2389f979d5954ab402b","impliedFormat":1},{"version":"b8bdb2db25b4100161e1c2e14c949d11459b518519127fd60fdcfa0adfb148b5","impliedFormat":1},{"version":"751d1a0f997306f1717749c227c7d15132331fa0c944b0b9f8b6d7b94e9ed0aa","impliedFormat":1},{"version":"8d274b8270e6f65dbb08924820976639ffc3e2ce3762863f4726b920bfe51a86","impliedFormat":1},{"version":"a8afb2426aae23fc745745a06c6171cffe9eaeb6be14d8eb2e24556b68f2b87e","impliedFormat":1},{"version":"01f21dec5d7e2fdcf8f298e4eedac690b92ca61e741d8b642e1b96ecd24458f7","impliedFormat":1},{"version":"c80846bdd075b043f17ad6c0e5c9f042b96c92904bab296d5546b66a0715c5a2","impliedFormat":1},{"version":"512db3d42fc11be2a41f0b504025d28f4952f3dad5d6bf8c22b234cf86b1328f","impliedFormat":1},{"version":"18a9089b46e0de4ebc1e7d2cb9996954b92db8b00df49eee73fe0ee230f78912","impliedFormat":1},{"version":"13f7d75a5b0bf0e80b0a2f659d4845a80929b7483138aa7d3b7f18390cb81573","impliedFormat":1},{"version":"95c29bcfbacdc30a6cb53ed70ec819849d2b1a919064618667000b739748fa3c","impliedFormat":1},{"version":"0d8e5dab8f673629c413af62dc258bc8434502b624e4dd0ec74a6f5d78901e84","impliedFormat":1},{"version":"b4f40889a37d3418a4817a44eb2afe9244e8739cfbc5839988aee305cf0d0568","impliedFormat":1},{"version":"cadf30e71517e0bcced437a7b108e8ca6d07523a5f6a5c04ee94b73d9c74e1d3","impliedFormat":1},{"version":"2a1ca525d4ccfe46df2ab79badd4c1d297fe137d92ddb61f283b6ab67bf33b5d","impliedFormat":1},{"version":"272c8775891922a95c49a17975029ff4ff2ce8c1ce6fdf19c010b6fd4510b767","impliedFormat":1},{"version":"b12a9ef1c5688ed8753797bbf3eb3d178dde3d8a7e74632caf38c04d9d2df374","impliedFormat":1},{"version":"5ea9983857848c97e1076f184a1f66ae19d7c786fcc6ef2a7b6e432c3d5f6a55","impliedFormat":1},{"version":"e7818f9b49f5aad015055446fad21bcb18221d90866be6869eed9c094d895a82","impliedFormat":1},{"version":"9818c9ea597b096e8a25ab57e559f9aa77a0ae77e95ac7683a30114233e109e2","impliedFormat":1},{"version":"51853d4d3bfdd8e95f1ce82a0d7c1ec9918381f23e5c7a22e74eb4e5cab8520f","impliedFormat":1},{"version":"07b368fe05496c6190d450787b41f409a78650e692f909d33bebfaaaa39215ca","impliedFormat":1},{"version":"91eb21bd82cc331d43613634c7723068ba7daaa7447d055779568fa831e65972","impliedFormat":1},{"version":"f25684e86972a0268a5275aa9016965ba4f3559a0b7d17123fa42be8505a1de9","impliedFormat":1},{"version":"d9bb2597305abd711c1870db94603cadc5d5c104f1d7f48a66d987e44072dfe0","impliedFormat":1},{"version":"94608743beb1c14e145861941191ba8dd8b092fe9f952cc361386929edde5297","impliedFormat":1},{"version":"fde1dcc51826b1de121ca3f31a34515777369ede0d9cdee92d9f69fcebe8c080","impliedFormat":1},{"version":"4e1d0821975eccac0d8dd05114cc8a4af85384974b5cc36f4f794701148ca571","impliedFormat":1},{"version":"9382cd80f51e70c7fdd16fa8aac523300bb2c0f84bc4e68f97d3be01861f327a","impliedFormat":1},{"version":"b591e541ae1f473ab7156a78db4628669542899544a5d59a42a4bd496458f92e","impliedFormat":1},{"version":"648b01f7b1a4b0ed9ae3975354d423aa471b4e7553f6ef1603dd75dfb850bf78","impliedFormat":1},{"version":"3cc973ce6de3b1438b1d08c5a0b415e65334cb4093ab45288ddb06f29a0a80b1","impliedFormat":1},{"version":"bb487ad87083eb51dc9989d6271c49bf600945ff6878c30b0cb04eca2c69a017","impliedFormat":1},{"version":"7f24d392a436c24f098e5de4879f8319da3d582d3765eb8ae5445c97790094bb","impliedFormat":1},{"version":"323c61bc70555f8cb23aa26d69ae084bd6e00934053bc79dfdf7778671c0410b","impliedFormat":1},{"version":"14b2bb335421928573d218b6e0ef4d32e819743efb9ad53d537c8590eb71a9a2","impliedFormat":1},{"version":"e13fb28e621f669a9a710a71f352fe060a70be237fa78e659e28acd2dd3f6cb9","impliedFormat":1},{"version":"c53ad6e31d7d703f61512a6760525bd2a362ed9d71ab87f7149b1fe991fb179c","impliedFormat":1},{"version":"43b95f7d3137ed93183aeea38092eaad1a29184717b8ada87d46941d0e72a973","impliedFormat":1},{"version":"8cb4a0da83c7e8e6dd38fc0a35150947bdd3df3c367e913f64a4ff6bf90ab695","impliedFormat":1},{"version":"8ad2a2cf8cac4806b41a83d7602d74a544df79b49fc3ceb09c385505163ab57b","impliedFormat":1},{"version":"546979cac4ef9c08209d6c8d27845e9f951bc6b1a8ad5f0332fef8e0dd5c1842","impliedFormat":1},{"version":"695ebf21677adca99575db290ebaaab052cd9b67db3e58b96ea9beffa2b8520b","impliedFormat":1},{"version":"efa83a855d11ce8b33380473d417f2f37b551a06aaf7e3fa4f1f2ad58386ea12","impliedFormat":1},{"version":"e8b58fbefa86c7c46084ed86bdc3fa22350df97d30328facc63fbd9c3f42f5ea","impliedFormat":1},{"version":"95d3cc26a98244ac9df2be6ea562e5df482c72e4218ef0d4dd89af87022cf229","impliedFormat":1},{"version":"97e350f1868b36b39c7fdd0d64bd91dd48e355b452dc84de6fc4bd05196548f9","impliedFormat":1},{"version":"0e64fb82055afeaa4d099a6549bae27defee10cb79e47ca69c25f66fbeac42a4","impliedFormat":1},{"version":"49a59b64320f39788cc504f35e12f01961161f7ad92e2fb688e0173d51d2901d","impliedFormat":1},{"version":"f7750c9359fd236aecf8bc74e47992c17ae8596a8df28dd134e6c56b418def07","impliedFormat":1},{"version":"cd54e59c82e4cac94f28705937a3340add15d9f82bd6b9014745f6e47d4d7076","impliedFormat":1},{"version":"d4c41976704a37da423464fd9bd8cbadba63372a996107a11c919b449c1cff6f","impliedFormat":1},{"version":"fba20a0ee9a7515c3739f9006e2225d0371e8c87bc29f334de0cc509399c79e9","impliedFormat":1},{"version":"b542cc10988bd6f8d96b234974156f3a094609f66d8ea64000ff8cb4d8402779","impliedFormat":1},{"version":"19b8859e2fffe28d35df9074227bcd7aea54542366578863736c0ad847ba8f75","impliedFormat":1},{"version":"f243156e2be74a6d3cfc55c9e15df882344402d9a1089c32eaa131a43f38c718","impliedFormat":1},{"version":"7fc3c1b288bf173eb7c6de841002cb9e337d027c0cec0bfecfd527f2cff484aa","impliedFormat":1},{"version":"ab7e5d40b1a24221da3a66b6b695b345345b80a79f0a287a9e8694a6c4fbe812","impliedFormat":1},{"version":"7aa57dd1af7fab949ab2957a779e6d619ce7b73aa938aa7331b63035587d645e","impliedFormat":1},{"version":"b6ea8d8b96cffc63051a7c9e995f4e2236b37413958d6b421e8a1109d350556e","impliedFormat":1},{"version":"e7d30568e771aecc7b55d92186593d0b54238fa438a8d32d5e65cc5e086f2d06","impliedFormat":1},{"version":"03200c67d843971eff3cecac9aad1223c956a9c5517aaab15e56f84974f9cc22","impliedFormat":1},{"version":"8ec07cf076bcd9d7ce5965917bf7f00681f2faeda773a41929b538d425b0287c","impliedFormat":1},{"version":"d6503783c2ee47092c86a7244e24ebee738d98cf640bacc2a047a01d72624844","impliedFormat":1},{"version":"148b30bce91ec08b57b53310519a473c2146535948d2e9f66ed507b236c6d7e1","impliedFormat":1},{"version":"6db4629ddfd86e4a6cab3d8c37cfa9ea64558c7150cf51a85491c5595f415784","impliedFormat":1},{"version":"18ac0fb6ba31c2b39b750fb8508cd8bb0fe1ffa534c45af9169e96c50d02ea90","impliedFormat":1},{"version":"0ba6c5e2b5d570668fa700fa4989a2cf0478f9eaa6f4e068b808641d747ddd74","impliedFormat":1},{"version":"56fa44665cca7c25b7f88a5771c027200e5ccc32a1a4182ab15ead1b6d53c693","impliedFormat":1},{"version":"ae93912c15f3df75766e7c50a2d7b0d89f41560f0906dc1a70ab45d3b3f30bee","impliedFormat":1},{"version":"56886eab67b8aebec58790f12d701573eba1a69c4f370914a97a8b9c29049502","impliedFormat":1},{"version":"fea7827f32ef9ab636f46b24db0d928ea653aa8e337968b3e6a1d6371ea57de5","impliedFormat":1},{"version":"dd66aabbd86c4044cd215b2d2d4b2ff58dfa06997abe9940a8e92cea05289cc9","impliedFormat":1},{"version":"261318119ec3cb7c7994e9f6f3711d16886fe603fa88367b7c12a6fcb928b17c","impliedFormat":1},"6454e25a71c53dbbe22d2f9fb90cdc86d5375fd5953a729b66f0c45c1bbb263c","823211e8351891dffd137c69aa15e44dd95e74a9c20f2e88f9d8a3a81d4d75f3","eebdfb08d565a558ccbf63d446d7068248e709390272cb9245935054771e7331","9a6d3380d834fe69e2c96a43dcf928c5bb0fbead9ad433e28df3d4cbec7b40e1","da502856b6ac120c7c35c0f18acfd4f74be1e6e1408ef3d328fdd33226295410","d74b98190869b24d176b6306291da7eaa4ec3bcf1056d314f5fbfb3b0b3dc34a","460f340c992918063fe4382bc92dd6669e28977d66ccdf49ad45a3b731ef521b","719c8777b74b07e3d3abffe371efc0e1adc337280b774181f87494c9d23e2d8b",{"version":"bfb309d2cf7c1d004b98eddc388db0f7b51e294f2af88569bd86e761c4305ba5","impliedFormat":1},{"version":"7d80d85fbd6b4e0fe11dde5fcc9aa875547f1ec1a499ca536a39b55d4e1ba803","impliedFormat":1},{"version":"f758fa994a025fefe33dcfcf68d89ed5209b53443285561e5bfe547f770ac381","impliedFormat":1},{"version":"f611b23dfebb4e4ba6fd4f519180526491a72aad2289f7bd8393556879b37502","impliedFormat":1},{"version":"3a93e73ecbb7a89241c58fcf30ecfbf788c3e98d01f5eab4573ce0f8635b6506","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},"e55c40622aa4a3e2649f1e2b315a8148db3bf21f90b75b3b71eb1f1e04ef3461","a52ed043dddca10377be152e3825cc7f138d09b36c07bc26f5c005c5cd10a847","e3b2c118fbff036a284ab62ad759c2ccbfdb424c8bb1abc0a86c89f271963319","196e5e264223000babfa4119641222eb5c9d868c1f98ec41c04b413961ff2ed5","599e9c3c0883e0143428dff62edd4e2770831c39715312f789bd280613c342e0","9ecc6551be9e30b6dcb9ac6a0d03a7e9fa46a6dbb7166353142ff59f0c9211dc","d0450f9f33bdda3c9f55a48e87e9b1ca80d5cbea57609b532487b09ae51c3a41","81a77857a11d3207eabf56c5fdaaed01a656bd90d6a9cadc477218c0d685ea5a","6ed52dab09474ca1fd52f5cd5a671fd03c373f4d3f401a983ad0894af1732e8d","74135ae7cd32292f13e3a161f7fec2736a6daf737a1c6effa4ae93a39f21e020","999fd32277308c2a9e6af6916870d29c08d764138f2a7a4eeb835753c0aab159","3fda9e68e5c7be43dedc124152bc0fcbe42a27a6bd9c86035898dee11490cce0","1fd9f037705d453f5e61f5707655adddf76eb755cc99b543b70682d89cd03bca","a27efe740ebacfcff5b66bb907902ac63b39454d2e0fb1c8877e5f064c2f79a6","f678c875faea49174b7ae113f0064b6578eedc6312cf24bf2d274852d91c5834","9a78f3fe07200404c187c46a3e343c0b3ad8e667df3430951b0d63d7bc79e89d","dc78ef4b7a4563c6003b65c5a772fe4290fe0c23c05ac95e6153471ae79bce59","3dfe429b35f908aa1b44ceeca98326cb1cbf9c9ca247cefd271bbd0edb329a04","6338ad1fd8cb51a7348d7f9b03eb89be904074fc0c761872669345eb1db76bdb","ece38a5091cd6f08b9b9124a271298b282b9f5069e0b50afeec5850447d34455","8b07ee2d35642798ce4a389ea1a5395ec71ffdda810daed94ef47af089a22baa","2541b61b0cc6a380bea67a3d60bad0367a114e00a099cbb63409d8fbd1ec80a2","b9397bf37e43dc5c91135f401705479b2fb9caaef63a77cc221b2980b35c1fc5","04b9afe7ca6723088ca70f20358d10c05844f2f642ac44f34432028288108844","941ed575b3d98d0270d9dd84a63363f808f940a22c929e7ce6ccb8ec0211e5d9","b1f78ca2786070e565a9f8ba71f8692553da35808bf62573049652a79824998f","b186cd27091d5af31fed9697878ad34784707d675222a886bdfc6564793d36cc","bb66a3b781994b02532f92e9afc9e4fc55a8afbb84c717f753297e7e82f8c2c2","2adb7997e5ddbe237d5673d94eb2e8ea57654a7699ade216f5876e2e0483d685","8634d12ac6d8f8b3cc0164dfc7e402a6b164619105180535e5f001c5f6cd6e7a","d3f56d79e63f9b7095321e5ce9c0f9b08815583950e47ea033386969f504691c","4a9eda1538919c911712f2f772a2bbb1eacd473b74d7a9f639c5a363b3c2d571","e3234ab1b515647b34bc3841e8a7aa3cd27c590791bd5539c21466a56cd7f63a","3383c96cbfa9f7fbf54640ee0e873fb4f575f5c4aa510ad47b7795f6f1c4cb34",{"version":"d09375d0b8ae3a3087d6e65b5315b85dfce9bef32a67d7e371d81e1485ed6df6","impliedFormat":1},{"version":"9b94545ea530fa6cd42bd804c67b323722bc52a5c307acf0745963647b2cc536","impliedFormat":1},{"version":"4de12e15f46dc063a7d8f35fb0e3f20a21b5363d9325321fd04aa45bd7fade7b","impliedFormat":1},{"version":"9e008fc8fd36d97bd94efd91794ebef485d21322769ad41dc2ade3ac93b500e9","impliedFormat":1},{"version":"5662b1180edff300f59beebdc535d28e63f425eb3188a4c13962513a41840e34","impliedFormat":1},{"version":"769f6ffa952c42f785831c23b9ad3a15d0f4f5a01965d869cf152d5324d10b1f","impliedFormat":1},{"version":"e8232631fadef3d175e6166d05412bb7745f236489c722fccdffc945b7a8c66e","impliedFormat":1},{"version":"571593d0fb872ff3f69a446cba068ba2dc738daf753166bc700b966a38e1615e","impliedFormat":1},{"version":"802247498d7d71a0ebf77bdab5cbbef06ecd3118bc5d35d616fce928b19a52a9","impliedFormat":1},{"version":"520d605654bc634b8e981268d4034f32dcfa15749357660a52086d9ff6417048","impliedFormat":1},{"version":"e8db60b139d70d1a4b6585c134ffa6b198ef45ab8f077bc75ed85100a535d6f6","impliedFormat":1},{"version":"d624c42ac2874e246f3d3d5b870b294db926b740eb49e41076faec28c65b6ae3","impliedFormat":1},{"version":"43ef1c3e7572e2db8b19d70805800279788fd8a21ef25f326de44a23107a2445","impliedFormat":1},{"version":"c45d005d37db6779816fdadfe3bc594a06109070502f6cb7a178939a8dc71e2e","impliedFormat":1},{"version":"a79975a80dfd05111a9ab450c89e940fd3cbfab72c5c5acbf5c083a676cee686","impliedFormat":1},{"version":"a51c8acd33d3cb34f88328ff2ba373bbcadb94a7aa109db85936ede5aa33d769","impliedFormat":1},{"version":"9f45d143097c192941d2233dd1f5631db1c354b479434443c24a68f612e35756","impliedFormat":1},{"version":"cfe7d9d063781df70b0d0bf6532e0db215bf78df8bf89cf14643ced82cec838b","impliedFormat":1},{"version":"7e0b695a04d63fef9a29271c43dbc0134731df95bafafd02fabbab958d11b84c","impliedFormat":1},{"version":"fafb62ff0d867f9cfa65772b844ee315b024ed1920d99803b48aceb4461934ef","impliedFormat":1},{"version":"00a515caf829319169afedab7105318f3e0f84059be730501f65e55bc2aa0f54","impliedFormat":1},{"version":"d1f1a4985bcee2d1ca4968124bcc941adfbf788c153d23da9b3618cc6d0bdbcc","impliedFormat":1},{"version":"b526691dd48cd60a9b3a0948fa2ec362abdf76b7075ac0b63d9a1d287a1412ac","impliedFormat":1},{"version":"32ebeea9bd221b5e4b48d00dd464ad8f808bd6e6f122479083a67493650ef4ba","impliedFormat":1},{"version":"04f24e7bf3b4757881dcbba7f6a7e76406c13af2c39cc7443f6e0a8c4333025f","impliedFormat":1},{"version":"69ff38c10170b949db2fdc1bbdf034ce10c53dc1e928c46f557f764d028b58cf","impliedFormat":1},{"version":"7e6ac692a05e88acf9bd70f8ecb004601724639cffb744cd6c6f00184b291dd7","impliedFormat":1},{"version":"97b919966e292a86c64d5380d1e740229d643401edf5ceff3b5b4c317ba0f10e","impliedFormat":1},{"version":"a632f74d889bdabc3fe45ee67669f10053e8f7e2382b73592b602415f47e2cc8","impliedFormat":1},{"version":"545bdda99bb5193b7185be42f9dcd4ad220623219f9fc55f67545580391c7019","impliedFormat":1},{"version":"9243758ba5395f14daa32abc91f57bd8bdebe434c0fa9d032167423d92144617","impliedFormat":1},{"version":"72d8345cfead0c29e8b4d3bbe7af86c87ed50af726403e9229adf69727c6fd50","impliedFormat":1},{"version":"238f11f508cc9fbdc6ce283680cb1c1bffa83d662f5588479205fbcb1c5b0039","impliedFormat":1},{"version":"d097201802fcbf25d79999817d7fa0297c9c3f8dd16ea4addd9c0df82f22496f","impliedFormat":1},{"version":"a56623b512c17758214fc7d63f570bdd649eec80b90a45150f3fe7100fdc91f0","impliedFormat":1},{"version":"e0f49a9341d2f448588783315482cc4579bc49a8003f001b40717d206bb6b29c","impliedFormat":1},{"version":"97d47743ec3962e85b183e3a82f92cec29fd77cc04e80ee4d8daeabaaef83cbd","impliedFormat":1},{"version":"6e5cfaad17f7186a2f37e795eb858f38e9a8ca659b51f0700339a00b4eab4c23","impliedFormat":1},{"version":"b32fa7e38efebe74c50ffd3b2bc03bf128f3318dee482562cfcdaf44a50ebe2d","impliedFormat":1},{"version":"6e7cd9be73c8f27300383838ebee513db30469afa6848b37a2935557418bd144","impliedFormat":1},{"version":"157d3c779e63d55f6c985c83384b19d32b604296c051499442a5f398e2fe32a7","impliedFormat":1},{"version":"86a7c300d71cadd326e4299133ab3f384e56c90c48abb9afbcbe21164d2247b3","impliedFormat":1},{"version":"de55125845f2f7617549a4d091bdcd526a8577e26203b5295cef4f6dd826b1fc","impliedFormat":1},{"version":"8485c13294467d11debb2ce65e2bd5a1a06345d664ad759780cf9ea0f352614f","impliedFormat":1},{"version":"292d44873aa2568c467dcf4dedadf666afc12bc72ef19f54a8dbce1d424cc64e","impliedFormat":1},{"version":"5954f628e30fca5d16f0f944104307b71510a35ca40cc779b73aa64da5454912","impliedFormat":1},{"version":"ac21ce340a05038e9bea7b94e9f471a0bd710d412329733253832313b52ed8e5","impliedFormat":1},{"version":"daa5a337ef92b665c4d3c7d1b423b42f8437b028973e04656d67cd90d3609f97","impliedFormat":1},{"version":"5b544f6cfeac0ab86e60fda7f8df2274ec269236cefb0d4de414cb911f8ec366","impliedFormat":1},{"version":"50b6050a2a9e32fb09d4bd87a043d1e389f72d3ceaf84c89f3a433b0bfa8b13b","impliedFormat":1},{"version":"5b94bd8f283e44e5fa59ae2ff9319201877eab8dc4e82d1b9548598d9836e562","impliedFormat":1},{"version":"0abfb15c92d4296ebc1bbef2be397da7e32ae115f368e7bf3183d67108f74f92","impliedFormat":1},{"version":"91356a9d3bbfbb3d6cdcf2bf266d164a4806316f41b561cebd439af209cf6b67","impliedFormat":1},{"version":"080c6ff951c38f28fa8d874139d4db12844acb1cc7144c574cbe58279a1c609a","impliedFormat":1},{"version":"86fdb2e51fa437f8ba429840263fba58bc1b50298667749003cc919454dd63e1","impliedFormat":1},{"version":"a2f3810fc6ab7d0c7cb0eee0aa5d8c07b3f845a6094e7b69ec4f8df22bf9dcec","impliedFormat":1},{"version":"18fdea817a1f9b2e4ca9938257e7074e4ba0c1aef5c07c77fda2f965c5710450","impliedFormat":1},{"version":"04e4c00c3c81419a67c283f67b0f7568f1f52f321ba45ab08ae8ccdd1eea737a","impliedFormat":1},{"version":"f20a3f58b03d543c625ff4219f86df730962a14d75307c14f1809851387f18e3","impliedFormat":1},{"version":"5552811c7987315c84381b934f7ce9168dea216e8979ff761158129106c06f53","impliedFormat":1},{"version":"53f8e61ff1ed46228ad430a8d6c8b4d8c2e7d18764c6993cce1919c35ea5b968","impliedFormat":1},{"version":"1de5a81eb2d35f69be1a241d5811f3f77677d194d8e17afea2a6cb1364aa1843","impliedFormat":1},{"version":"a0fee15eb5957c146aeb0e0b9063c3722630cea51b338e2143141eab4d413967","impliedFormat":1},{"version":"c6139768378d2250104f2430f41623000b0e936b057a87e9bb6ca74c4cc35667","impliedFormat":1},{"version":"10558063f0741148d4adab4decae8695efedbe2dd3f7531948b3613ef935e02b","impliedFormat":1},{"version":"c81a7d5024970bf34cb4e0553d6554b864d2d1ee9b1cb9efd89129d4c3dac997","impliedFormat":1},{"version":"5f65004422c2d5e97f2050075e03e22c4b30bc942facf4f0e22dc4c4e5964e6e","impliedFormat":1},{"version":"a98ab8329757a47b4994e17005f85fccc187c433a673b01734f9c6d7df9291a5","impliedFormat":1},{"version":"03e5cb0828c073896ce4ddc5486af04a0f1dfda965ca8d40e83cac0379c9b36f","impliedFormat":1},{"version":"72179ba23ae57a181141f641b8a20930d7fb0cbb75decf76bba539f72720ed21","impliedFormat":1},{"version":"9dffb9c5958e915e107316229b09d7ae543bc71cb4bab77fe46b62019f3f7ba1","impliedFormat":1},{"version":"95b02dad10c7cfe9727856ad7b863bc98a794d9c9dcd78fbbef0d59e82138e1d","impliedFormat":1},{"version":"36ce8f4245fd0ebf1563225b5b6d6255aae02956575659f07a1118f273cc61d8","impliedFormat":1},{"version":"487bf870931b1db1f6f9b05455fb532d484535a54b89ff5e92cdd38cbb27e4b0","impliedFormat":1},{"version":"1bad797b2f31970a1c2095e694586a1100c4bf90ecb3ea21ac766f7c5ffbb5fb","impliedFormat":1},{"version":"014662f66b41db86f71fd72454b37d334f320cf359ba22474dbf462b90f92faf","impliedFormat":1},{"version":"08f11d512e5e5566163bea17f0a00ed178ed80f92ef54961178f9d38e0307e65","impliedFormat":1},{"version":"e747a86b6011e157e4938f1653d0b9d49864c66af983c04a9cd8c676ac60530f","impliedFormat":1},{"version":"4e07e9a3eb8eebdd8604b03f1969b559e626a8ccd749b98a97da609fc1717465","impliedFormat":1},{"version":"9457ef4968761ef140a335cd82df0ce0c182fe34873f46eeba539b8b18ce7197","impliedFormat":1},{"version":"6d8e5b25c100edb24b96f751fc32e65a1e0bb5f4ebe3c37f248bdd4ac6205fc8","impliedFormat":1},{"version":"bcf062f7c610cba057f8266cf7bd9dce1ec618426ffd107c95718bcfd2b068c3","impliedFormat":1},{"version":"d392d2dcb4c249799f356ba61b5ec1cf1562919794e53290fe4789fe7056e172","impliedFormat":1},{"version":"533b5b627dc3ab444434b532e711f938ece553177314da07f22e42b2a3b25fe1","impliedFormat":1},{"version":"6ba1e80c263737a6f71ad46b17ee3b99bc6910b7e5d0ec18887ab8bc3df93df8","impliedFormat":1},{"version":"910feb38a433653f4f052913f92dfdf85b59777972067218cd373cf2c4c1f352","impliedFormat":1},{"version":"10509ee359d0b852f024946cd75a2f43641ba3cea7b764d5f0d04644d30376a5","impliedFormat":1},{"version":"6577f73a847552974d5a78a57855753bff3c48dd08c7218759030a1f9398aab3","impliedFormat":1},{"version":"c7fa3c5444d68c08f751efe1463831662acca5e985724d58573e977f6e42a32a","impliedFormat":1},{"version":"22dc5d3b866360756aa94850e330d4b03ba0e28aa56070f79fb3b9ab59b641b3","impliedFormat":1},{"version":"afa6e9f245b974887a05f632fbaa77b4b828cb6addf929c6c26d347642d1f3c4","impliedFormat":1},{"version":"2db262f19b477b62c729ba70cbb08b0dead495381c71564eb89eba3ca5aa130b","impliedFormat":1},{"version":"f3517ad5062e51dc990fbaf68bd74b2feeb74f0fa5d1d8d19843bfd76380b975","impliedFormat":1},{"version":"4783515107e6231ec0aa4daa73dcc649d9878a13581471d2c1eb1e07b33382ac","impliedFormat":1},{"version":"bca9e95b232380ba826a7b9a5691328c228be42cacec9257901907486a7f6071","impliedFormat":1},{"version":"00e85be72d85e468dd04fbfc418083abc740a1374f7d769d0f4c3c32ae191401","impliedFormat":1},{"version":"8a78dad5482aab2a0e7266e4b4243605a3a3de1b3e3cc5c4013f6d268e3fff18","impliedFormat":1},{"version":"046070e19c26a95cdb399347a016dfbef1e98d18bbf54e5443fa8427c1013494","impliedFormat":1},{"version":"2ab90c0c8dc7e54c35012d7045b02c26ecd73a1f13a0af3a8e1d9276cd84cfe3","impliedFormat":1},{"version":"cffb93fbfb4eaf5d8f90b5c44c37bf51625c0be1a850aa3e7f635d96f9086a17","impliedFormat":1},{"version":"2cfcc254bebb202dfe8adddec625880a660ca8d40b5f93826efdb2899bcf5a4a","impliedFormat":1},{"version":"7000e97fcb16524d327cc0c71310b81b4f2a8cf1ae09da5a46166c4f521ad307","impliedFormat":1},{"version":"ca0c4d6597919034347e0d18d7cb093b24d2832a76b7258e4cbc47d6cb2b80c5","impliedFormat":1},{"version":"298c1e539abf43d7699f3c56d44e9c200d1a8d6783595a9e9559f8829f9aff82","impliedFormat":1},{"version":"94b42d915abd6a0aee7757d2f634120cbfb2b25a261d7726d345dbc04d8a15d8","impliedFormat":1},{"version":"0cb778bb1c376970e66f64baa49b010349dd6ba6738622e940479bdd8adc3f5e","impliedFormat":1},{"version":"fd9d3b0a5960bec1ac5b17995198f9ddcec9f0f22d7f4d0cb5a1794d0015842e","impliedFormat":1},{"version":"fe754283ea9ea1660658034e18a5aa5d4efe03ebf18bfa3f1f6cae10dc5fd216","impliedFormat":1},{"version":"ebde3951d3800444b3932fe47bc66edee2022ffc1fd66ab2812006af0c1fbd6b","impliedFormat":1},{"version":"b0c0c004ccbdf5851beb48121cdaabb105fa0fd81deec1d38239769aa51c1f0d","impliedFormat":1},{"version":"d217a63d18c6235f73995d02ad14c2dd2c03fd226cab19188fc721622a0c0e9e","impliedFormat":1},{"version":"ed49d2e0c72485ecb8fd316cf8dff9dfb7c7ace8faff75a9244a74b790d9fe51","impliedFormat":1},{"version":"58bd448e81da3af57439dd71817cb06306753fd26ef984040379b2871aa0745c","impliedFormat":1},{"version":"41c11d14bc03cf4dd17e74e71a377bb221d167f50ada10d4364e3ada5dfcb4d3","impliedFormat":1},{"version":"b1fb50f63377766bed70615d76c7933f2f2674f14eb34b8e0011931a22cd0b6e","impliedFormat":1},{"version":"2556451f7bad6e7d62c30594cec691e1b71667f006460ea73da8acf04021f1b8","impliedFormat":1},{"version":"01e035ef4a842296517dcbdbf95d7e86c4ad90cbcd7eec3ceccd1a5ea60cc919","impliedFormat":1},{"version":"c722fb42b32b0d8af47ee6afbe044aff0133d9e23877f3a5e555edbe576a935e","impliedFormat":1},{"version":"bdf55e5467ac502b47f8a6dad728c049fc38419edf6f335900602c3017341680","impliedFormat":1},{"version":"71d6c834fc80129f690120ac1648e33ad8726f8374dd244984f662070df8b7b6","impliedFormat":1},{"version":"1dd3473de87ec322004ef3632b8855bc6d8fd27a2a42e8019d2cc83662c6083c","impliedFormat":1},{"version":"beb0c09ed3611843698d0c9bf7f8b20678f3f2ca12a7e3563a1f0c19249970a2","impliedFormat":1},{"version":"042d6ab69de7a2f2b2fc50c990e0f63aae492ca99dd402756a28662c0c4aab8b","impliedFormat":1},{"version":"484b879b6d2e268baec6f63dea3de40eb7933ba6d13dc659213f79ef30be3c3e","impliedFormat":1},{"version":"6748aa4c56b72d961cd17357cafce25459c488f4c0f58c4c814da61aa955b185","impliedFormat":1},{"version":"0630a3c42067ce65f028bd2aa1a792611c4a1473a72a5af0e4f13cd361aa8a19","impliedFormat":1},{"version":"ecb05882658fb9a3511e96bbcbf5afdd6ed6211fa8bce9621beac9ac251a9c48","impliedFormat":1},{"version":"3fefc78331c4b1cba8e4decc3b99a60b20256fced4f7ab82a83c9876fd91d288","impliedFormat":1},{"version":"c0e5e58ae13b3e1c124978538ff1f37305d014a02e72cc1e13e8bc39c19c2ec7","impliedFormat":1},{"version":"d1db5308bfcdc01a1c882f9adf61f15eea173ed96c0efc89334aa3f5f4554028","impliedFormat":1},{"version":"2aae2c03fcc398cdeb2ebdbe0b2ef75f931a3047bbaf3a7e595b77a1099c3a62","impliedFormat":1},{"version":"74c859e463a329aa5979fb2564d62c539c73872d486c059006fa3451f89009df","impliedFormat":1},{"version":"106c8c8c329985a63a36913def4cb31e8b5b3ada152796cb053772b4875f11e4","impliedFormat":1},{"version":"7f5eeb603ec42ee6452bac0c5a6fbe4bdb2282b0679e29a27a7a8f27eda4ff2e","impliedFormat":1},{"version":"e778ff5aa4e8690b60f314b3cc069129bd925be0d0b725efbc2e0daa096729f7","impliedFormat":1},{"version":"f01ccf24ea873eb216a9d97b80a80f438eefce6231ab0e815092763ef0349c66","impliedFormat":1},{"version":"1ccb5f714564122fdcb0165c4243727ac8391d76696693bd936ef04ce10a1411","impliedFormat":1},{"version":"f45fcbb319bfba9d2e055a25235485e4e84d02d29e5048a5ef641d11e23f8682","impliedFormat":1},{"version":"2900af127a837ab452778e2f1c81b29b283bf21127e0fcfe72ad7be4791229ca","impliedFormat":1},{"version":"e97079d84be581b287039d077a863c0852d8155d66d7f33b7227985bd60f1e10","impliedFormat":1},{"version":"ca772a303716d324ba2fbdd37cd7b6f6df4dd6282576e5f11e6b99c06b7566c8","impliedFormat":1},{"version":"66876ea4dab4349d6bfc74ad4801e9c59ae15a3ad55c0bbe09aea172055cbdb9","impliedFormat":1},{"version":"d89854c62871dfbcb5bcc617d9e70dc64fc17b00d6add0a450dfed1426a749f8","impliedFormat":1},{"version":"08b381422213ace211a0e21b7a39387c73c23ffb38de2c4c96aca2c816173323","impliedFormat":1},{"version":"0b03f4f2c255d5e0a22f958b99860b151ac86b2ba33afd0518b97252913fc71a","impliedFormat":1},{"version":"97d64a33cbc5d29e3109818b61e7b8bdd5e9e16ef602cace081dbbb30dc78deb","impliedFormat":1},{"version":"eb756a53777092f4aaaa7446c1fb120a6b5192b9584683c2b93f11b1f7c1fc94","impliedFormat":1},{"version":"25716553e4d38c13c085a146a80e2d59880e41c8bdacf0f35b5bce7c40df1b01","impliedFormat":1},{"version":"f94acbf4bde891da6f4e5733c9a0a89d5fd612556c5c55a78e481e409aac3729","impliedFormat":1},{"version":"27d444659719b3f998d3abdd948daa29b61e17cd156fe215648b45c2a01c4acd","impliedFormat":1},{"version":"602e5a8fb7503cd90bce7bdb89c8991ecaf03b959ea011dbe15d820a92003c4c","impliedFormat":1},{"version":"aebd9a8270db1d6e4180bfe23d291509e0b7b3346c6c3a74b96063e229496f7a","impliedFormat":1},{"version":"a16c109042a159260ea652e6f04869ca2cd802d6e29e88c460ebbaa56bc49937","impliedFormat":1},{"version":"72cb37e13f5f75a6342873acf3b5b9e05541f4ced091ef9377be76f0d38198e4","impliedFormat":1},{"version":"c9bab1cb74f13513911d7ed9b6f54c6eff06954acf3d93ac3f63673175bd3c08","impliedFormat":1},{"version":"aab2cb5e15dba1be5c2bde01631e6ab896cf2f939360e20faa6bf0d6184b75d9","impliedFormat":1},{"version":"040eb505f604c7cd6d19784ef39a28116688423757e45bd205cf5c180732c21d","impliedFormat":1},{"version":"ecdc8ac5068c76f56ed409332b30f18431303689550f29db4459be42a8d57759","impliedFormat":1},{"version":"bcdd86f750376e91b8b20fb49f89651254b10673925835db5e15bc87446f57ec","impliedFormat":1},{"version":"12a8a4c86e052cd8826d49341ca9dc6242eadd692c41b6aacc991c4fc3221b39","impliedFormat":1},{"version":"f2a81818ce964220422bd65814603aba213c97c9d3e288c906c501b13c3a7e9a","impliedFormat":1},{"version":"e00b39cc3da81347fc71742ce3c368d5d698562968ea9d7283aac177e6dbfa4e","impliedFormat":1},{"version":"4c10ac04b1ec81b106c268ddd1a230167b42910ffdcb366af7c8a9103c04977b","impliedFormat":1},{"version":"a5726d11d9c2bfb12713a2f24944e37084817ea1eff815ad3a37bd831c9dfe94","impliedFormat":1},{"version":"3862bd74c44b62012ad5026eda4655ed9f80d9a51a60b4a75c78cd672849a2e7","impliedFormat":1},{"version":"c868836435628bfcbe89c5b26700aa35e19f1dfb4a2f48705aa5bcef5e48e58c","impliedFormat":1},{"version":"be4ca0869e80b17354edbebf4ca6089c7d2b0b94cf43232a2cbf51576a59971d","impliedFormat":1},{"version":"7998940ad8fb5f194944544c411f4501e99a751c89b48d2279a84a77f412e05d","impliedFormat":1},{"version":"96441db1cfc4321f1ef2e4d49f38f142df0cb99aeea3fe2771f9e0467666e4d9","impliedFormat":1},{"version":"5e96adf1673b37fa57e1e61c5f2d6449dd86f63bf396870338579f5f4b83b82e","impliedFormat":1},{"version":"15de2dd257478e3b928c67b86f20489309aa2ac332fb63d0d700d6d07a50235f","impliedFormat":1},{"version":"84236d8d439b9e35fbef52a75b4f839d50a9a0f3c7186a6adc04a082713adfff","impliedFormat":1},{"version":"a8bafc1a43e787afca38022e0abd4eaf835bcab89cc45b423f83fbd3a35eba20","impliedFormat":1},{"version":"62f2c846b033c528716e5e75ad5e6198ef30e96b52cac1072dfeb44fe3e2e332","impliedFormat":1},{"version":"ff8ff08d83b50336c330ce7e09058d249c2718606a79359e18f279c375f4260d","impliedFormat":1},{"version":"0a34cac0ab7a31ee9da7c1e3ca5db9460bed3c7055c811680292c84c95fe2a08","impliedFormat":1},{"version":"57f5eb93a6662cb82d2c6de9468a1290227a5b40ac6b1d00a5d1bd1cd32b074f","impliedFormat":1},{"version":"a7ae2e27a082e3608df88a7c540b16c430c54c1db42f630d3428198778196d10","impliedFormat":1},{"version":"ee233ad3f4510c003e60a0388c652abe363a6d9f8fa76829eb2829208ea294d9","impliedFormat":1},{"version":"1aed5213987906e5b6823e651a5b943b8ea6ac35724967d5677a6ab5ac92da3b","impliedFormat":1},{"version":"5a8ef4c2202ed36f937660ebfecc93cd8cae285291297b1e55d276930564abca","impliedFormat":1},{"version":"ccf06ef2d3ec5fb22b25389e22b688b9bad80e9f8066d6c582888f05b73afb5f","impliedFormat":1},{"version":"0b8ffa2f5624dded72302a105a47e7d935f73294720eddd501c1d6c672c25f44","impliedFormat":1},{"version":"d81ad031b1862baeeba60fe20d672b3ba54dfa0d975cedf52307e6281daf859e","impliedFormat":1},{"version":"a0f29eba1654dd97e3f419481e51b963b367a5e13612d8e6af83542e5c396801","impliedFormat":1},{"version":"cbe7a8cb6b71cf29eca8803798b0fee90b71fb6e10e7abc7c56bfdc225b8c567","impliedFormat":1},{"version":"b47fa7ccad7aea4d0c48b6f25705c253cef3c682a6476b975c7b163dad4bbe06","impliedFormat":1},{"version":"0d1eb7e0728f88697993119fc18c0a31cfb99b928a28b61317b94bd07b6db2ba","impliedFormat":1},{"version":"b15cf570be51c525bdca8d1a6af80f7ca814a1d8eff02fcc0a90c3796d0cb80e","impliedFormat":1},{"version":"f1e0a503d7be571155b2ca2a3ff758a8906e736d54b2014e7e607f83b77a3414","impliedFormat":1},{"version":"9bc2b4e914e8f47546da8f6fd41adfd066a7924cecc808ad90c5e12f898e1003","impliedFormat":1},{"version":"abc8fcffae5e218a332ff53c3c6e3b5a0f462fab23656f6c068b02f70e8db16c","impliedFormat":1},{"version":"052f2a7a8d2f15cbdd5f7d7f85f9f6a8534743e7b467b611a5135c9e1ecfcb24","impliedFormat":1},{"version":"40c17eaafa266e9579865b0936f4099c9fe52cda31037a822eb53e7201ea6afb","impliedFormat":1},{"version":"6ceb421008ba131ea222660229988d13ac432d119a3129106cdd33f99246b8cf","impliedFormat":1},{"version":"20e71fbbcbfb66f81c52b3471d97e059a7333dc55a149d69945889758e89c3b4","impliedFormat":1},{"version":"4915a72d81013ddc15f5fa5e21f77ef37b568df0541674c0e706f5602a61f633","impliedFormat":1},{"version":"f0d40164418127a95d611f1015d38fa68e438fc0083568936d923b49bdde843b","impliedFormat":1},{"version":"708d531e1292e4f313c6f5b659d42da70cdb2aa4aff1f1e2bd824076440144dd","impliedFormat":1},{"version":"70030889d375c18b9c238cf0e5653f2e849af7e3e97b4a72563e56121e3981c5","impliedFormat":1},{"version":"d1e91e42db77de51b4f2640dad51faf46a98d569e9ea0cc0a90b23b4d89ac8dd","impliedFormat":1},{"version":"b0aec9ad94bed92e08808b6d6ab579caacc97dc4c4bfdd6a84e47dab789efca6","impliedFormat":1},{"version":"fc6b08ef226b0cd1fc0e25491d4787f8f53816c6126aa166f88044db0cce8a42","impliedFormat":1},{"version":"aa3e69f92d8a3e3f85799e1508b99cf28473c94603a70ac2f3d04390c84ed6f3","impliedFormat":1},{"version":"12019613c8cfa8b85821174d369d63efb775ca21f19254e75055a18a61c3b5d9","impliedFormat":1},{"version":"26fd823e635748293190cf4b531df4a6e4e70d3fd08c83721c8b4b4e9902f849","impliedFormat":1},"331069d6d1afd88bfd495f42da1fb5ad197d4602b70e6e399ad52916a338815e","27e0fe58d4a1625ef1ee6a2ae36d5bd4d7b3c6f3bb536926312c8075f652f896","cd36227dbbbdd5eaa2b450a9544c412e1d718035d217c370aad96800dd7726d5","e66261f0c418d22d5fa57ca0e30eef1d91a778fbe6e369b5eb1b7c397e790304","6c84de997f379c581cf641b155c81aa9aa7ec4e169ce6060ee1f7c4a192fc8f9","73c9fe2b3dd58f4f027508a10e6ba98b4ca1cb2d52a32dc5ac5440a076d233ae","0bde33958aada731b2382f05d4910216785062922b695805292709dd30e81b95","779bf39f07dc1b23365c9ba85ae46eaadaa0bca7f08c518355ffba431a8506a8","5bf0019a99c7945f159c82332eba176a3735753a1bbee9c5fa5b78a4d34411a2",{"version":"6d8b4a1ff8b81b5f9c9d37234427435c18e4627364ec105f1caabeae5926a995","impliedFormat":1},{"version":"545669924f3f3d15a1b948660a5af8c815184616e3f4f206c29fd8a1d2348459","impliedFormat":1},{"version":"195b43fb03896d064b88cb04d5f749809f6366415a6e433b70df295bc3cdbece","impliedFormat":1},{"version":"0ebd8d27ba65f83621e8324644dc19c7e64fe3a9d979137cd80ebf40e28b0462","impliedFormat":1},{"version":"a4e037aee1bfbe770ba3bbca72963ad00c8eaea2aade8e03cdaa126a8a1ac60b","impliedFormat":1},{"version":"8ad2f4ca786f17b3aa031e7474c7270e60cad48f17c0e3f4655ced6a8d22eb19","impliedFormat":1},{"version":"1847ed6a92c0558751560e0c8d84cc6192244266536777a5a23f6d7e2a7a0f1d","impliedFormat":1},{"version":"925a01de67c40bb0536e511bf51721f5af6ece57b97d202f9a3d5ca9e3935591","impliedFormat":1},{"version":"dffa2b8d7f35ccfa857bc50f164b9535dc04b8e3171c60d659073577ae593558","impliedFormat":1},{"version":"9c52064fc892b8e85017bb32400c5131c470b0a67caf734e606080712e3b8790","impliedFormat":1},{"version":"c9bf780f99a02f19e2ea5b06856774263dd623c54bb5196e5aa7d11ed0e6c846","impliedFormat":1},{"version":"98034ffa6b681874efc880f751cdba60bdebd485e98822ecc4b39877781de1ca","impliedFormat":1},{"version":"c7cd5cbacd5f544c87dc15b7d41097bd7174dae26d7d0886ca695130de3ab23e","impliedFormat":1},{"version":"04342b37fe58ab10f85a82cb095d0ac7603a38048b742856a82682e42e5374b6","impliedFormat":1},{"version":"b31f82a41bd1098d4564074dde6beb2aee7dc3923b4f036e3fa3af2ac919c2c5","impliedFormat":1},{"version":"bdf095a90885e0fe7b3c4107baccda7ba734a879948d8536d7f69f0142299cb4","impliedFormat":1},{"version":"02467e832e085a7868dbad2a24ca24b1b64f2355905914403c0004ce72fc2e45","impliedFormat":1},{"version":"7720346d5c7bee837106301ca87d3d385a530a6e81251ffadbb3a7fb87a2bc05","impliedFormat":1},{"version":"1a20e2cd64c713ba19e3d8d68b48ffe8e3552674f0d10ecaf954b9595e4054a3","impliedFormat":1},{"version":"cbb46766815bf68c1831c2cf0e9ee0aa9c4ef9fb1f2f992a1ce2d20138cc3c1c","impliedFormat":1},{"version":"1de21a323bfcc897a20c0f5f41ab2f3daf2742e7cb68191037286c63b166bc37","impliedFormat":1},{"version":"bad0ad0b881099cfc1e1ee670eff334d3fd6f17ae748951dd75c261789172a8e","impliedFormat":1},{"version":"79f33d7f269c713197498beb51fe18ec1708f783e9ee36e3aced3bc01d9b327e","impliedFormat":1},{"version":"76fdf754adb2b5614936b8ec1853648145f1b28f2f2ff6216480b92446766f0a","impliedFormat":1},{"version":"a6a6683feb381fc8b72b288fce7a6b32e1a132503bbc3b51f31c6da6e6375232","impliedFormat":1},{"version":"3865a4974e908f332b03a34ebd228d770db0f7a1e311f2229d7df46c0ab2bf38","impliedFormat":1},{"version":"8ec4ba730b04d1afb04c131d25ec72efc36dd305dbb62b37f5045424a4cf9341","impliedFormat":1},{"version":"ead634afd934ba3124ff72eaeb02350bc1d8ffbcbb1b5541b297b928f10a4d7a","impliedFormat":1},{"version":"6a58f3a3ae8a7f2aa8988aebe392f8544dbd3bef96795deba908f3f1f90eebe7","impliedFormat":1},{"version":"6b5ea9f0d12bd3a0014c1970cb841d99a066e0b71d2e268c68cbc3389b2a9527","impliedFormat":1},{"version":"c3b360be6b8292032f8e567175acab932bdd53f9641a307c8dc0feb5817c5757","impliedFormat":1},{"version":"3824b6994efd620592570559e58b10c810c56de51ec9e0c918064aa15d9a0a6a","impliedFormat":1},{"version":"91992163e20474f706604a2bf8ddbbef93475a7fc81a2bd931958f2652fb1f22","impliedFormat":1},{"version":"7318fc87ac80ab31b5da74ee7968ac0e7a208fa3347615caaafa1dc17d83499d","impliedFormat":1},{"version":"b7ad4e29b4608dcb74555bbcd889c453337d5d60dbe9c318e6ecdada4d850ebb","impliedFormat":1},{"version":"5831fb954ea120dfa81a3380c9b4b3825f3a775553c5f35acf9fe9f0f8dc8ab1","impliedFormat":1},{"version":"72bb53897694a7aac8be9bd0c1b707d69549f33fe157b488488546a5901a09c1","impliedFormat":1},{"version":"a3295b4807d520e8ef955e53cedc428da9777a9bace75b61b06b194abead2f01","impliedFormat":1},{"version":"8abfe66a868fa15d1a9f5793a8789d65598901468376b7d33684a183f540da94","impliedFormat":1},{"version":"6cab2a965d8cf4c880908bc5427aa111297197e0976f4df8dc072a12998e5c5b","impliedFormat":1},{"version":"9fafc8a20a54f227f8ffc97fb526e01d7d40e7f746b97c3c4dcc786a8f583e0b","impliedFormat":1},{"version":"ac81562b12319a11a039148b94e618c626fd338ee08df2fe916711a50893108f","impliedFormat":1},{"version":"e47bc0c0462b2836daf4029ee3793b18485aab82ce6df16a1d1dae7dadc08753","impliedFormat":1},{"version":"c27521131d17fc8dcad6a2bbf32b1453d39163130d28db1e02d966f9fdcb0b3c","impliedFormat":1},{"version":"2199d6152493029dc0fea17954b65b1dfebe3f421ccbf62fbb65027964f7464d","impliedFormat":1},{"version":"bd6bb3e0808673fe31479e678feb4674750c879bcfa6247960f51098713186d4","impliedFormat":1},{"version":"7c57ef3e76406327d7a127dadd6d48a25e001fb35fc744b5032b56cb7038cbdc","impliedFormat":1},{"version":"b4d9f3f364719bd0ca9e6710a62670662745e16a7e3281298ded10b2797c9e02","impliedFormat":1},{"version":"f7ba7111e582efd9a804135bd0b156fdc56c20c57f030c30012f62317d6642af","impliedFormat":1},{"version":"557d04797b262fd5e0a6911f2bd8953e849f6e845385d351fd6c0a43ca7ed3b7","impliedFormat":1},{"version":"1d6fa4d92a8099cbbf35c8043965dd29aebd41aec0c21cf8795c2ba4791ff876","impliedFormat":1},{"version":"1b82d649be97e143b557351194072925fd8a6a3922ae25a6307faa3eb22989ec","impliedFormat":1},{"version":"3f9677a142a54645e39b2dcab82af8008723dd53d024e205435675065d3d85c5","impliedFormat":1},{"version":"da2f1ca5b91dc3d9c621b0ec31d3bbb281e1d50f4e134d433a5172cf9beef103","impliedFormat":1},{"version":"f6247942f8d5c18d46f87cc5c69efbb34940e3994023112053119aa6a4154776","impliedFormat":1},{"version":"ed78f3bd0ebc494d018df20548b8a49b114795c62de629be63c32c513dadae64","impliedFormat":1},{"version":"58da935c41436c0f1bb50506f3db51db2900f7f66c528f4080b47a9e2d5eb457","impliedFormat":1},{"version":"5a12e1a765a358c8a868110068be3168c4bb2622f2a2cc6bedc286fc4982626a","impliedFormat":1},{"version":"ebfb84bde4ad61f1f1304e84a8b68bf38ac95fdfdaa15c3f820967e3cef24e61","impliedFormat":1},{"version":"805ccf9dc6a2b41f75d676696de93aa242b86daa40eb9d4764370b266023bbba","impliedFormat":1},{"version":"e25289f80df4cb60ae6a28ddaaa67b3eb6eef0593a856d559609259e894581cf","impliedFormat":1},{"version":"60c97ba6ec78a39b5fef27c89c1d53f724e1dbceec301d6fcee4a8b5eeae9e5c","impliedFormat":1},{"version":"ae456a438db6b13b89582d2c2ffe847b1c180254dce2885b29079ad23dfbfce3","impliedFormat":1},{"version":"4125d11c36d4430e986018869ccbf63e33dbdc0ecea1b05ae9261af5f3855f22","impliedFormat":1},{"version":"95427c6bdbf8371f1ff39cf33d2f165b4912f80f6e4b92c1d44325df64ac151d","impliedFormat":1},{"version":"1e2fd8bf569648448b7aac6d2a2754aef1f4d4abdeb651a77c7169034221bf95","impliedFormat":1},{"version":"4bd93c7f2437e4efae595f9df306bd61a3994486183f48afb5f5c9dc6437a6ff","impliedFormat":1},{"version":"e5c67dab955560d09036c7bce6eba00b7db004270756b9b59825deb2f7c2dfa5","impliedFormat":1},{"version":"0c1ec9da17b07f968bc0eb7498ee8e174abbe4f8fb9dacc80b061b9299318151","impliedFormat":1},{"version":"e3643917f9c524d32fdcdd7712f0c4cb12b99a8acbe31c39170ae31c58d553ba","impliedFormat":1},{"version":"c94f3dace4b68b6c128787790059b67a4812154d0d34764092ba403de64f0093","impliedFormat":1},{"version":"726ec25efd9f3fc4147f787c5353de63b29e35f42e35f700514d9fa8a8659706","impliedFormat":1},{"version":"d422778d3b801a52a3d8cb9d72659e05ff279ab5e6d52da9f8f439ce1a7f42e0","impliedFormat":1},{"version":"024b1fefdb53328a9750e67dabe3512ef527757bd037785ee930d7fb0fad7ff8","impliedFormat":1},{"version":"1ffdd22f172945afeca614c7dc70503de50c1a026bc7d1c22a418f37fc21d4ae","impliedFormat":1},{"version":"db9d3c3f87bfeed613d822f493a3b266310da244d63e159cb101f9097bd165d6","impliedFormat":1},{"version":"06657c2de01950b2e35a8d354c1ce19cb05ed85aeadd5a13813f760c9eb5fe25","impliedFormat":1},{"version":"331461a6784e9645aa85262f3665861581e68d20af109ef5169a4599a948a0f9","impliedFormat":1},{"version":"3c600863f391974b6d469c025ed4de6300e480b7b217182f267277acf1767e12","impliedFormat":1},{"version":"a14e0a2e0461bf1329698a4365d3daf0bdf56ca55200dd17ba0ce1decafb3a1b","impliedFormat":1},{"version":"157dc9988dc42d73d0c41a3af153fdfdf97413c899ee4b7d8837f33ff9d11bb6","impliedFormat":1},{"version":"3c65ea8a95ce5a11f482abae178572a13a337216664fb481e151bfe696bbe9e9","impliedFormat":1},{"version":"a0a8eb2072f7780bc794b545e3e760685fe67575d7c3cec11ba6f134624fa8dc","impliedFormat":1},{"version":"dc04dc2984778ba15a13361aea9c915182f64a1d5bcdb5b4c430168e2b154daf","impliedFormat":1},{"version":"6eb767cbc4425973586c4442987c1a1eb607650891e4a43a9b8ac782a85ecd1d","impliedFormat":1},{"version":"da2ca744f4a1810b9a8c0ec68412cbd053459ea7334797db526cbbe18f022b89","impliedFormat":1},{"version":"1d2190bc27ec5febe38395dceed4786120cd18abcabb2621eb57d8595fec0d69","impliedFormat":1},{"version":"139aa892d9f6359b19074c70b6d2d7893f7176ff08c6694ce392dcb9ea7404fa","impliedFormat":1},{"version":"1bbed39a47c98542adf6e51e89b5fe8216744040352d5bc3e3c9280846fa575a","impliedFormat":1},{"version":"ed3bb32dbd5fad44455953437ea29330438250ff4399bc1a5550bf422d085ff7","impliedFormat":1},{"version":"7059703f16b3a21b84cef695602fcec57ae635ebcdb3db9c1d4973d7d7b49287","impliedFormat":1},{"version":"8d438b33107c05ee65b0d6cdd72ddf63e1b68e237b551ee4d5ac9c3b3a0f0bc7","impliedFormat":1},{"version":"be02b769ee841ea27c1ec02c1737e1966ff6a1204eeb9bfbcdfda9dd8879a7f4","impliedFormat":1},{"version":"cefcf164e983cfe9c73e7322ecb81149e84b2a3ab1cdc95b4d118560f1279b91","impliedFormat":1},{"version":"b889acae3f1d102dc1e19ae6bde31065becce6b4f3229c7cb5c155515cb81a7d","impliedFormat":1},{"version":"5e3cea7a1cc38b2bb8730c66f3b3da817370ea208ef3da43e6e2a8208c32679b","impliedFormat":1},{"version":"ca053bee7713e04e7f4a91acb76e04a77a0e55d3abe5020ef24d29ce0d448d53","impliedFormat":1},{"version":"f4ef7266130f87644a6f12501854cbbfa1070c42cd87faf33f87d64335abc23a","impliedFormat":1},{"version":"0e7e76615747c3248426731aed1e1452e7a2c7f063ba35eaac3e6a8faae1a32f","impliedFormat":1},{"version":"64ac7c1e4f911961a13f32954a53c5d722c57b27c390a3cb9cd05dc74026b480","impliedFormat":1},{"version":"5dd4777dd32f8b35fdc136233e6e560229e564063ced9afd5d0dc3ba3900f8a4","impliedFormat":1},{"version":"c7af068c6e67ee986d76cbad0acfa58de2a338ab9a1ca306f877973cf5461e03","impliedFormat":1},{"version":"899da2fb199e6e16fb89a92c0129e62ad7c586df1a0e5fe0fee579d391b72ae4","impliedFormat":1},{"version":"64b1028b92467d6e87b1bce90d012e5cdbbd4dd35dba54c5f53be19095b11b6e","impliedFormat":1},{"version":"d7c7c82b8a1a5e9aeefe804d2ce54891cfbd2f8798ed426cb9f44e67338b32a3","impliedFormat":1},{"version":"cf1b35415240b6dff4ca871afe1a6eac0d94cb2dad2480576ed05c9512c281fd","impliedFormat":1},{"version":"019721ef1a2e4e735bf370f32ec513ccf719f4919e3be3b41acf9c0b353c9d16","impliedFormat":1},{"version":"4726482f05007d15e6c371514a5baf33cadd38dbe285bac0234c75a9463558b8","impliedFormat":1},{"version":"a26c924fd884bcb2e55cb5fd0008ce58113d459bbc1cb924c99847767f2a8886","impliedFormat":1},{"version":"df29c17f6bf025612d273e1cc897eb4c8221578498db75da3b822cf90ea1c42b","impliedFormat":1},{"version":"a9ed29c4dde54a0c6da1a8092a3754e24de6b74040c68612081267506c7c14b3","impliedFormat":1},{"version":"cd2886147f367b00e24c700f7e6d6e7f4cdb27284f5d9463f412a172b094dc24","impliedFormat":1},{"version":"7409ef36c625d60a86ac1407e49de90da47270bda094e2b9a37d470e26c10c4e","impliedFormat":1},{"version":"eca94a391666cab6347eccf0ec1eabe75bbb012ac3c8efc72c2fc1096ad84bd0","impliedFormat":1},{"version":"ea53fd137a25bd7a0e99f1bcd2f1f190ce396d6818381145ce233c9ae68576d9","impliedFormat":1},{"version":"08c998a1bfb56141b71b737887b9b5c38216e6d6d906803d74b5d53583f39484","impliedFormat":1},{"version":"8a48faaf1f07ee4753492b8fa16e4f496c58fb1dd50414788bb1b9ce2f013d24","impliedFormat":1},{"version":"71ce6907f7e00a0c258e61bbdc49b8a47ca987e2fc0acc065955cc05897d1ca2","impliedFormat":1},{"version":"f06897e7240dd39f8c1a6b79a3943788f03d2aed9858e0c756d045705064e8dd","impliedFormat":1},{"version":"32497a0d6b963314a64a46a427a316aaa9891674e43f4b5409e81d00a0f5376b","impliedFormat":1},{"version":"3dee34e5d87f56f24506eb8cd1cc37d55f942c976c16b8b501e0ded32445c978","impliedFormat":1},{"version":"54513d592a8964918d79a1bf82c41172958ddf48fb13b8da0cbb9e7e7ee33eb9","impliedFormat":1},{"version":"eb950fee5f2afcd1ccd541ff892b150c22a36e682bbf7262701c97f7aa6a49cc","impliedFormat":1},{"version":"f78a9018e694de0e92d90f6454a545f7a778b90a75e62b9f582f3bb74653ebe1","impliedFormat":1},{"version":"a6fa2f52fafaf68395d59f0419cce9e5dfc532d3c39eb0b4ca5d2f6314e5a678","impliedFormat":1},{"version":"dbe240f7eec217e2d4a77ff6f66e29f8be67f060188c4b021908a35fa99d2d40","impliedFormat":1},{"version":"fa66f4d1d69c409c1ea535aceeab5bb0ccb1e93d92e1b2abfe9835900d828adc","impliedFormat":1},{"version":"4604dc1a62ac42e1fd3f5d0a5478359ba92958b237aa2e24403a86e7210c53ac","impliedFormat":1},{"version":"98de9ba7c7ffc2ed99fd98aae6ebea602bd1bc402c34212401c0d2d5084c600f","impliedFormat":1},{"version":"2df41b284a9220755ea0231eb9fa1f9dcf4b16a9ab55e2cf86eeea53f23dd032","impliedFormat":1},{"version":"97e95b0d014b9cc1b330d0c7d3accc30221a091b349f2b79f5958551fb57510d","impliedFormat":1},{"version":"533065f21f74aaa2582172ed02c74e8a2b7a5bfde460ebabb4cbd447ed57ad2a","impliedFormat":1},{"version":"612bad7b6de5dbec926fd384a9f1f6be35d62a64cce688fb5acd511659ffd5bd","impliedFormat":1},{"version":"4186554f153f2017e63dc5edef1aed5bcf9b2dbfe30f9d846463d8300e54fa03","impliedFormat":1},{"version":"4dd9591055cebdc6b78701de85b5f7f03896ef6bb382dd3c120c1a165e1a7e7a","impliedFormat":1},{"version":"c5d1e1c2f9b903a2aa1214ba378cc81f5d4cb9d6d4ac308ad680c4572f8a7f92","impliedFormat":1},{"version":"77bff35abb1af437503fcc6664055c947fdf967869326a2f5285587574e3442f","impliedFormat":1},{"version":"9e38f76f68ee422f536b5565eb38dc8f57b56fef6f4ea498d6c2d81d243df745","impliedFormat":1},{"version":"792f975ba01585fc7050dddded36700a9c50662835235cd44a7d6228f0944832","impliedFormat":1},{"version":"7eb762cceea1cc343e266dcabf7a2e667d8a83035ba1c2558630c4549a8123da","impliedFormat":1},{"version":"b68868886ec1fdf0ebe9ac77ffc64f551d4e6fce20ed81b43ad269482635a44d","impliedFormat":1},{"version":"e42e97e41aaa1471fd1c32f58b21bf8188654c4972d70f2510db726fa9a69c6c","impliedFormat":1},{"version":"eeb540679fcd5c9ed3f0b0f0d2701678f6ac3048f2e06ed794ea9207036a2474","impliedFormat":1},{"version":"40bac16ecdd7ab2d2ce0f91c7e0752f6765453568512f32ef0d7a77c58dc6d3f","impliedFormat":1},{"version":"6e2c902f3f98a271441605452a9bc6b9617c64421fb4b95230f15e4f819bcfa4","impliedFormat":1},{"version":"3aca4dbdc398645e18b35eb1ff3da172c75738f137efdb0a60fe49696f7fa431","impliedFormat":1},{"version":"e940dfa74c91fb34e1e90800d02c20f3a4822c16bcd76e4f661ea4973a53a42d","impliedFormat":1},{"version":"2bd525d2a414347e1a686e5a7361e11bb78d447c0ca812c88971137b971d3672","impliedFormat":1},{"version":"c1f3febd0b3b773f44d15f472fa06a1456c445762acdc72ee481415619a5b204","impliedFormat":1},{"version":"032103352468437f55d36ff748a317f519114f80a3e495b7680c75e038e5886e","impliedFormat":1},{"version":"16f875256267380bf369ef248df4dc10fd9efd4f268e8f0bee7987497dbfc841","impliedFormat":1},{"version":"d4cc1fa2579465da35331a1d00014c55117fb7676996ece38a25b80a1036069a","impliedFormat":1},{"version":"e464a0c5a7c76f9e775861de7e5b342aac7922b1c0b6e2948a5db0aa931e5010","impliedFormat":1},{"version":"c23090199551eaf0130c1ae8400726a7e4749a915cf1634e9093dea8d25d70be","impliedFormat":1},{"version":"ae5d03618590d27f170976005e97817d19ed73d620ec58d707dd650e4c1c3dca","impliedFormat":1},{"version":"f20534945b8b1e39361089d8aaa53c96a3565fc717668c4e6d4850072ffe40ca","impliedFormat":1},{"version":"2e7c6fc58ff11d5df58982645f08fd1edf66d0b987f31dfa29cb430f6b8e090f","impliedFormat":1},{"version":"1ed7804b318cc5f9d51974b8d7565beb90347a9d40a97b2a09a794083399196f","impliedFormat":1},{"version":"a1bcc8bc8f1197eea9b81045844e86dd7f44e3c370314527ee01a9172206a384","impliedFormat":1},{"version":"aa708748232db585634b2eaa61cfd23610df792d60998941c8d977e6f95f8801","impliedFormat":1},{"version":"edbe202e2470e988dbe68756d73b35bc658569d5c98bc2143dd8a531fac669ba","impliedFormat":1},{"version":"caf6506cfcf52d2a66b71503f19b70202dc13a468f9bb0faad8f900ae34f3a2c","impliedFormat":1},{"version":"59e1cb320549256f1170acbdd2b18577c9d0cbf44f0b8c1fb819df966139850c","impliedFormat":1},{"version":"68abc3d16873a318412e6bae31025c3ebb946301311d52172398c3e59d396866","impliedFormat":1},{"version":"296aa66fb1d44df18aaade53bc13bcdf3bd187a6fbaa73b5450823e572b92cb1","impliedFormat":1},{"version":"09fd7e2c1608d3181cd69e23ea833d25653f60012331a2db0c9bab6616090338","impliedFormat":1},{"version":"9648eb8f1f1e8bacddc8a852f0d3ef8e43fe13fff6c4b0d53eb54c256e2287a4","impliedFormat":1},{"version":"cad46ccd68670a60603dd7440b8f2a90e1e89f7766ee18cd95f03ad75d451253","impliedFormat":1},{"version":"4717ec6b7732c61a0e7ab383a476c9817b29328b16b34156cfc5ad0b9d7d8f58","impliedFormat":1},{"version":"9584cf628af3669890d3c67a7a0be8e7cf36454025eab012b6b3ae1c1ed41e8b","impliedFormat":1},{"version":"38e4807e6727998e7092f5b60dd0318abd6e53843eef178ec5f53dab55454fbc","impliedFormat":1},{"version":"dd07c93b6e66096fa542451304bc3f5bb2364a78829364a82b8196d6fd179d1b","impliedFormat":1},{"version":"2dd2aaaf73eef844dcc54c5c3887350e817d128c539a1011ec1b2b176c7c39aa","impliedFormat":1},{"version":"216328150a2fdae431ed0e447aa3295944af1bd199d9cecce4aa4924d162fc66","impliedFormat":1},{"version":"d57321e94af48b9e72edae49243b8d5479a8d85d43c5e21dbf7ee40e578b4a15","impliedFormat":1},{"version":"c0e43217feda3af69a9c044588b132f1a77acab87bf8ee8a0671ea57f191cf2d","impliedFormat":1},{"version":"df53ec34e7ad7e938aad672dd08d9b75c16123464bbe1f387164a3cc78d1cc51","impliedFormat":1},{"version":"2a80821012d3997c8f9a42a16f82f6aa4b2ff334d00d0b057c0aed333bd5b116","impliedFormat":1},{"version":"0425b2bc79473346386335c4a98730288e27e6b3be900a86e42f0492b74a7222","impliedFormat":1},{"version":"e334b173acb82236c822c76c2a9867364e53375cfc8d0020cf4f1302cc126645","impliedFormat":1},{"version":"404d9d0ea2927613a7e98143ec064879380eed25f66be8913d133aea11ad06d8","impliedFormat":1},{"version":"8cb66810e5568395abda507ecd8e8bd362c616b0e5b4a596b781168c67d11535","impliedFormat":1},{"version":"ad6975af7ee038b7f75af724695551c7fc9f3e28a1aa7497d78f5f6bf9e95d46","impliedFormat":1},{"version":"ab6aa891f03df0f4369739449c0d00933e3d130a6eac0d7cfb9f665f3423ea45","impliedFormat":1},{"version":"84f17aecfb88b19c6682eb3c4c5549cacc598d1272d86e383f3c5b9e651392d9","impliedFormat":1},{"version":"83b8e13fb9894aefd5815a4f8db86afeaa0e2a177636110a63e49da567d12726","impliedFormat":1},{"version":"be332510bcd8c40389283bea35e93f1f902cb9d7ef180cf2c0e927a05f085369","impliedFormat":1},{"version":"4d1a546b25a9a3ff60b1ac5e55a3878519c2dcc300dc06a04a6708d3d8e640b9","impliedFormat":1},{"version":"80a62b835bfb91609d1ccac81cf94a3276d2ed8aa9024eb99f629e9aa0f27460","impliedFormat":1},{"version":"d7c9728c0917636fe16505ad9c92348ae4235a589cdff377194bfbaec4a24519","impliedFormat":1},{"version":"8dc226f7642712a4bf68991506ab095578fdb540c2da229c3abfa67f7fe4c030","impliedFormat":1},{"version":"20a143dcd7b924cafc6ab6b5b70a94a04e318e897fab3bf0a2eb21f67f67aed7","impliedFormat":1},{"version":"112e1088892011d916ccf37834361c708e7c8e204ffcfcca78ec22ed48f517e9","impliedFormat":1},{"version":"9d173b6991690ad49bb604ab45e8fddfb8eefb1fda36e68235774c9c45ba5a77","impliedFormat":1},{"version":"a71944ec810edbda13dd7523dcf48b4bd466ef18e21343f17cd48d50b0293736","impliedFormat":1},{"version":"e4709b4e4bccf69b1f25c419310c35083e8d0f6b02acd9a0d31c1863ff4cf9e4","impliedFormat":1},{"version":"7e2b2b72bcd68ec1a659dd2652a4cccfaf3b341202270546c8b9ad9207e6a5d0","impliedFormat":1},{"version":"d152cdd7a87c3ae967813e29eabe763f47a289a38fbee947c23dec3720e87cda","impliedFormat":1},{"version":"de78fafaa11172c4de3e99f0405b938bfe1c5b2889c8beea5c0fc21dc9e277cd","impliedFormat":1},{"version":"69d98158a8e8542ca2624f554255f3d4067b3c95f8af4adc7c523f1ecc785924","impliedFormat":1},{"version":"f100519e3b2430408da9555f6b86fac193ec3b7d63d63e8be5671937e2ab27f6","impliedFormat":1},{"version":"565222b8a4fc2d084b6fee31818cea707d0d372b67a06a311db4d2a035126f34","impliedFormat":1},{"version":"70774af3c44d8599483041d731ff3c5f87fcfa422eaceba7b65e45f5ece90f4b","impliedFormat":1},{"version":"5f9866aef84db3e9cd0bab95029c5a4619625e75e813058f86b92058d23bfed9","impliedFormat":1},{"version":"b16e1eee3045f66b1c5706429a14510da2305cd358dfd8e3277ed36805ca9f01","impliedFormat":1},{"version":"7fdf1aade7cb7f843ccf273ae0fd426cd526a31323faf51f029b029c043c87f3","impliedFormat":1},{"version":"bc873fcc1bdc8fada5654939ec6d24376f376a688c252b3b0d659b71701d0475","impliedFormat":1},{"version":"5b2acd68e2450a06612cd77a2980b24e85d4ee66fed4ee65b528ad10fcd7926e","impliedFormat":1},{"version":"1449deb449c18236f39dce4ae6e8066063be14faaedb51e4a3379f201b0d521e","impliedFormat":1},{"version":"849a24a41402fc007986121747da0ded06ac5434e33396d92a2088fbecbcd66e","impliedFormat":1},{"version":"2623288f4e5208829979e3892de673760de3da9abb7884b4a346e18eec2e6418","impliedFormat":1},{"version":"bc4a8b849345970e5bcb4b2b09cb55145966c23deb96dfb958c3e212aa8f6549","impliedFormat":1},{"version":"74be6f2d91f8486556e00d99b87e535c104c0db4a7fc46989b137797c1f6032c","impliedFormat":1},{"version":"ded16fe486cad27d4f31bd99134c106fc85863a76679ed7dd7e4cff819943bb3","impliedFormat":1},{"version":"d3b5d88efc7ee78ca3d4250ec060bbfc58cf74b28b4339cf8efb672fc5d60ebe","impliedFormat":1},{"version":"45f9074ac03fd96db661e045bae3bca046d01da19b505d5b3f07401481b218eb","impliedFormat":1},{"version":"eb6db105038e5ba14923208161fa58ba2b6d108b0cdee8ce4c73e6199336d7a0","impliedFormat":1},{"version":"2c7b1d5fe345850ebfd5a6b7da9647dfa48047c238c9c8d925c04bf73e24ebce","impliedFormat":1},{"version":"7bf38bce0c3a2c1ab1e08b22d4e66ab9466d3512a4dc27138b01c299fa65cedf","impliedFormat":1},{"version":"3822fbda54201506f0daaf2e5f68c0b487dc07aa8fcba8efa0546409361e5ad9","impliedFormat":1},{"version":"4ab33959bb8e1d202521904adf074c34f0025fb1b9cd90d5dd434a2cdead98cb","impliedFormat":1},{"version":"f12093eca3c838997b2a0227def106995f3a7b6994fa240c0dc2c120b59e39df","impliedFormat":1},{"version":"dad57f2c0560b4d47f9b27bd1413192e52ebcae46d9c3eb00a587eeb5bdcdee0","impliedFormat":1},{"version":"4df8b43f9b70d121452d6d0e2fef3fefa2ac740887eaf81e59ae4e59774ffc7b","impliedFormat":1},{"version":"a9c4875e7cf70b355b32ebc8a9623017be71a5ce18cd01b21e8a5369ce1c9e42","impliedFormat":1},{"version":"906456423f006f67de143ed8a45e067fb19bf207f28ebf1cd111fd02263bc6df","impliedFormat":1},{"version":"d656ca3edc31daf0dad3bc4c932322e09cd063906d0b93ff6666cb92c09d6dca","impliedFormat":1},{"version":"f2047388ef0faa883e6a6d7c1bfe5791b57054f9576f649f92eb150d14f07e2a","impliedFormat":1},{"version":"6edf9b42533a18aeeeae10a8814f4cedae8f80223baada16df13961f35864856","impliedFormat":1},{"version":"8592fe0eea7d1f67204f0b48c793be466fdba4a90a2c3ec7b1f181753d1282f0","impliedFormat":1},{"version":"8bb13585d1dc18a5ab75f64e7f7b253e94fa96fae98d2294b351f3186737f019","impliedFormat":1},{"version":"989f246c3daeac4a4b6dc39fb8ab044448635fbeba7837a59f029cc110ad3340","impliedFormat":1},{"version":"25134baffac2b03586eda34193b974a821e567b1a5bffbc9cde61d0e16074b4f","impliedFormat":1},{"version":"a36f879f34d812bc8c5466490725b8ed9a9f9a315eee41de20a9af9b7c2a4820","impliedFormat":1},{"version":"122cfa505c7ebd278f37dc10283f4c441d150445c92a6c2bf2ba3e9176bb2f91","impliedFormat":1},{"version":"dd311b5e4f0b0d716a435526c715cbc7fc9e3b1922c1366ec6eab2da4de63a53","impliedFormat":1},{"version":"71d3a5b0a9e744b1ccfc3522954cde5dff68a6d9ffe9e142b2f9e76991d4f113","impliedFormat":1},{"version":"4db55870814528d3d643a9130589b8ea0d0addb82b2f4bd8a331fc7e5636dd8f","impliedFormat":1},{"version":"46f1bd13ffc3cff3ae73c62a892eff25b110adb2c38d9aba67feec3c90cc9469","impliedFormat":1},{"version":"b557d45c168a6a585beca8a23c5441bac7708b6a033ab6f8713116cea1118453","impliedFormat":1},{"version":"761c60844bb3c797e811a07a760fa44e61be419a1b37dfb710260cd5bc060815","impliedFormat":1},{"version":"53eb2db64558891b5dfa684e391e1c90ec47301a9f1505cf428f16f6cf7de5cf","impliedFormat":1},{"version":"82c68416e0136334a66275a03d68adfd7835c94cff3a33826b2bd20498db6ab7","impliedFormat":1},{"version":"6f879a6a80158e94c3d96259396c06beeb87fe863915dbb03e632872c664263f","impliedFormat":1},{"version":"85c296c62669d3ee0345e54a238426b30cdeed628fabc2564172188adbe55c34","impliedFormat":1},{"version":"2a36ce8ae2e791468c660c444ebdd93f2c22c5e1b214733217149b989d2d6eb9","impliedFormat":1},{"version":"de6a9d567a8ea5ca9ceb705c49abb5d5dff446332e3c77f5bd585e6bd4df8970","impliedFormat":1},{"version":"21e456a1960552dc0c81e102fb476854d09875e17a37b0fe52218683c9f4b3cf","impliedFormat":1},{"version":"c75691507d19b3b3892a809f0e51af04b8423f81963786b1e4ddd54b178c1f35","impliedFormat":1},{"version":"b9969f5526b4d78fb5f0001f8a2a654d61c7ce601535013e5416c4300bd9db7a","impliedFormat":1},{"version":"33803b2ad29051bd8c4d3ab9b1cd1d5a2603b1b48f03f21e62602c7a610a7ea3","impliedFormat":1},{"version":"624c1c00baca2051458efa764fb4537039b3ce88f1c79db7dd27033041b93e8b","impliedFormat":1},{"version":"b74233955e8b40025dddb55303ac2cb5189f22685b4f30dbf21ce1b1a5656ded","impliedFormat":1},{"version":"dba0925c06074e9ea94c9fce95887dd8bae26262e1522d52b769e59e90a311e5","impliedFormat":1},{"version":"d85050c0cfd804e9b849d9974741773e8577b6ae5636dad052f07b156972114d","impliedFormat":1},{"version":"99d81b8b1866cdfb815b440fb151fa1689b7c5f534d36aebe8065d902b3f9f70","impliedFormat":1},{"version":"8a589a19d5354a1cc02a28ab5084ac04ceeb00658a95ec089144b3ddb226e259","impliedFormat":1},{"version":"9bab286af000d35a72ce6e8acb45aa31fc89b8792ef937ca51566a996fc827f4","impliedFormat":1},{"version":"54d5e3ce769fe2154d98368f2f723c8003f04aab9c2067312fc515d8a20580e5","impliedFormat":1},{"version":"a37d93b145fb91cc40f6460a785bab157da27286ba72d0b50ef8119a5b16b85c","impliedFormat":1},{"version":"59ab362eaa15d882855ae715d8eb0cc757b7d5770ad97db5db4cd1319d760c2a","impliedFormat":1},{"version":"380220f859692be28024967a3df11402112f41cb9b029f02b7a4ac6e6176b037","impliedFormat":1},{"version":"ae482fa3b788e4923764c1ff3aad5d18157e0ffeb8cb0b012482721a0f613af5","impliedFormat":1},{"version":"11211b24dca9ffde7b9c863450b189b47e33f040f8264114b7761d07bee4c57a","impliedFormat":1},{"version":"96009cf527ddfe50e728ff6339f2aa58b687b678bee6a3848f6f7d56d8d34987","impliedFormat":1},{"version":"0e0649c37dc6e85592933fe6f5ad004ce1853e8d21bb6ac0e3f4f9c924e6ff2e","impliedFormat":1},{"version":"68d8782bc87568b260473a027c7c6e5417e1498a5f0a38037da23e2552474546","impliedFormat":1},{"version":"b6c0494560264d7dbbf8a495e37c709cf59cebdf7c5c03590e69eb743503bc89","impliedFormat":1},{"version":"1ec36a3efdcd2023825f83cc7e73756fa6572e62019375b727cbb393a7095e83","impliedFormat":1},{"version":"0a348c23db159a9d44a7ca1c310a59e05bb2055fe5f3a0283201fb5535f97deb","impliedFormat":1},{"version":"9003ad7177e5658410de57fd2ec0660da9a53250c30e8271f78b1e89a76b5d8a","impliedFormat":1},{"version":"dd5680f0e0a9865a764ba3c55face588247f30d5161a37a4e3b379323c4b7320","impliedFormat":1},{"version":"d1b784232e27adc314aa27fa529e56176bc0698dd17d6050e2c5fbf4d3e84b21","impliedFormat":1},{"version":"e618d6b5e3730c3456a83994fb9a5200ba6bbae3fd0d9eab87d06e18466fec11","impliedFormat":1},{"version":"1825c30debb44b2a193dfb2e2028599d222cef3160abfce320e9ebc6330b556a","impliedFormat":1},{"version":"a5a1623a24188a7a18679e3f89f2768fdc10c4fe17bb79598582c7196c454971","impliedFormat":1},{"version":"5e22d0ebd4e1c30152675aa52dece859cc8c9acecd7959c4b2b10ce8d051598a","impliedFormat":1},{"version":"8ab7225e3f5cba6478f7a4313d419b6579b631f5cea92f7ff0ee893513206216","impliedFormat":1},{"version":"f10acd2fd2f128bcf3f0828feead49d6155fc25300816af1471cd1f980137914","impliedFormat":1},{"version":"729569c50801e4ba94051cfb82b64c37f4598795270e434f0ee339c6248ae24e","impliedFormat":1},{"version":"3d2ba9e7ce503152d42c4478ac6bb424d78b42874259512d8472ec7674a72ed2","impliedFormat":1},{"version":"15b942189aaaa800907422b19c53d062eaf6f33bcf6d3a407e3a89054bab2509","impliedFormat":1},{"version":"a11fa6fe4e386a644807c6bab068dc731c1eedcac07584844d5b0885758dd00e","impliedFormat":1},"a0633c4ca5878ebf8a7c920bb23255f28dffec2221ac1caa1a9131f56fc8e7af","3a322b1df4c5f8d579132fc91a6cb0f22f0877912c372137d43ae17535eabadf","fc7eff52dbe79088e41197821d36faeacac986465cf7f4d5d3e6e86fa7751382","17e5bd67cdddc097bb196dc6e063c0847f6b5aadb5ea1b93f85d6a08a293a19d","aa2db4852cc05d028d34e203f25a90e25fd928baff7ef313f54e67eaa294a6b8","507cb1e84da778d241ced2193934fe7f7d48445b3d607a144d42cae7d7df411c","35bc7e7f8aad4f94338072466b5ba0295d7bebeb3fab14c0b1693ee54e571d38","55fd4014b41bb7d02c24374e8a6cc6287b6ba2b9ec3a5b007aba95f0175e553f","131d0abfb9e896469dde339fa981b87e8679e5ea35394c9a99dfd9899d6211fe","b544bc5a6ab39bfb99096a7ea7ee3fc7ef517afdf6e0fd7df7e99a4fa84d1d45","aec82f8607385c9cebb011f17e9c9e276c27e348950a7f2662119068c71da7cd","9d38b4c338ce7e0733e11e167d5105a1a7da1d252cfc9d80406f81709038dfda","d70cadc4041b88da7a0aee84cefa7cae897cbfe51cb5d2c9d69afb2c05e78ca6","e25b450ad30d959450bdd1c11355cc5b45e70a042d7da2a3c5d2c3c7d803517c","d655d0d0dc3456a383d8b82560d2317092b693b095627f1e5e4c273fd4035e09","b34ae463a3f2a1023928024b87d3fab423af40c5c255d2b767540c9e6c82f6d3","00bc00174b64dcc4d079546c86a6a3b0744f873a1ad8cb9daa16081015e84488","49df64732421c3535190d9d6643ee306d17eb4bea1862617b8bf1544615e3bda","b05349eeff92227a754a2a20eb8a1c79eca5b8bb29c6491a04630cef900a96ca","eb54ab0efaf468cfc07c4d7bbba4066218d38b593bc0918fa62a9f95326b8c4d","5b95836b7881961be04415dbea5d126b2f36b68f6a55569d1ee4264403cbaafa","4a443b8cf69e6b8958c68b51d2c16369f6bbd922d27f557779fae5298ea3d035","a3c52a8f26df339ff73f4bb113ad07588fb6d27de83433ce192e63f59294c649","772d7beed6392973d9909574890a6a7d2766df7b692ac2ff87cc92acd57745bf","8b1d2501da2332a74cfc884468d0b014e9be0bdcbf0e6bc4f6a71706a89d2fd5","be070d0b01f6dcb2d87276ce41530c2b855beae5a7b15bc8778ab71ceead85e2","2e7886b89b3078fe4d44dbc7a1ce50a23ecfb53a365f43aca717fbce7d1990c0","7fafe594af46fd6c5babb0b1d8834b2d3fa2ac31d9c5c87ed72693b8f8904b6e","00cdd02de6143ed46104460e1205bedeb1ff102bd0f523ece3a82ec42dd14065","739e076cd9202458df6fd40d19042790b5f3fe9e569aa8d13d6d5c13e5b0fe54","fa3fa5f5ce5aef0ef6fe39e153627fa0e44d3eaf8d64c53edfb79f3ce82f84c0","e062386c5211e305ddff28b3f00cf5b6fbf5355b4ffdec33d368c4d36513ad9c","29f6d232759aa6c88620f78a90e876d4402b1e711a7cb4554c58153728df72d0","36169334e83d2f9eee96770c73bfe8beb61defcf193cbaf1e5b1f22179d5b2fb","fd7b66f482bf137fa64dc2a92c7495555433d8b8caea528c0f40cf56d0fe5ac9","08218ad86aacd6ae58c59174a7b00028512a151bfc164e1685eff5b8a1226ef3","90c4b03e84942027024692cfbf7a02bd156903483d966c0d7810d40bb7635842",{"version":"ff2501a2fbf4cce9fb7c4bf8ed5a709d60f678e62bdfda44c26f956b6c0c91f9","affectsGlobalScope":true,"impliedFormat":1},{"version":"398d3b753d3741fcf0b069a6b8a1315fe527e328e84160119d81d8425f798c33","impliedFormat":1},{"version":"a02124c0ee850443005ca9a4800b743c1afed28f9752afaf8c95cac6baf83877","impliedFormat":1},{"version":"4db8335cb921acd3448288c991f721ad7f4939069946819221b8b3cbfe9b6c5c","impliedFormat":1},{"version":"554acf414793d442a5602055e4f8d0b83edbd0e4a977bd4a90fdcf7b0b1fd898","impliedFormat":1},{"version":"52e2d806ccaab9371209f3fe262252e44cb4f79592e21500559bc6261c534d1e","impliedFormat":1},{"version":"b123d189617fe233217ddb7c8c0fd06b4581fdef69e77d63728d3006499f33eb","impliedFormat":1},{"version":"3cd4c2711fd3496674d4267305e67b69a5a42caaeca1b45c294a1de5ea3897c1","impliedFormat":1},{"version":"117c9403487420a1fa5936658fee58de9d77c07b2c7510aba24e9dbf92e9b622","impliedFormat":1},{"version":"3c201db56028e893e4de5bd9d048bb804daabcf820be6bf96bb5905a0ffa1024","impliedFormat":1},{"version":"a29d69c75a5a7b1d451b30dae820b745eb7efb5cb74abbe546451b1185c8b339","impliedFormat":1},{"version":"f2bac29fb3514f46c0c1ea981340c674098aa74c5fffe1e7630d31c177686450","impliedFormat":1},{"version":"b5499e8d3e39a1523d4d12718f77f1e2dcfa3f825f67898fcb90a9edb680e43e","impliedFormat":1},{"version":"96fc3dae2f110377fb32c48acf3efcebffd12df01c798466287183ade087719f","impliedFormat":1},{"version":"a64e1342f2c32cfa1b9da3f5282b2185dbdd0989e30b48ad07962e7771244a18","impliedFormat":1},{"version":"cefc795bc727964f7ec2991703fafe975a0b265ef4938d342f4dbd93ed7a5f86","impliedFormat":1},{"version":"0d390748eee58a959b560b93da0d6261a1d3ff87a246f459d8b3e7a20391b62c","impliedFormat":1},{"version":"fb652d576e7c73b08eb6f9a4f322aa841c1b857195745e6ca436724c179de2fb","impliedFormat":1},{"version":"d192c4305f2add7ebbe22e9328f788b341fcb66e5ce4bd23cd2b1de097fe890f","impliedFormat":1},{"version":"925c28c5e11d57a08d05059a147f7a91c0e447ec27726dc8b79655fa1ff05301","impliedFormat":1},{"version":"8c4242fbbba473b36879fb5c23f29482335ab05e4150f06c22edae4e44c894dd","impliedFormat":1},{"version":"2681cb7b5767f12041662d1db8c0994012ef98c8b41d26d2388497adfcb0cc64","impliedFormat":1},{"version":"4ac4739a6edf9fbd20a18b5b675b08291fc860dbf89784fbd85f6f303df8047c","impliedFormat":1},{"version":"1840ac8a2b18e0982da57d80a09f5e5ec0e38d18afea7ac4ce069d9bcb3b3cb6","impliedFormat":1},{"version":"681c823b35bcc451c501382a6ebecf0b09fc792d83efa1279a005aa62285ff7b","impliedFormat":1},{"version":"cff0422eb92de48515743c3638bed6c73cd7d1312513df94030dc4c41090457b","impliedFormat":1},{"version":"d478539c608c8ec78b2d0a7817c29efab421e29d80b641ccaa074a96fb577f04","impliedFormat":1},{"version":"f97e2644e1e7763c6332e1067695ab3a2e51c06baab3985507da46a6e1200958","impliedFormat":1},{"version":"e3c8c01adb8d63c65f122778d8f63911437024ec3f4733622c510273ce3b8823","impliedFormat":1},{"version":"a12603dea0828662dc971e86e1169ec7b243a606e460a04ba1e01051c4f52f36","impliedFormat":1},{"version":"b86d0df4f4c8abcf28e629ace836c0f6423ea1509853178f56c6329b2a26ccfe","impliedFormat":1},{"version":"0e62d4ab3949b67c679fd23b39e55ed9f19597c0afb21d8ceeaacc4716ed20a9","impliedFormat":1},{"version":"04771a6db3f7b054afac1bb6d540d18efdbba7439415d4bbb759b8f39f1f5377","impliedFormat":1},{"version":"d0cebbf45fa0f4b492284e0be4b3cbd1610f05e33ed201ba8937b1c147bc974d","impliedFormat":1},{"version":"6a1b55618aef82ea35596613159dd7cd7805b07dbfcdc8fa288e41745f3ec98c","impliedFormat":1},{"version":"572fa17bfde079d0d5159c47702addc4f2e0060f8abb0437a5ce9d451473f53b","impliedFormat":1},{"version":"9c2971938ec0bb237bc330aeb026d82d1e7ed0da7391c8761263e717875f2b21","impliedFormat":1},{"version":"8db1b5e284bdd0df8797b1f70406cc7dd126587fca77be01e711910cd04103fa","impliedFormat":1},{"version":"31549213d7a9f3cf3aa96845b5860144e3900997771713c689d60276b4786664","impliedFormat":1},{"version":"822a8277cc73b8d96ce336ff56a1072c9f66485a64a562cc0f29cd7e550a87fa","impliedFormat":1},{"version":"a097e76e2b3a5a7ab5db2db9a5787dc4a3bccbc65228951c243fc0d58675467c","impliedFormat":1},{"version":"e996cc50e5bae651f0565e8499873d38145d8955e521e758426ba73758eb3bf5","impliedFormat":1},{"version":"8ad61067b3ba801965c04c2815c231847631a61c4da2b1987500b5aca6db161c","impliedFormat":1},{"version":"aadd40c020be82d01ba79caf35e1169bd3cd53bb6b999a4ddc5f00c9db847a46","impliedFormat":1},{"version":"f16df5990c987807a817d3d4218335095cf2783a1a7521e2871e64b8d0f6648e","impliedFormat":1},{"version":"81320fc91eea90e06f8781d5f6bd0d3990e0cc7a50e766a78b56e0a1cd44a332","impliedFormat":1},{"version":"224f89650a8724c67f36b98b5e5325d4a224cadfb9b387bf076adb76437443c7","impliedFormat":1},{"version":"36338d4f4ac9768967f2cdc092734373a3d0eb70b808def5222765825dcde534","impliedFormat":1},{"version":"3bc987bed2ad368f08e5e0b05148783a46621d3386c4a123af145d12082a3850","impliedFormat":1},{"version":"27ae73c6c31329e8268cfadf194e127a92b5c6ada1164693e784622576b94f81","impliedFormat":1},{"version":"0e5a227256596eb516def2d3ab823c2321cef34c28cacbb559c924b2374143e7","impliedFormat":1},{"version":"718d456c2624bdff0b7683ed67041995519f657b98f52b7890f11cdccac36f89","impliedFormat":1},{"version":"4b2e887e533849e74020b1c594604e990dd8fb3abf693b1d82c96d5079b27ea8","impliedFormat":1},{"version":"2f4f0059c74e8ecf9a5e962c6a8fc3aa258941dfc18343f50e2efc2923ea5c56","impliedFormat":1},{"version":"92e0c20c54604feb984ddc519b56460c61dd9b285fbc30174839286545ddf848","impliedFormat":1},{"version":"54a336776a1161336928376c78fcc9deda2b5890f9008631c7aea700b6727bb5","impliedFormat":1},{"version":"14d18076cf79b3c6ff515123a71836644f50c2956312a2ffc960028111489316","impliedFormat":1},{"version":"632e5af6af4bc7c3977dd4782ad03b37c0229806de4eec9666fd79841b6a68c0","impliedFormat":1},{"version":"8c3e1c25eff5752f6642204351420c99844c1b2a73aa0dd5f81b315cf38b32b0","impliedFormat":1},{"version":"2e51565212c8cd03202a9492d57e93c431041114762dedf69ac3be0f62e7fb20","impliedFormat":1},{"version":"06f894fea5d5bb81048440482e750f7cbd4932cabb95e4d485cb0b9be1d3eeaa","impliedFormat":1},{"version":"b617019b6a719ce7a920e1909f3e62be8ac6a914746667bcfe10d8f350cc7089","impliedFormat":1},{"version":"cecf293195c298e093742c82e5995cbde08af76d41f9440224de7f83e077c4aa","impliedFormat":1},{"version":"575ce340b9a859c0589a04305dd9846184710eaee71a4effd071d5d406377ffd","impliedFormat":1},{"version":"8c08e433b7d3a998a2cadff9d54e3da445c26a13815f0c2fc8e7f3fdc7cbd1db","impliedFormat":1},{"version":"ed872db0e2a3622d6d92d9b110b7165d8cf23d44b6993783328e0088fdc6a33d","impliedFormat":1},{"version":"e34adafe9efbbe6d7af7e346ca7df8bb2e77a3a851d8207ae6199357b903b192","impliedFormat":1},{"version":"42b2b30e00d58b57a4204ba1118422e1bf83634816f950869e7454472fab4e35","impliedFormat":1},{"version":"958fc2e0308e04a48b1f3a793d66aaec672278fc1ae0f31efb89febb84dac1a9","impliedFormat":1},{"version":"4e771fb2e12b05ef96d1a215adfd119643c057ad3e97739f85d1d7533a18caf7","impliedFormat":1},{"version":"02ffcc56317b8d9ee19f209b7cd8e037074ab508a1ad06754a2b1f2e77911f66","impliedFormat":1},{"version":"ab570c33c53acbc83ad2e24433a433fccf12c28389271cf3f5c44b871f547b2b","impliedFormat":1},{"version":"8d6c8608ae5cde40483ab309156f5db32bbb2ee86486f7cf0f1d85733fd39f53","impliedFormat":1},{"version":"f4529b8473a9022e02fc7a4b5f92e93659d1874809f2c7b38fc367441a93a339","impliedFormat":1},{"version":"b92c58600fd18c32ff687b783eebfd0796cd995e5965a86ca17275b523d1fabb","impliedFormat":1},{"version":"ac46a79d9cfb4df1f024d98c886e4a47ea9821a2a467e4cc9623d96b8f753766","impliedFormat":1},{"version":"ff8b5fb2bca480984f53b064e53a0f3ebf4400c0a1a4b53cede4f77de95002ed","impliedFormat":1},{"version":"ab1a99b4017155d8040b5456cba7bfef33bb767da1eb8e4ca369d5185810f349","impliedFormat":1},{"version":"0c4286c0a84c20e05c1f5cda5add26956e696aa817b33d45ee2458eecb3ec375","impliedFormat":1},{"version":"eb155438a82c3e7228cfda102f1d6e1ab4652aa83cb8ca01d8afeeb782803f1f","impliedFormat":1},{"version":"1f0012e2fac75a6ef2406eba7a9ca9ea16c553947583d663eb726c97a26880c3","impliedFormat":1},{"version":"54ec65aad2d7775fab779d01763bf55d7e768920d68f7a05946901eae49ebbfb","impliedFormat":1},{"version":"ae1099212ffebd47c3f0e51162fb0c1e5d4b104421b8a66edddbdf920899334d","impliedFormat":1},{"version":"9cbe0b736b34de9fcf54ba1db60133cfcffd413bc87ad008384ec6442d4ccc14","impliedFormat":1},{"version":"3f713c2dd9b26d5e3e475c811a8d7ce219f1346cbe46dad4596dc6e1d8d35cf7","impliedFormat":1},{"version":"d538fbbf8fd0e073bb11279bff9a37deddbd192513362737f98cce00f2fa3c34","impliedFormat":1},{"version":"a7d869e34e5b3201695b1fd231884d163cf41d24975e1e6a407eedc31d7b9efa","impliedFormat":1},{"version":"d5b6042c1806e7f8ef08b9be9cb72ee50cb7b991a28efbda30a61434b1610216","impliedFormat":1},{"version":"8d30f52bf78ba0b0435286cfa393e2f62077d64fb9536eefa9cddd62c1252884","impliedFormat":1},{"version":"431e627e173edc76477913c8fe99dbbb96f8fde04aab57f133553f359d4eb20e","impliedFormat":1},{"version":"2d37f551e55c0ecdc324b7af4d2ce823026a3109f64d8c4f5d9fb050a01460da","impliedFormat":1},{"version":"dc6e89155b11ff755ade6d9eb497d247f1f7beaae87f64332e2c1e5ef05a3cec","impliedFormat":1},{"version":"4e2fb39d94dac2785eeac18a5d53df1f73bdbad18900fa22e46bef5cb3ad789e","impliedFormat":1},{"version":"734e259be474d6c6c2b63e161974d600158d3defe08e78f378301fff846413d3","impliedFormat":1},{"version":"ef71f578ad24aa892b5f52e9e5aca43fa56434ec07ce5d62423a6499c15708f7","impliedFormat":1},{"version":"176d770c6577804c34df935fa0d0fc3f60396ab125fbf20d95698e35c08bf077","impliedFormat":1},{"version":"314c4b1b0b4977f9f55a5854a6c6effdeba1342edbbb89e7492e550cc38ce4cb","impliedFormat":1},{"version":"b03941a397cbaac2a62ea7094ca56acd26e90d73865d750788a439c68c75dfdd","impliedFormat":1},{"version":"67d0d710465d9f4e26c3e55865d110596b95e84f7598164ad3046345f422931e","impliedFormat":1},{"version":"34e8ade0345445320e23a677a1011f78efae36e8653446fda313b38957865dfd","impliedFormat":1},{"version":"abb451358f4487f36cd7fe07120948ee6b700f0e7f5ad07591bb63c2b5c79b71","impliedFormat":1},{"version":"a02935c93c12620b4602f0adc823f20e6e1846d3d7a7a8bebfe303794c9671e1","impliedFormat":1},{"version":"465d478cecb928a2cd1b3ba3a07d1a2837cf3779375d5fa42fb46e9dc1fa4a76","impliedFormat":1},{"version":"42843220b0536af6997a420dcf33a0be817b4df50eebd20d1fe154cb6e8b7cfb","impliedFormat":1},{"version":"a0df4b1e4af6007211dbd710098e3ab753b119886c94ef877730644c66c166d7","impliedFormat":1},{"version":"91e683674172f9b5318021598e99f907dade21763a157a204faa7590771fe353","impliedFormat":1},{"version":"d45edcb4ac7d734f90d9624f33b01c54106bdc39c659ea6f36a0f7e935324f90","impliedFormat":1},{"version":"a7187425d04412674ff5f311012c11b459860cefcffa463b286068edd43c0776","impliedFormat":1},{"version":"b344bed7a9efe077679682e1ae59030c40b377cdb8d9995b024ada7c6ed61eb9","impliedFormat":1},{"version":"940b599dd43946b484a70368fdfd5fcf240e5732cfac5e08564b879f96470eae","impliedFormat":1},{"version":"55da79a1f289545c2d4d5fdc56f797a18074e62c252b8472d2680f592a159310","impliedFormat":1},{"version":"4123ddc1f4f7b4652c4da7aff4807605ad15279f100341ef4f5ea10ce8fd58cd","impliedFormat":1},{"version":"e3dbafa3bd33f2155464db1ff01bbc7a54635625d3266ab3632a9db13366faf3","impliedFormat":1},{"version":"71b2749ddc97ea329d22f141bd0d2231dfbcfd0c77b673fc54f22a2560d570d0","impliedFormat":1},{"version":"d8bdadb41912e8bd858a06c7550282243d2a4b8c7e9b827ef06da11068914218","impliedFormat":1},{"version":"7047200b3de005a6e8da3175f01387c228830644aec97c07edda000308991eeb","impliedFormat":1},{"version":"d99bece612c3ce15c8888c25df7e0444a338049cdb815d3931fa85891cad7027","impliedFormat":1},{"version":"a8e3e304d5584080c99e081ccce819d05ee369062c2f0e1c174560ed2cc2c034","impliedFormat":1},{"version":"09bef2dc3067c333c109f7b7b79bde43b843335b32d8ae51e712778941f89d68","impliedFormat":1},{"version":"aaf6132ab602e07ef329ae6aea0f4b976988e7d88432d4309f2a9fd0e6d9f895","impliedFormat":1},{"version":"dea409b891c81c9cc218cbabf4964c43c6cdf247aded796b3a30fa3f0f14593c","impliedFormat":1},{"version":"82aa855a9e602dc5203cf517582f06f2f1bc4e4a43300433d14dc2e909408880","impliedFormat":1},{"version":"92fa6c066987a4cea71a0ffe9fbfb683b45b5300ae9f5584b02592f3a27b3ed0","impliedFormat":1},{"version":"a5c018512673b7e1ff6cae34d14713e89e94479fff33c14696f7e2153e4f4755","impliedFormat":1},{"version":"e459c1d4e7623343476da01e7e4edf8290bca1f1315287559137af5557f3ba39","impliedFormat":1},{"version":"5981c27079aeb53fb96829328f014ae7a5a690cec8b1c93815bc23e6fe7189e7","impliedFormat":1},{"version":"2b69fbd1f361e82dfe9bbb786133f0b58845c79d7094fa5790306e5ec271e5bd","impliedFormat":1},{"version":"c10c88f1daf9fda0323c9205ee7a0fd63ae4f67320d3b673468242d89061a459","impliedFormat":1},{"version":"a68ae02c58a9b6ffb29eec100c886ce8eb80201e454fcae79c299bc2db0b37d0","impliedFormat":1},{"version":"d764056449904a73c1f2c6f8c2ae79edb0d1cc799eda5fc3a60a30fa97b94749","impliedFormat":1},{"version":"7e73db72fa480a32afd616f2ab23edb4702316c7b898bd2ba6b5eff6e8ab9412","impliedFormat":1},{"version":"916e84931e102ae5091d09c1ac5aeb2cbf5458f11e0057b23157f5c062254999","impliedFormat":1},{"version":"226d624e4776b837abb8c1eb775f27fc265d7ab4c7473bb48f39c535cac94285","impliedFormat":1},{"version":"4173e4d951eb16efa7943068fcb21aea81bdf4c996dd047ee78625874836dad7","impliedFormat":1},{"version":"9c219a351e0e80e556380fb3372a3fd2c54fa3f1bd9574710ab4e577ea26063a","impliedFormat":1},{"version":"ac18a2d24df81dbbb885e1601fe94fb9a7ba42f04c98df04d16e69f4ca9ee9db","impliedFormat":1},{"version":"8a9b3c96ea397dc289581c1aa4f045cdd2f8a55fc5d917c56d40370a83eedc5f","impliedFormat":1},{"version":"5b289d52c1414fc6737fc451b85fca5f70ead22c2294f5a9484ec1ffbe233a83","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"95f50c554cc9bd129a38fc61cabf1b0b56851c3b4db34177c909c2a03d3e2b9f","impliedFormat":1},{"version":"a2b35bc1378fbc1443e1678fb3ab6e8023269500146537b5a098e8db214327e2","impliedFormat":1},{"version":"4428a7e681f6ee541f90b54d6422325fe26263c333b27293203ec8cf98da4dae","impliedFormat":1},{"version":"cf25b77288f29a84be0a436ea2f5b8cc00bc06b6e142ff975f60a2252a6fc18c","impliedFormat":1},{"version":"9fbd375bb1f6ca5490ddc37165bf761f2fe89d93bd0de57e5bf3dd12cf94baf4","impliedFormat":1},{"version":"fc291372c7992060d4222381491902295756466f44fbc6f0889a6d4e28d0b937","impliedFormat":1},{"version":"6ca9bc3ae7c4fabade7fbf2659731cecce54a745d286d69755fa2496c545456b","impliedFormat":1},{"version":"e6a2e3b9fe781537a0f91c301de9961bebdd4b33187f98a582d694177acda462","impliedFormat":1},{"version":"27316752e6998304fd7db2d67ba0b26d8029b6f503b3ea89efe845c39fabf9f9","impliedFormat":1},{"version":"4add6412e18d83b5bd7c65dd07c3a1544bf6b31baa22473775ce967d685aca27","impliedFormat":1},{"version":"8a7d6fe5fbb7e37ebb0bb81241d59c4a806cbda97a5f1f15af3fb9c903672598","impliedFormat":1},{"version":"74efaf42f5842c4c4fb398d2f74d59a86792320c53df9f63dbe00ba8b24b88bc","impliedFormat":1},{"version":"08603c7d3cc9cecd1ac97cc1baef2d90b116759b541eb4181109bdabc64788a9","impliedFormat":1},{"version":"64068fb5c2c88a2b7016d34b02b03582e759b3f0ffb89e9e07f968838275a564","impliedFormat":1},{"version":"1825619ec278edd94785af65ae589289792cc6db662f63adfddf2a79f6bd4233","impliedFormat":1},{"version":"3c63b1b5ea4b20822405d6b7a8ba96429ee2b400367b61c82507f80be6202289","impliedFormat":1},{"version":"cbd2ae79794fd7bcff66e343af71f7e174a6eb95f17a8742b1ff62c8482f7cf7","impliedFormat":1},{"version":"e36a635d498defc980ef7b96eb5cfaddc11eeda2bd0de7ae249adde06b2d168f","impliedFormat":1},{"version":"5ebe388148630bc63525fbf1f408370c657ae888abe179001e64711b0dc8d663","impliedFormat":1},{"version":"cda3e014cbfab2a60d7e598a50f90dd30c7fcfeaa73a60dd916507bd9965260e","impliedFormat":1},{"version":"432ba4ec869745ed9de5ba6a12c76549dd76ae0a146faf0bfdf35ffd4a4e6ea7","impliedFormat":1},{"version":"a88437446e80a492b1c4d3f5c9fff5d80b5c5e52754cbb3eb2cfee3d3690ca94","impliedFormat":1},{"version":"bace2dc66c954f2a81c641fa9f0dcb1b36ddbc6db3635ea446ee10c47ada15f1","impliedFormat":1},{"version":"c5c7f25f198dfc5ffc62fe2e8ef3f25647bf21070a5f05ac200748c83ab7da4f","impliedFormat":1},{"version":"60390e7b89c19d160b3bf2c854a9e06414d001debd9947a5db54623004a4be0e","impliedFormat":1},{"version":"5909d075463236fbe7fd8a785248203f6f32d73b032c774d5018adc3c44846d8","impliedFormat":1},{"version":"c08e7bfca5a8bb244cad7689ddf7546cec8a5bc5367b18bcadc0628ae927f797","impliedFormat":1},{"version":"b7506549d0f8ea4c74e4b4b4263932090578f193cb37bf719b44c5f149a934f6","impliedFormat":1},{"version":"992aafb2a060c3e2099941c7128d88aeb9bf8f5fcc594e9fe561d19003b5e4be","impliedFormat":1},{"version":"9874f63b3f3167f344d2a30047722b409e2915a502d9b9a50a91ab1a23b49623","impliedFormat":1},{"version":"b55dfdbd1e893c0b6cf91dca75395f4bd8aab8e624007f9fc70d650d8b340137","impliedFormat":1},{"version":"1740fa9c57b951441b1db4478a7f6a82ccec9de1de650920cbce74ed10e08eba","impliedFormat":1},{"version":"6948d2c91da770f73b9a6459c3daf8ab23d80bf7b70e215551ca3219ac041b68","impliedFormat":1},{"version":"32dd24f732e9efb063532a95a730e6bd68b7c8e54a3d4f026fb4a1fa71d115d0","impliedFormat":1},{"version":"e39c146a2b8a3f48452973628042cabc94bb2893488bd6a79b3e04cfcd89c729","impliedFormat":1},{"version":"60f5165cd2492544cf497f3eb4e8a75fa340185b4b98b8aa87b62853d57d1549","impliedFormat":1},{"version":"fe9cc3f1d04297f8d6995789f4df2b531a1ee7f1d0c8add6371281f4a31d195b","impliedFormat":1},{"version":"66b9b5e8625e6ada62c4d070918350dd10d01fa260426674448b25ffc7009488","impliedFormat":1},{"version":"0d25032744f0015a340edeb2e84e685a4c79ee1c9066d761d7fb0affbc2dfdc3","impliedFormat":1},{"version":"dbade843dd62f6241d1a30d944dd80152ddb554ea86a2778caf0f3ff990402ed","impliedFormat":1},{"version":"c5fe75259bda7aba824205a9138ea7f3bbc47d20ce777cea79d40930685b6ac8","impliedFormat":1},{"version":"3d485a48053321817c3ce51afa41c07b180b462274551d53c5a4927a5d052038","impliedFormat":1},{"version":"9e2f9ee99f0e172ef91af1d571e09743304b3b2430d41a8bcab357b878114757","impliedFormat":1},{"version":"5d6257ebe252d97b3d6fe3e0a49a0f148cd7312849f5f1d6f6b7265d3d72b5d2","impliedFormat":1},{"version":"2c60950709e37e95cc5dfa2ca27c5da53521ee09c254f894f8d91ae8717e7885","impliedFormat":1},{"version":"8bfc090ffec588f44eacbd6714f798a8a0c3dc1b02855f5e12e567b4f161b30b","impliedFormat":1},{"version":"b302d3e1a806fc890c324ebe90dfe07a780e973267c66bd159d0dbc1f6e3d055","impliedFormat":1},{"version":"b1c627fa2a4cc9199f937f4d35ccfdef2efd6ef40d5525ffd384acb29cbaf66e","impliedFormat":1},{"version":"5211c6778aff2c045d7fb44236ef9b91ab424ee916f6575901dd0dcd9d58857a","impliedFormat":1},{"version":"39959ee712b3455499af3b1c95bbfc9ea59d584d5af2b01dcde120fe5dc6fceb","impliedFormat":1},{"version":"bc27582d90eaa5a793cc4f3e27acff890eab95641431c263144f3162bbd4a8bc","impliedFormat":1},{"version":"2992d19be476415c0296bd548764c20fc9cac2876e45abbbce23dafbd65438d9","impliedFormat":1},{"version":"dc117b16848058e94c39b68cddd38b36be885a63a0130097e6e992cce6ad9bf4","impliedFormat":1},{"version":"11bc3d6606ca11c982d848ff3946f1d978360e7861dedd8bb97348a21b4a8ad7","impliedFormat":1},{"version":"989b88698577f76069fe791b224d2157a0205aa2b029718dfd386b7b4706fa0c","impliedFormat":1},{"version":"fab62208329b9bb74dfe558a6b05f802bceda19274c763efd8ea0b47cb68925b","impliedFormat":1},{"version":"b7126f4bafda5e7424504458d53b298d92139de65e34859ee67f6f8f1d43bfdd","impliedFormat":1},{"version":"34e922a2625a8cc2f2c586afc7f7068cfdac27bd5119cdfc10f8e22562b2a142","impliedFormat":1},{"version":"d57be402cf1a3f1bd1852fc71b31ff54da497f64dcdcf8af9ad32435e3f32c1f","affectsGlobalScope":true,"impliedFormat":1},"125096b701b584ba96624c640e90b98fccb17e4f5806d70b1d579be0840f919b","3388357ca7760973a4f320636f575812458e2d2b1fda30667e61b4f4c946afe9","d958b523671c3a6ad61b27ba92c75fe95647a09ca8363d227668f38cb7c316c3","f4d0f59bcc27596bcc34bad5f2179a1e3180da78acabc42df49e5020898c175c","d89db70e7026b0d949ef236b6f1d5be4a3c95705af91dffc617a59a691d291bc",{"version":"53477a1815e915b8c20222a2ac8f9e3de880a1e8c8dbf9dae529b3d2e2b4a53b","impliedFormat":1},"b0b9dfc5fffb715ea068acee41e321c6c33eac892f65263cf6d3ca9d6f033930","cf6fbcdd696a4f296b1ca64b6dcee6ceef7f30b75a76dd5775af6dabb47628e6","e530697b10a31bf6332e6f570aa728c114cd47c0780aaa7dd50ad1b5e50f8a2b","9ed6185c39468bb8899f8f600b01834704311f1e8c06b63ca3440ace3c728426","1e80ba6a3a82da3a0cb3370f79461d40225d8e66ffab3d9159e328e32a1f9a2e","fa19ff54cf0839a4eade769ca75a9607e7a9e313e4b3ab11a1b13739a65f9086","48b1a92cfef5c2e3cd5ba8f67dddf6d4a144cd40bfebfe4ec8cfb03ac9567ead","316a7004280cdb9275dbf4030bcb44ba68bc05960d371cc66041377198a50956","535e12cd6371c94d30bfe7604d2ab578e26ef1a788213f6194e59f060e546f9a","f65e09a2e5b2aacc52ae5d377bf085a264aa639b828b5fcbd73ee94fea9f3511","69d22ddd4f56519c0d8c12c786dfaf61b5c3d398f9b6e600b2974918cea0e247","e8055139ed9ddca8149d4ccd6a2b884300d8cefb9b7b8b39caafb58082fb4f6e","fd2a625362ee9ba68682417ab7da3adef39244f694b27dedbad8f6f05b622bfb","6c82fda41297adf58c67b41940534d842cf26cbb45178516872bb0383bec6b3c","c740aa1c17a39da8e96a51e5d48e0a9eb3f53c7720b44f476bd1dd66751629ce","aeacd131f543336365e4214883c8d49969a99918808c57bf4851ba4c46016d2a","9ef958a73040b66dd8b56348b0570211734faf1c1a4e59e131de477850230cd8","5e0d7992dc094b620127cf36e0a69bd7ba2fffcd7194ec7928b4d62283f8f711","709e2ca6dc9608236fabdcb8a042cbeddce243ed3758280d40a088e37de90f1b","de35364a7d8a1a504db3e2d92e8f6a55e0bfa028531d315846d2b8c61f6c69f8","2973f8155f28be2031fedd70efddafb0e79483f44097b7414ee52ea15f4df654","5ab4002f27938c82a8163fcd2295bdffcb8524a74c4b5a3dec215992be66c08a","2aa69c6b3c465c5652238c68db5a65bceb1d32bdd1e667e6fe9cc5132196ba98","acb4949ad029833499f0d509e0a415a247921be9e8ac5a18a5db3a9a4614c4a3","1d1c4fb86a3c0e37f4733519a37b7471cc3ee42d310e2bd5c61db202bd1c381d","7909d395a0a16a3e84a5948c4bbeff5f1e7230dc92090e5ee4f10a736166db1b","96e5be6af3096b7526ceeb751f425f3243bc92c834012472b86e1dbf024ce4e9","330fc45f05f5098d77eb088e3c129822acd15483a2a073b5b5c4fc5242d77110","3e846a16bb203f7ea02a364cf5a2b7f03e9a2ab21067b9a82a0d6cd2419379ad","45a38422d77e2298b34e253fc3da6b2302f04fe71e87c38ec9c65a79a2973510","8c440684c3845d829916f1c75b983f69263707d7bf1db14aaee75fd890dddb26","06258dea6cfebbe810e1bafc9fbbb891b1c8370339562a5d2d65c9b34f5caa92","38e7f24c7ac2f3b8025ee5240b7ea26f219e7684b1b856ab4b938871e77a7507","07bf5700a763e91760556a4b524c4d6f1cb1e4d7e4c5b5d23dec8ff7ce40f383","93ef149f2d05336a0445409dcaef93ac7574f3495607ef32c2065a88bf0baf82","8da774ccb5d4116c0fed80280e54673994aaf3bcbc1b853ae794eae1727c3857","9645ac77f2918604def8c3f4a813b928d92d3b93c414e72303ae13704781e06d","941a56c14cfad99b336accd81e7091be0f76b3a2b10641ecae59970e32ed10dd","260dcd321df47025bcc34102e2b49d899c91ba0f871b6d11ba4a9abf02f2ac11","19015079f3d31031d83fe58e417e91b2c0027387247e88878ce8baa677975f34","25e5cd00a370eaf0361a475b501b0d0e89ef6d2b0ab2a9bb53b7b4ce8a8a86e8",{"version":"2bad09c4dc0810666ef5b6150aa910dd711051ce5f2184050c9859c708092a36","impliedFormat":1},{"version":"eece99a6cf69ff45c5d4f9e0bfb6450f5c57878d048ff01a6a6343cf87e98230","impliedFormat":1},{"version":"f7ab1fe738bbe7fdd1e9bc9887f55ac0d7eda0d234a7eb35c77304430f7d6715","impliedFormat":1},{"version":"1f763e81e71ac8ce0dd1b82e67cd9d1d2e56e77c844b46349e1623c82df9b130","impliedFormat":1},{"version":"1a9c0db9d65449e9dbcbf23baa3b8bfa48806cddb2adc5e172eb8eff5afbb702","impliedFormat":1},{"version":"4dffcfc24d378d52ade01a1ae0c5baf6dc7a50c35f8302be34f06b9eaa5798ce","impliedFormat":1},{"version":"6586eacd77a813c50d7d2be05e91295989365204d095463ca8c9dfb8caac222d","impliedFormat":1},{"version":"3f2b3c5d3f5fd9e254046b9bf83da37babd1935776c97a5ffc1acfce0da0081e","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},"4e747630949dcf63174e6239bb7c4877bc82b844bd3485008f4efb27c8ca592b","d9b823121178c4ae6811c661d9cdd070325c056bbd952c97d019bcbc8cbe6e29","4ebe9a219e78f9aba4e3ab4d363da3b59e29d4c7687aedabbb9c3cce3b1ecc73","45865a24bb817d318b9cd3d4c7b606e9fd75dea59471a51ae41896da87f8ac28","6a96ebe6851d842de17367529fa1f79ec5c840543dccdbc12d90db55c47ae632","968e4f577a42a1d67bec67d0e18bb73ed21c1848cfe6663e945b7005a6e88f43","9c68879cfc4fc60d9d71f4e367ad7a1e9460a9d970152902414a131248034022","334d81391c2a34919ef9fd4b0b833f313b6cab8dda58b1a1b3fbf8c9dd75df47","2d740d7bd1d6a30de357fb10f68e3312265bdba454ec9904740b6791750e7a7d",{"version":"cdcc132f207d097d7d3aa75615ab9a2e71d6a478162dde8b67f88ea19f3e54de","impliedFormat":1},{"version":"0d14fa22c41fdc7277e6f71473b20ebc07f40f00e38875142335d5b63cdfc9d2","impliedFormat":1},{"version":"e1028394c1cf96d5d057ecc647e31e457b919092f882ed0c7092152b077fed9d","impliedFormat":1},{"version":"f315e1e65a1f80992f0509e84e4ae2df15ecd9ef73df975f7c98813b71e4c8da","impliedFormat":1},{"version":"5b9586e9b0b6322e5bfbd2c29bd3b8e21ab9d871f82346cb71020e3d84bae73e","impliedFormat":1},{"version":"3e70a7e67c2cb16f8cd49097360c0309fe9d1e3210ff9222e9dac1f8df9d4fb6","impliedFormat":1},{"version":"ab68d2a3e3e8767c3fba8f80de099a1cfc18c0de79e42cb02ae66e22dfe14a66","impliedFormat":1},{"version":"d96cc6598148bf1a98fb2e8dcf01c63a4b3558bdaec6ef35e087fd0562eb40ec","impliedFormat":1},{"version":"f8db4fea512ab759b2223b90ecbbe7dae919c02f8ce95ec03f7fb1cf757cfbeb","affectsGlobalScope":true,"impliedFormat":1}],"root":[438,439,1591,[1602,1614],1690,1702,1703,[1718,1740],1760,[2240,2242],[2244,2248],[2266,2269],[2271,2278],[2418,2420],[2498,2512],2537,[2567,2593],[2621,2626],[3113,3115],[3118,3155],[3165,3172],[3188,3194],[3197,3217],[3219,3251],[3299,3323],[3595,3668],[3769,3776],[3783,3816],[4023,4031],[4315,4351],[4550,4554],[4556,4596],[4606,4614]],"options":{"allowSyntheticDefaultImports":true,"alwaysStrict":true,"declaration":false,"emitDecoratorMetadata":true,"esModuleInterop":true,"experimentalDecorators":true,"module":1,"noFallthroughCasesInSwitch":true,"noImplicitAny":false,"outDir":"./dist","removeComments":true,"rootDir":"./src","skipLibCheck":true,"sourceMap":true,"strict":false,"strictBindCallApply":false,"strictFunctionTypes":true,"strictNullChecks":false,"strictPropertyInitialization":false,"target":8},"referencedMap":[[2998,1],[3081,1],[2921,2],[2924,3],[3008,4],[3088,5],[3003,6],[2997,7],[2927,2],[2931,8],[2929,9],[2930,1],[3001,10],[3000,11],[2999,12],[2996,1],[3004,13],[2928,2],[2926,2],[3002,14],[3086,15],[3080,16],[3010,2],[3014,17],[3012,18],[3013,1],[3084,19],[3083,20],[3082,21],[3079,1],[3087,22],[3011,2],[3009,2],[3085,23],[2922,24],[3007,24],[2923,25],[2919,26],[2920,27],[2917,28],[2914,2],[2915,29],[2918,2],[2925,2],[3006,30],[3005,2],[3989,31],[3817,32],[3993,33],[3992,34],[3820,35],[3821,35],[3822,35],[3823,35],[3824,35],[3825,35],[3826,35],[3827,35],[3828,35],[3829,35],[3830,35],[3831,35],[3832,35],[3833,35],[3834,35],[3835,35],[3836,35],[3837,35],[3838,35],[3839,35],[3840,35],[3841,35],[3842,35],[3843,35],[3844,35],[3845,35],[3846,35],[3847,35],[3848,35],[3849,35],[3850,35],[3851,35],[3852,35],[3853,35],[3854,35],[3855,35],[3856,35],[3857,35],[3858,35],[3859,35],[3860,35],[3861,35],[3862,35],[3863,35],[3864,35],[3865,35],[3866,35],[3867,35],[3868,35],[3869,35],[3870,35],[3871,35],[3872,35],[3873,35],[3874,35],[3875,35],[3876,35],[3877,35],[3878,35],[3879,35],[3880,35],[3881,35],[3882,35],[3883,36],[3884,35],[3885,35],[3886,35],[3887,35],[3888,35],[3889,35],[3890,35],[3891,35],[3892,35],[3893,35],[3894,35],[3895,35],[3896,36],[3897,35],[3898,35],[3899,35],[3900,35],[3902,37],[3903,37],[3904,37],[3905,37],[3906,37],[3907,37],[3908,37],[3909,37],[3910,37],[3911,37],[3912,37],[3913,37],[3914,37],[3915,37],[3916,37],[3917,37],[3994,38],[3918,37],[3919,37],[3920,37],[3921,37],[3922,37],[3923,37],[3924,37],[3925,37],[3926,37],[3927,37],[3928,37],[3929,37],[3930,37],[3931,37],[3932,37],[3933,37],[3934,37],[3935,37],[3936,37],[3937,37],[3938,37],[3939,37],[3940,37],[3941,37],[3942,37],[3943,37],[3944,37],[3945,37],[3946,37],[3947,37],[3948,37],[3949,37],[3950,37],[3951,37],[3952,37],[3953,37],[3954,37],[3955,37],[3956,37],[3957,37],[3958,37],[3959,37],[3960,37],[3961,37],[3962,37],[3963,37],[3964,37],[3965,37],[3966,37],[3967,37],[3968,37],[3969,37],[3970,37],[3971,37],[3972,37],[3973,37],[3974,37],[3975,37],[3976,37],[3977,37],[3978,37],[3979,37],[3980,37],[3981,37],[3982,37],[3983,37],[3984,37],[3985,37],[3986,37],[3987,37],[3988,39],[3990,40],[4022,41],[4020,42],[3818,2],[4021,43],[3819,44],[3901,45],[4014,46],[3996,47],[3997,48],[3998,49],[3999,50],[4001,51],[4002,52],[4003,53],[4000,54],[4005,55],[4004,56],[4006,57],[4008,58],[4007,59],[4009,60],[4010,61],[4011,62],[4012,63],[4013,64],[3991,65],[3995,66],[4019,67],[4015,68],[4016,69],[4017,70],[4018,71],[2478,72],[2421,73],[2424,74],[2425,74],[2426,74],[2427,74],[2428,74],[2429,74],[2430,74],[2431,74],[2432,74],[2433,74],[2434,74],[2435,74],[2436,74],[2437,74],[2438,74],[2439,74],[2440,74],[2441,74],[2442,74],[2443,74],[2444,74],[2445,74],[2446,74],[2447,74],[2448,74],[2449,74],[2450,74],[2451,74],[2452,74],[2453,74],[2483,75],[2454,74],[2455,74],[2456,74],[2457,74],[2458,74],[2459,74],[2460,74],[2461,74],[2462,74],[2463,74],[2464,74],[2465,74],[2466,74],[2467,74],[2468,74],[2469,74],[2470,74],[2471,74],[2472,74],[2473,74],[2474,74],[2475,74],[2476,74],[2477,39],[2479,76],[2497,77],[2482,78],[2481,79],[2422,2],[2496,80],[2495,42],[2423,81],[2486,82],[2494,83],[2485,84],[2487,85],[2488,86],[2489,87],[2490,88],[2491,89],[2492,90],[2493,91],[2480,92],[2484,66],[2216,93],[2105,94],[2108,95],[2109,95],[2110,95],[2111,95],[2112,95],[2113,95],[2114,95],[2115,95],[2116,95],[2117,95],[2118,95],[2119,95],[2120,95],[2121,95],[2122,95],[2123,95],[2124,95],[2125,95],[2126,95],[2127,95],[2128,95],[2129,95],[2130,95],[2131,95],[2132,95],[2133,95],[2134,95],[2135,95],[2136,95],[2137,95],[2138,95],[2139,95],[2140,95],[2141,95],[2142,95],[2143,95],[2144,95],[2145,95],[2146,95],[2147,95],[2148,95],[2149,95],[2150,95],[2151,95],[2152,95],[2153,95],[2154,95],[2155,95],[2156,95],[2157,95],[2158,95],[2159,95],[2160,95],[2161,95],[2162,95],[2163,95],[2164,95],[2165,95],[2166,95],[2167,95],[2168,95],[2224,96],[2169,95],[2170,95],[2171,95],[2172,95],[2173,95],[2174,95],[2175,95],[2176,95],[2177,95],[2178,95],[2179,95],[2180,95],[2181,95],[2182,95],[2183,95],[2184,95],[2185,95],[2186,95],[2187,95],[2188,95],[2189,95],[2190,95],[2191,95],[2192,95],[2193,95],[2194,95],[2195,95],[2196,95],[2197,95],[2198,95],[2199,95],[2200,95],[2201,95],[2202,95],[2203,95],[2204,95],[2205,95],[2206,95],[2208,97],[2209,98],[2210,98],[2211,98],[2212,98],[2213,98],[2214,98],[2215,98],[2104,39],[2217,99],[2239,100],[2106,2],[2238,101],[2107,102],[2207,103],[2237,42],[2231,104],[2226,105],[2227,106],[2228,107],[2229,108],[2230,109],[2218,110],[2223,111],[2219,112],[2225,66],[2236,113],[2232,114],[2233,114],[2234,115],[2235,115],[2098,116],[2097,117],[2101,118],[2099,119],[2096,120],[2100,121],[2103,122],[2102,2],[1988,123],[1989,2],[1990,39],[2009,124],[1998,2],[1995,125],[1996,125],[1994,126],[1997,126],[1993,127],[1991,128],[1992,129],[1999,39],[2006,130],[2005,131],[2003,39],[2004,132],[2007,133],[2008,39],[2001,134],[2002,135],[2000,135],[1852,136],[1848,2],[1854,137],[1853,137],[1855,137],[1856,138],[1857,139],[1849,140],[1850,140],[1851,141],[1858,39],[1859,39],[2011,142],[2010,143],[1861,144],[1860,39],[1862,39],[1905,145],[1904,146],[1907,147],[1920,121],[1921,119],[1933,148],[1922,149],[1934,150],[1903,151],[1906,152],[1935,153],[1936,39],[1937,154],[2012,39],[2014,155],[2013,156],[1863,39],[1864,39],[1865,39],[1866,39],[1867,39],[1868,39],[1869,39],[1878,157],[1879,39],[1880,2],[1881,39],[1882,39],[1883,39],[1884,39],[1872,2],[1885,2],[1886,39],[1871,158],[1873,159],[1870,39],[1876,160],[1874,158],[1875,159],[1902,161],[1887,39],[1888,159],[1889,39],[1890,39],[1891,2],[1892,39],[1893,39],[1894,39],[1895,39],[1896,39],[1897,39],[1898,162],[1899,39],[1900,39],[1877,39],[1901,39],[3588,163],[3499,164],[3500,164],[3501,164],[3502,164],[3503,164],[3504,164],[3505,164],[3506,164],[3507,164],[3508,164],[3509,164],[3510,164],[3511,164],[3512,164],[3513,164],[3514,164],[3515,164],[3516,164],[3517,164],[3518,164],[3519,164],[3520,164],[3521,164],[3522,164],[3523,164],[3524,164],[3525,164],[3526,164],[3527,164],[3528,164],[3529,164],[3530,164],[3531,164],[3532,164],[3533,164],[3534,164],[3535,164],[3536,164],[3537,164],[3538,164],[3539,164],[3540,164],[3541,164],[3542,164],[3543,164],[3544,164],[3545,164],[3546,164],[3547,164],[3548,164],[3549,164],[3550,164],[3551,164],[3552,164],[3553,164],[3554,164],[3555,164],[3556,164],[3557,164],[3558,164],[3559,164],[3560,164],[3561,164],[3562,164],[3563,164],[3564,164],[3565,164],[3566,164],[3567,164],[3568,164],[3569,164],[3570,164],[3571,164],[3572,164],[3573,164],[3574,164],[3575,164],[3576,164],[3577,164],[3578,164],[3579,164],[3580,164],[3581,164],[3582,164],[3583,164],[3584,164],[3585,164],[3586,165],[3496,2],[3498,2],[3587,166],[3497,167],[3379,168],[3378,169],[3325,170],[3327,171],[3367,172],[3366,173],[3377,174],[3376,175],[3369,176],[3372,177],[3371,177],[3373,178],[3370,177],[3368,179],[3324,2],[3374,180],[3375,181],[2989,182],[3072,183],[2987,182],[3070,183],[2988,182],[3071,183],[2990,182],[3073,183],[2991,182],[3074,183],[2995,184],[3078,185],[2993,186],[3076,187],[2994,186],[3077,187],[2992,182],[3075,183],[2951,1],[3034,1],[2974,2],[3057,2],[2971,1],[3054,1],[2944,1],[3027,1],[2977,1],[3060,1],[2976,1],[3059,1],[2985,1],[3068,1],[2964,188],[3047,189],[2972,190],[3055,191],[2982,188],[3065,189],[2958,1],[3041,1],[2945,192],[3028,193],[2980,188],[3063,189],[2949,192],[3032,193],[2948,192],[3031,193],[2938,188],[3021,189],[2935,1],[3018,1],[2937,188],[3020,189],[2939,1],[3022,1],[2967,1],[3050,1],[2981,192],[3064,193],[2947,192],[3030,193],[2957,1],[3040,1],[2946,1],[3029,1],[2934,1],[3017,1],[2963,188],[3046,189],[2986,194],[3069,195],[2978,2],[3061,2],[2932,196],[3015,196],[2969,2],[3052,2],[2970,1],[3053,1],[2984,197],[3067,198],[2933,192],[3016,193],[2961,2],[3044,2],[2950,192],[3033,193],[2979,2],[3062,2],[2954,2],[3037,2],[2965,2],[3048,2],[2942,199],[3025,200],[2943,192],[3026,193],[2983,201],[3066,202],[2940,188],[3023,189],[2953,188],[3036,189],[2959,1],[3042,1],[2952,1],[3035,1],[2975,1],[3058,1],[2956,192],[3039,193],[2955,1],[3038,1],[2960,188],[3043,189],[2936,1],[3019,1],[2962,1],[3045,1],[2941,1],[3024,1],[2968,2],[3051,2],[2966,192],[3049,193],[2973,2],[3056,2],[2736,1],[3761,203],[3768,204],[3759,205],[3765,206],[3767,207],[3766,208],[3764,209],[3762,210],[3763,211],[3724,212],[3725,213],[3726,212],[3727,210],[3723,207],[3721,207],[3722,207],[3729,214],[3730,210],[3734,210],[3735,210],[3731,210],[3732,214],[3733,210],[3736,214],[3737,210],[3738,210],[3728,207],[3739,210],[3758,215],[3754,210],[3755,210],[3740,210],[3741,210],[3742,210],[3743,210],[3744,210],[3745,210],[3746,210],[3747,210],[3748,210],[3749,210],[3750,210],[3751,210],[3752,210],[3753,210],[3756,207],[3757,207],[3720,216],[3760,2],[3716,2],[3708,217],[3718,2],[3709,2],[3714,2],[3719,218],[3717,2],[3707,219],[3715,220],[3704,221],[3705,2],[3706,222],[3669,2],[3710,223],[3713,224],[3712,225],[3711,226],[3670,2],[3671,2],[3672,2],[3684,2],[3673,2],[3674,2],[3675,2],[3676,2],[3679,2],[3681,2],[3682,2],[3677,2],[3678,2],[3680,2],[3701,227],[3683,2],[3685,2],[3686,2],[3687,2],[3688,2],[3694,2],[3695,2],[3689,2],[3691,2],[3690,2],[3692,2],[3693,2],[3696,2],[3697,2],[3698,2],[3699,2],[3700,2],[3703,2],[3702,223],[4615,2],[4618,228],[2913,229],[2912,2],[3093,230],[3094,231],[3095,232],[3096,232],[3097,233],[3098,1],[3099,1],[3102,234],[3100,1],[3101,1],[3107,235],[3089,236],[3090,237],[3091,238],[3092,239],[3106,240],[3105,241],[3103,242],[3104,243],[3779,244],[3780,245],[3781,246],[3777,247],[3778,248],[3782,249],[1615,2],[1616,2],[1618,250],[1617,2],[1619,251],[1620,252],[1623,253],[1621,2],[1622,254],[1652,255],[1648,256],[1657,257],[1653,258],[1654,2],[1655,258],[1656,259],[1658,2],[1659,2],[1660,260],[1664,261],[1649,262],[1647,259],[1651,263],[1650,264],[1661,2],[1662,2],[1663,265],[1676,2],[1679,266],[1680,267],[1681,258],[1682,258],[1683,268],[1687,269],[1684,270],[1685,271],[1677,272],[1678,273],[1686,274],[1688,275],[442,2],[315,2],[53,2],[304,276],[305,276],[306,2],[307,258],[317,277],[308,2],[309,278],[310,2],[311,2],[312,276],[313,276],[314,276],[316,279],[324,280],[326,2],[323,2],[329,281],[327,2],[325,2],[321,282],[322,283],[328,2],[330,284],[318,2],[320,285],[319,286],[259,2],[262,287],[258,2],[489,2],[260,2],[261,2],[347,288],[332,288],[339,288],[336,288],[349,288],[340,288],[346,288],[331,289],[350,288],[353,290],[344,288],[334,288],[352,288],[337,288],[335,288],[345,288],[341,288],[351,288],[338,288],[348,288],[333,288],[343,288],[342,288],[360,291],[356,292],[355,2],[354,2],[359,293],[398,294],[54,2],[55,2],[56,2],[471,295],[58,296],[477,297],[476,298],[248,299],[249,296],[369,2],[278,2],[279,2],[370,300],[250,2],[371,2],[372,301],[57,2],[252,302],[253,2],[251,303],[254,302],[255,2],[257,304],[269,305],[270,2],[275,306],[271,2],[272,2],[273,2],[274,2],[276,2],[277,307],[283,308],[286,309],[284,2],[285,2],[303,310],[287,2],[288,2],[520,311],[268,312],[266,313],[264,314],[265,315],[267,2],[295,316],[289,2],[298,317],[291,318],[296,319],[294,320],[297,321],[292,322],[293,323],[281,324],[299,325],[282,326],[301,327],[302,328],[290,2],[256,2],[263,329],[300,330],[366,331],[361,2],[367,332],[362,333],[363,334],[364,335],[365,336],[368,337],[384,338],[383,339],[389,340],[381,2],[382,341],[385,338],[386,342],[388,343],[387,344],[390,345],[375,346],[376,347],[379,348],[378,348],[377,347],[380,347],[374,349],[392,350],[391,351],[394,352],[393,353],[395,354],[357,324],[358,355],[280,2],[396,356],[373,357],[397,358],[533,258],[645,359],[646,360],[650,361],[534,2],[540,362],[643,363],[644,364],[535,2],[536,2],[539,365],[537,2],[538,2],[648,2],[649,366],[647,367],[651,368],[440,369],[441,370],[462,371],[463,372],[464,2],[465,373],[466,374],[475,375],[468,376],[472,377],[480,378],[478,258],[479,379],[469,380],[481,2],[483,381],[484,382],[485,383],[474,384],[470,385],[494,386],[482,387],[509,388],[467,389],[510,390],[507,391],[508,258],[532,392],[457,393],[453,394],[455,395],[506,396],[448,397],[496,398],[495,2],[456,399],[503,400],[460,401],[504,2],[505,402],[458,403],[452,404],[459,405],[454,406],[447,2],[500,407],[513,408],[511,258],[443,258],[499,409],[444,283],[445,372],[446,410],[450,411],[449,412],[512,413],[451,414],[488,415],[486,381],[487,416],[497,283],[498,417],[501,418],[516,419],[517,420],[514,421],[515,422],[518,423],[519,424],[521,425],[493,426],[490,427],[491,276],[492,416],[523,428],[522,429],[529,430],[461,258],[525,431],[524,258],[527,432],[526,2],[528,433],[473,434],[502,435],[531,436],[530,258],[3590,437],[3591,438],[3593,439],[3589,440],[3592,441],[3594,442],[1666,2],[1672,443],[1671,444],[1673,2],[1674,445],[1675,446],[1667,447],[1669,2],[1670,448],[1668,447],[2627,2],[2805,449],[2806,450],[2807,2],[2808,2],[2809,451],[2810,2],[2825,452],[2811,450],[2812,2],[2813,453],[2814,454],[2815,2],[2816,2],[2817,454],[2818,451],[2819,455],[2820,2],[2821,456],[2822,2],[2823,457],[2824,458],[2880,459],[2881,460],[2884,461],[2886,462],[2887,463],[2885,453],[2797,1],[2883,464],[2878,465],[2888,1],[2882,2],[2889,2],[2879,466],[2890,467],[2911,468],[2628,2],[2803,1],[2732,469],[2903,470],[2733,1],[2734,1],[2731,1],[2735,258],[2799,471],[2800,2],[2804,472],[2801,1],[2860,2],[2802,473],[2798,2],[2826,474],[2893,475],[2891,1],[2892,1],[2851,476],[2840,477],[2838,478],[2841,479],[2850,480],[2845,481],[2853,482],[2847,483],[2854,484],[2843,480],[2855,482],[2844,485],[2852,486],[2856,482],[2848,487],[2858,488],[2859,2],[2897,489],[2832,490],[2827,2],[2833,2],[2834,2],[2836,491],[2842,492],[2846,493],[2828,494],[2831,495],[2829,496],[2835,497],[2896,2],[2830,453],[2839,1],[2837,498],[2895,499],[2849,500],[2894,501],[2857,502],[2861,503],[2862,504],[2863,505],[2874,506],[2877,507],[2875,508],[2876,509],[2898,2],[2899,510],[2901,511],[2900,512],[2902,2],[2908,513],[2904,514],[2905,514],[2906,514],[2907,514],[2909,2],[2910,515],[2757,1],[2778,1],[2750,1],[2782,1],[2781,1],[2795,2],[2792,2],[2790,1],[2771,516],[2767,517],[2787,516],[2794,2],[2764,1],[2751,518],[2785,516],[2755,518],[2754,518],[2744,516],[2741,519],[2743,516],[2745,1],[2774,1],[2740,1],[2786,518],[2753,518],[2763,1],[2752,1],[2739,1],[2770,516],[2796,520],[2737,196],[2776,2],[2777,1],[2789,521],[2738,518],[2793,521],[2768,521],[2756,518],[2784,2],[2760,2],[2791,517],[2772,2],[2748,522],[2749,518],[2788,523],[2746,524],[2759,516],[2765,1],[2758,1],[2780,1],[2762,518],[2761,1],[2766,516],[2742,1],[2769,1],[2747,1],[2775,2],[2773,518],[2779,2],[3163,525],[3159,526],[3158,527],[3160,2],[3161,528],[3162,529],[3164,530],[4604,531],[4599,532],[4597,258],[4600,532],[4601,532],[4602,532],[4603,258],[4598,2],[4605,533],[1691,2],[1695,534],[1700,535],[1692,258],[1694,536],[1693,2],[1696,537],[1698,538],[1699,539],[1701,540],[1744,541],[1745,542],[1759,543],[1747,544],[1746,545],[1741,546],[1742,2],[1743,2],[1758,547],[1749,548],[1750,548],[1751,548],[1752,548],[1754,549],[1753,548],[1755,550],[1756,551],[1748,2],[1757,552],[1581,553],[1584,554],[1582,2],[1583,2],[1562,2],[1563,555],[1588,556],[1585,258],[1586,557],[1587,553],[1589,558],[399,2],[400,2],[403,559],[404,2],[405,2],[407,2],[406,2],[421,2],[408,2],[409,560],[410,2],[411,2],[412,561],[413,559],[414,2],[416,562],[417,559],[418,563],[419,561],[420,2],[422,564],[427,565],[436,566],[426,567],[401,2],[415,563],[424,568],[425,2],[423,2],[428,569],[433,570],[429,258],[430,258],[431,258],[432,258],[402,2],[434,2],[435,571],[437,572],[1716,573],[1707,574],[1713,2],[1704,2],[1705,575],[1708,576],[1709,258],[1710,577],[1706,578],[1711,579],[1712,580],[1714,581],[1715,2],[1041,582],[1039,583],[1040,584],[1045,585],[1038,586],[1043,587],[1042,588],[1044,589],[1046,590],[2514,591],[2513,592],[2515,2],[2516,2],[2529,593],[2517,258],[2527,594],[2528,2],[2531,595],[2530,2],[2532,258],[2533,596],[2535,597],[2536,598],[2518,599],[2522,600],[2519,2],[2520,2],[2521,2],[2526,601],[2534,2],[2523,330],[2524,2],[2525,2],[3260,602],[3263,603],[3268,604],[3271,605],[3292,606],[3270,607],[3252,2],[3253,608],[3254,609],[3257,2],[3255,2],[3256,2],[3293,610],[3259,602],[3258,2],[3294,611],[3262,603],[3261,2],[3298,612],[3295,613],[3265,614],[3267,615],[3264,616],[3266,617],[3296,618],[3269,602],[3297,619],[3272,620],[3291,621],[3288,622],[3290,623],[3275,624],[3282,625],[3284,626],[3286,627],[3285,628],[3277,629],[3274,622],[3278,2],[3289,630],[3279,631],[3276,2],[3287,2],[3273,2],[3280,632],[3281,2],[3283,633],[4474,634],[4475,635],[4476,634],[4477,634],[4478,634],[4479,634],[4480,634],[4481,635],[4482,634],[4489,636],[4483,634],[4484,634],[4485,635],[4486,634],[4487,634],[4488,634],[4490,637],[4473,638],[4450,639],[4448,2],[4446,2],[4451,640],[4449,641],[4447,2],[4457,642],[4453,643],[4463,644],[4461,645],[4464,646],[4456,647],[4455,648],[4354,649],[4462,650],[4452,651],[4445,652],[4472,653],[4458,654],[4460,655],[4459,648],[4367,634],[4368,634],[4369,634],[4370,634],[4371,634],[4372,656],[4373,635],[4376,634],[4361,634],[4377,657],[4378,635],[4362,634],[4379,634],[4363,634],[4364,634],[4415,2],[4380,634],[4381,634],[4365,634],[4382,634],[4383,658],[4384,634],[4385,634],[4353,634],[4387,659],[4389,659],[4390,659],[4386,634],[4388,659],[4391,659],[4392,634],[4393,634],[4394,635],[4395,635],[4400,634],[4401,634],[4396,634],[4398,634],[4399,634],[4444,660],[4402,634],[4403,634],[4404,634],[4405,661],[4374,635],[4406,634],[4407,634],[4408,634],[4409,662],[4410,634],[4411,634],[4412,634],[4397,634],[4366,634],[4413,635],[4414,634],[4441,634],[4442,634],[4443,634],[4416,663],[4417,635],[4419,634],[4418,635],[4420,634],[4421,634],[4422,634],[4423,634],[4424,634],[4425,634],[4426,635],[4427,635],[4428,664],[4429,634],[4430,663],[4432,635],[4431,635],[4433,635],[4434,635],[4375,635],[4435,634],[4436,634],[4437,634],[4438,634],[4439,635],[4440,635],[4355,2],[4356,665],[4454,666],[4470,667],[4471,668],[4469,669],[4468,2],[4357,670],[4359,671],[4358,2],[4467,672],[4466,673],[4465,674],[4360,666],[4491,635],[4492,634],[4493,635],[4494,634],[4495,634],[4496,634],[4497,634],[4498,635],[4506,675],[4499,635],[4500,634],[4501,634],[4502,635],[4503,634],[4504,634],[4505,634],[4507,676],[4510,677],[4511,678],[4508,635],[4512,679],[4513,634],[4514,680],[4515,634],[4516,681],[4524,682],[4517,683],[4518,684],[4509,685],[4519,686],[4520,634],[4521,634],[4522,634],[4523,635],[4525,687],[4526,688],[4529,689],[4528,690],[4530,634],[4532,634],[4527,691],[4531,690],[4545,692],[4533,693],[4534,694],[4535,688],[4536,695],[4537,696],[4544,697],[4539,697],[4540,697],[4541,698],[4543,697],[4542,699],[4538,693],[4546,700],[4617,2],[2019,701],[2015,119],[2016,119],[2018,702],[2017,39],[2029,703],[2020,119],[2022,704],[2021,39],[2024,705],[2023,2],[2027,706],[2028,707],[2025,708],[2026,708],[1939,2],[1938,2],[1985,709],[1987,710],[1940,711],[1986,712],[1971,66],[1972,713],[1953,714],[1954,2],[1975,715],[1974,716],[1984,717],[1977,148],[1978,2],[1976,718],[1983,66],[1979,719],[1980,719],[1982,720],[1981,719],[1973,39],[1955,39],[1970,721],[1957,722],[1956,39],[1964,723],[1959,724],[1960,724],[1965,39],[1962,39],[1961,724],[1958,39],[1967,39],[1966,724],[1963,724],[1968,39],[1969,725],[2082,39],[2083,2],[2086,726],[2094,727],[2087,2],[2088,2],[2089,2],[2090,2],[2091,2],[2092,2],[2093,2],[2030,39],[2031,728],[2034,729],[2036,730],[2035,39],[2037,729],[2038,729],[2040,731],[2032,39],[2039,39],[2033,2],[2055,120],[2059,732],[2056,39],[2058,39],[2052,733],[2053,2],[2054,39],[2051,734],[2050,39],[2057,735],[1846,736],[1831,39],[1844,737],[1845,39],[1847,738],[1926,739],[1927,740],[1928,39],[1929,741],[1925,742],[1923,39],[1924,39],[1932,743],[1930,2],[1931,39],[1836,2],[1840,2],[1832,2],[1833,2],[1834,2],[1835,2],[1843,744],[1837,745],[1838,39],[1839,746],[1842,2],[1841,39],[1910,2],[1916,39],[1911,39],[1912,39],[1913,39],[1917,39],[1919,747],[1914,39],[1915,39],[1918,39],[1909,748],[1908,39],[2060,39],[2061,749],[2062,750],[2063,2],[2064,751],[2065,2],[2066,2],[2067,2],[2068,39],[2069,749],[2070,39],[2072,752],[2073,753],[2071,39],[2074,2],[2075,2],[2095,754],[2076,2],[2077,39],[2078,2],[2079,749],[2080,2],[2081,2],[1761,755],[1762,756],[1763,2],[1764,2],[1777,757],[1778,758],[1775,759],[1776,760],[1779,761],[1782,762],[1784,763],[1785,764],[1767,765],[1786,2],[1790,766],[1788,767],[1789,2],[1783,2],[1792,768],[1768,769],[1794,770],[1795,771],[1798,772],[1797,773],[1793,774],[1796,775],[1791,776],[1799,777],[1800,778],[1804,779],[1805,780],[1803,781],[1781,782],[1769,2],[1772,783],[1806,784],[1807,785],[1808,785],[1765,2],[1810,786],[1809,785],[1830,787],[1770,2],[1774,788],[1811,789],[1812,2],[1766,2],[1802,790],[1818,791],[1817,792],[1814,2],[1815,793],[1816,2],[1813,794],[1801,795],[1819,796],[1820,797],[1821,762],[1822,762],[1823,798],[1787,2],[1825,799],[1826,800],[1780,2],[1827,2],[1828,801],[1824,2],[1771,802],[1773,776],[1829,755],[2043,803],[2046,2],[2044,804],[2047,2],[2045,805],[2049,806],[2048,2],[2041,39],[2042,2],[1941,2],[1943,39],[1942,807],[1944,808],[1945,809],[1946,807],[1947,807],[1948,810],[1952,811],[1949,807],[1950,810],[1951,2],[2221,812],[2222,813],[2220,39],[2085,814],[2084,2],[2555,2],[1599,815],[1598,816],[2546,816],[1595,817],[4352,818],[1601,819],[1600,817],[2270,820],[1596,2],[4623,821],[3157,822],[1573,823],[1566,824],[1570,825],[1568,826],[1571,827],[1569,828],[1572,829],[1567,2],[1565,830],[1564,831],[3156,2],[4549,832],[590,833],[591,833],[592,834],[546,835],[593,836],[594,837],[595,838],[541,2],[544,839],[542,2],[543,2],[596,840],[597,841],[598,842],[599,843],[600,844],[601,845],[602,845],[603,846],[604,847],[605,848],[606,849],[547,2],[545,2],[607,850],[608,851],[609,852],[641,853],[610,854],[611,855],[612,856],[613,857],[614,858],[615,859],[616,860],[617,861],[618,862],[619,863],[620,863],[621,864],[622,2],[623,865],[625,866],[624,867],[626,868],[627,869],[628,870],[629,871],[630,872],[631,873],[632,874],[633,875],[634,876],[635,877],[636,878],[637,879],[638,880],[548,2],[549,2],[550,2],[589,881],[639,882],[640,883],[3186,884],[3173,885],[3180,886],[3176,887],[3174,888],[3177,889],[3181,890],[3182,886],[3179,891],[3178,892],[3183,893],[3184,894],[3185,895],[3175,896],[3196,897],[3195,898],[1697,899],[1593,2],[1594,2],[1592,900],[1597,901],[2321,902],[2312,2],[2313,2],[2314,2],[2315,2],[2316,2],[2317,2],[2318,2],[2319,2],[2320,2],[2543,903],[3493,904],[3494,905],[3495,906],[3445,907],[3460,908],[3446,908],[3442,909],[3448,908],[3449,908],[3451,910],[3457,908],[3452,908],[3463,908],[3453,908],[3450,908],[3456,908],[3461,908],[3459,908],[3462,911],[3464,908],[3447,908],[3465,908],[3454,908],[3455,908],[3466,911],[3458,908],[3470,912],[3438,913],[3474,914],[3399,915],[3478,915],[3387,2],[3389,916],[3400,915],[3390,915],[3388,2],[3401,2],[3402,917],[3403,915],[3393,918],[3397,919],[3395,2],[3405,920],[3394,2],[3391,915],[3426,921],[3406,915],[3407,915],[3409,922],[3396,915],[3410,2],[3411,923],[3398,915],[3412,915],[3413,915],[3414,915],[3415,915],[3416,915],[3436,924],[3437,2],[3417,915],[3408,2],[3392,925],[3418,915],[3427,926],[3428,2],[3429,927],[3430,928],[3431,928],[3432,929],[3434,930],[3419,915],[3433,915],[3435,931],[3404,2],[3420,920],[3421,920],[3425,932],[3422,2],[3423,933],[3424,915],[3467,934],[3482,935],[3475,936],[3443,937],[3477,938],[3444,939],[3487,940],[3483,941],[3479,942],[3480,943],[3481,944],[3484,945],[3486,946],[3485,947],[3471,948],[3440,949],[3441,950],[3439,951],[3472,952],[3468,953],[3490,2],[3488,954],[3476,955],[3489,2],[3491,2],[3469,956],[3473,957],[3492,958],[1717,2],[3117,959],[3116,2],[551,2],[1646,960],[1561,961],[1412,962],[1413,2],[1414,962],[1415,2],[1416,963],[1417,964],[1418,962],[1419,962],[1420,2],[1421,2],[1422,2],[1423,2],[1424,2],[1425,2],[1426,2],[1427,964],[1428,962],[1429,962],[1430,2],[1431,962],[1432,962],[1438,965],[1433,2],[1439,966],[1434,966],[1435,964],[1436,2],[1437,2],[1463,967],[1440,964],[1454,968],[1441,968],[1442,968],[1443,968],[1453,969],[1444,964],[1445,968],[1446,968],[1447,968],[1448,968],[1449,964],[1450,964],[1451,964],[1452,968],[1455,964],[1456,964],[1457,2],[1458,2],[1460,2],[1459,2],[1461,964],[1462,2],[1464,970],[1411,971],[1401,972],[1398,973],[1406,974],[1404,975],[1400,976],[1399,977],[1408,978],[1407,979],[1410,980],[1409,981],[1049,2],[1052,964],[1053,964],[1054,964],[1055,964],[1056,964],[1057,964],[1058,964],[1060,964],[1059,964],[1061,964],[1062,964],[1063,964],[1064,964],[1176,964],[1065,964],[1066,964],[1067,964],[1068,964],[1177,964],[1178,2],[1179,982],[1180,964],[1181,963],[1182,963],[1184,983],[1185,964],[1186,984],[1187,964],[1189,985],[1190,963],[1191,986],[1069,976],[1070,964],[1071,964],[1072,2],[1074,2],[1073,964],[1075,987],[1076,976],[1077,976],[1078,976],[1079,964],[1080,976],[1081,964],[1082,976],[1083,964],[1085,963],[1086,2],[1087,2],[1088,2],[1089,964],[1090,963],[1091,2],[1092,2],[1093,2],[1094,2],[1095,2],[1096,2],[1097,2],[1098,2],[1099,2],[1100,25],[1101,2],[1102,988],[1103,2],[1104,2],[1105,2],[1106,2],[1107,2],[1108,964],[1114,963],[1109,964],[1110,964],[1111,964],[1112,963],[1113,964],[1115,962],[1116,2],[1117,2],[1118,964],[1192,963],[1119,2],[1193,964],[1194,964],[1195,964],[1120,964],[1196,964],[1121,964],[1198,962],[1197,962],[1199,962],[1200,962],[1201,964],[1202,963],[1203,963],[1204,964],[1122,2],[1206,962],[1205,962],[1123,2],[1124,989],[1125,964],[1126,964],[1127,964],[1128,964],[1130,963],[1129,963],[1131,964],[1132,964],[1133,964],[1084,964],[1207,963],[1208,963],[1209,964],[1210,964],[1213,963],[1211,963],[1212,990],[1214,991],[1217,963],[1215,963],[1216,992],[1218,993],[1219,993],[1220,991],[1221,963],[1222,994],[1223,994],[1224,964],[1225,963],[1226,964],[1227,964],[1228,964],[1229,964],[1230,964],[1134,995],[1231,963],[1232,964],[1233,996],[1234,964],[1235,964],[1236,963],[1237,964],[1238,964],[1239,964],[1240,964],[1241,964],[1242,964],[1243,996],[1244,996],[1245,964],[1246,964],[1247,964],[1248,997],[1249,998],[1250,963],[1251,999],[1252,964],[1253,963],[1254,964],[1255,964],[1256,964],[1257,964],[1258,964],[1259,964],[1051,1000],[1135,2],[1136,964],[1137,2],[1138,2],[1139,964],[1140,2],[1141,964],[1260,976],[1262,1001],[1261,1001],[1263,1002],[1264,964],[1265,964],[1266,964],[1267,963],[1183,963],[1142,964],[1269,964],[1268,964],[1270,964],[1271,1003],[1272,964],[1273,964],[1274,964],[1275,964],[1276,964],[1277,964],[1143,2],[1144,2],[1145,2],[1146,2],[1147,2],[1278,964],[1279,995],[1148,2],[1149,2],[1150,2],[1151,962],[1280,964],[1281,1004],[1282,964],[1283,964],[1284,964],[1285,964],[1286,963],[1287,963],[1288,963],[1289,964],[1290,963],[1291,964],[1292,964],[1152,964],[1293,964],[1294,964],[1295,964],[1153,2],[1154,2],[1155,964],[1156,964],[1157,964],[1158,964],[1159,2],[1160,2],[1296,964],[1297,963],[1161,2],[1162,2],[1298,964],[1163,2],[1300,964],[1299,964],[1301,964],[1302,964],[1303,964],[1304,964],[1164,964],[1165,963],[1305,2],[1166,2],[1167,963],[1168,2],[1169,2],[1170,2],[1306,964],[1307,964],[1311,964],[1312,963],[1313,964],[1314,963],[1315,964],[1171,2],[1308,964],[1309,964],[1310,964],[1316,963],[1317,964],[1318,963],[1319,963],[1322,963],[1320,963],[1321,963],[1323,964],[1324,964],[1325,964],[1326,1005],[1327,964],[1328,963],[1329,964],[1330,964],[1331,964],[1172,2],[1173,2],[1332,964],[1333,964],[1334,964],[1335,964],[1174,2],[1175,2],[1336,964],[1337,964],[1338,964],[1339,963],[1340,1006],[1341,963],[1342,1007],[1343,964],[1344,964],[1345,963],[1346,964],[1347,963],[1348,964],[1349,964],[1350,964],[1351,963],[1352,964],[1354,964],[1353,964],[1355,963],[1356,963],[1357,963],[1358,963],[1359,964],[1360,964],[1361,963],[1362,964],[1363,964],[1364,964],[1365,1008],[1366,964],[1367,963],[1368,964],[1369,1009],[1370,964],[1371,964],[1372,964],[1188,963],[1373,963],[1374,963],[1375,1010],[1376,963],[1377,1011],[1378,964],[1379,1012],[1380,1013],[1381,964],[1382,1014],[1383,964],[1384,964],[1385,1015],[1386,964],[1387,964],[1388,964],[1389,964],[1390,964],[1391,964],[1392,964],[1393,963],[1394,963],[1395,964],[1396,1016],[1397,964],[1402,964],[1050,964],[1403,1017],[1465,2],[1466,2],[1467,2],[1468,2],[1474,1018],[1469,2],[1470,2],[1471,1019],[1472,1020],[1473,2],[1475,1021],[1476,1022],[1477,1023],[1478,1023],[1479,1023],[1480,2],[1481,1023],[1482,2],[1483,2],[1484,2],[1485,2],[1486,1024],[1499,1025],[1487,1023],[1488,1023],[1489,1024],[1490,1023],[1491,1023],[1492,2],[1493,2],[1494,2],[1495,1023],[1496,2],[1497,2],[1498,2],[1500,1023],[1501,2],[1503,1026],[1504,1027],[1505,2],[1506,2],[1507,2],[1502,1028],[1508,2],[1509,2],[1510,1028],[1511,964],[1512,1029],[1513,964],[1514,964],[1515,2],[1516,2],[1517,1028],[1518,2],[1535,1030],[1519,964],[1522,1031],[1521,1032],[1520,1026],[1523,1033],[1524,2],[1525,2],[1526,962],[1527,2],[1528,1034],[1529,1034],[1530,1035],[1531,2],[1532,2],[1533,964],[1534,2],[1536,1036],[1537,1037],[1538,1038],[1539,1038],[1540,1037],[1541,1039],[1542,1039],[1543,2],[1544,1039],[1545,1039],[1558,1040],[1546,1037],[1547,1041],[1548,1037],[1549,1039],[1550,1042],[1554,1039],[1555,1039],[1556,1039],[1557,1039],[1551,1039],[1552,1039],[1553,1039],[1559,1037],[1560,1043],[1048,1044],[2611,1045],[2612,1045],[2613,1045],[2619,1046],[2614,1045],[2615,1045],[2616,1045],[2617,1045],[2618,1045],[2602,1047],[2601,2],[2620,1048],[2608,2],[2604,1049],[2595,2],[2594,2],[2596,2],[2597,1045],[2598,1050],[2610,1051],[2599,1045],[2600,1045],[2605,1052],[2606,1053],[2607,1045],[2603,2],[2609,2],[2282,2],[2401,1054],[2405,1054],[2404,1054],[2402,1054],[2403,1054],[2406,1054],[2285,1054],[2297,1054],[2286,1054],[2299,1054],[2301,1054],[2295,1054],[2294,1054],[2296,1054],[2300,1054],[2302,1054],[2287,1054],[2298,1054],[2288,1054],[2290,1055],[2291,1054],[2292,1054],[2293,1054],[2309,1054],[2308,1054],[2409,1056],[2303,1054],[2305,1054],[2304,1054],[2306,1054],[2307,1054],[2408,1054],[2407,1054],[2310,1054],[2392,1054],[2391,1054],[2322,1057],[2323,1057],[2325,1054],[2369,1054],[2390,1054],[2326,1057],[2370,1054],[2367,1054],[2371,1054],[2327,1054],[2328,1054],[2329,1057],[2372,1054],[2366,1057],[2324,1057],[2373,1054],[2330,1057],[2374,1054],[2354,1054],[2331,1057],[2332,1054],[2333,1054],[2364,1057],[2336,1054],[2335,1054],[2375,1054],[2376,1054],[2377,1057],[2338,1054],[2340,1054],[2341,1054],[2347,1054],[2348,1054],[2342,1057],[2378,1054],[2365,1057],[2343,1054],[2344,1054],[2379,1054],[2345,1054],[2337,1057],[2380,1054],[2363,1054],[2381,1054],[2346,1057],[2349,1054],[2350,1054],[2368,1057],[2382,1054],[2383,1054],[2362,1058],[2339,1054],[2384,1057],[2385,1054],[2386,1054],[2387,1054],[2388,1057],[2351,1054],[2389,1054],[2355,1054],[2352,1057],[2353,1057],[2334,1054],[2356,1054],[2359,1054],[2357,1054],[2358,1054],[2311,1054],[2399,1054],[2393,1054],[2394,1054],[2396,1054],[2397,1054],[2395,1054],[2400,1054],[2398,1054],[2284,1059],[2417,1060],[2415,1061],[2416,1062],[2414,1063],[2413,1054],[2412,1064],[2281,2],[2283,2],[2279,2],[2410,2],[2411,1065],[2289,1059],[2280,2],[4548,1066],[1575,2],[1574,2],[1580,1067],[1576,1068],[1579,1069],[1578,1070],[1577,2],[2783,2],[3218,2],[1640,2],[642,25],[2538,2],[2540,1071],[2539,1071],[2541,1072],[2547,2],[2554,1073],[2542,2],[2548,1074],[2545,1075],[2544,1076],[2552,1077],[2549,1078],[2550,1079],[2551,1080],[2553,1081],[1665,2],[4622,1082],[3383,1083],[3382,1084],[3380,2],[3381,2],[3386,1085],[3384,1086],[3385,2],[1405,1087],[3112,1088],[3108,1089],[3109,1090],[3110,1091],[3111,2],[2865,1092],[2864,1],[2867,1093],[2866,1092],[2639,1094],[2706,1095],[2705,1096],[2704,1097],[2644,1098],[2660,1099],[2658,1100],[2659,1101],[2645,1102],[2730,1103],[2630,2],[2632,2],[2633,1104],[2634,2],[2637,1105],[2640,2],[2657,1106],[2635,2],[2652,1107],[2638,1108],[2653,1109],[2656,1110],[2654,1110],[2651,1111],[2631,2],[2636,2],[2655,1112],[2661,1113],[2649,2],[2643,1114],[2641,1115],[2650,1116],[2647,1117],[2646,1117],[2642,1118],[2648,1119],[2725,1120],[2719,1121],[2712,1122],[2711,1123],[2720,1124],[2721,1110],[2713,1125],[2726,1126],[2707,1127],[2708,1128],[2709,1129],[2729,1130],[2710,1123],[2714,1126],[2715,1131],[2728,1132],[2722,1133],[2723,1108],[2724,1131],[2727,1110],[2716,1129],[2662,1134],[2717,1135],[2718,1136],[2703,1137],[2701,1138],[2702,1138],[2667,1138],[2668,1138],[2669,1138],[2670,1138],[2671,1138],[2672,1138],[2673,1138],[2674,1138],[2693,1138],[2665,1138],[2675,1138],[2676,1138],[2677,1138],[2678,1138],[2679,1138],[2680,1138],[2700,1138],[2681,1138],[2682,1138],[2683,1138],[2698,1138],[2684,1138],[2699,1138],[2685,1138],[2696,1138],[2697,1138],[2686,1138],[2687,1138],[2688,1138],[2694,1138],[2695,1138],[2689,1138],[2690,1138],[2691,1138],[2692,1138],[2666,1139],[2664,1140],[2663,1141],[2629,2],[3187,2],[4555,546],[3326,1142],[1638,1143],[1639,1144],[1637,1145],[1625,1146],[1630,1147],[1631,1148],[1634,1149],[1633,1150],[1632,1151],[1635,1152],[1642,1153],[1645,1154],[1644,1155],[1643,1156],[1636,1157],[1626,885],[1641,1158],[1628,1159],[1624,1160],[1629,1161],[1627,1146],[4620,1162],[4621,1163],[4616,2],[1689,2],[1047,2],[2361,1164],[2360,2],[2916,2],[4619,1165],[1590,2],[4547,1166],[52,2],[247,1167],[220,2],[198,1168],[196,1168],[246,1169],[211,1170],[210,1170],[111,1171],[62,1172],[218,1171],[219,1171],[221,1173],[222,1171],[223,1174],[122,1175],[224,1171],[195,1171],[225,1171],[226,1176],[227,1171],[228,1170],[229,1177],[230,1171],[231,1171],[232,1171],[233,1171],[234,1170],[235,1171],[236,1171],[237,1171],[238,1171],[239,1178],[240,1171],[241,1171],[242,1171],[243,1171],[244,1171],[61,1169],[64,1174],[65,1174],[66,1174],[67,1174],[68,1174],[69,1174],[70,1174],[71,1171],[73,1179],[74,1174],[72,1174],[75,1174],[76,1174],[77,1174],[78,1174],[79,1174],[80,1174],[81,1171],[82,1174],[83,1174],[84,1174],[85,1174],[86,1174],[87,1171],[88,1174],[89,1174],[90,1174],[91,1174],[92,1174],[93,1174],[94,1171],[96,1180],[95,1174],[97,1174],[98,1174],[99,1174],[100,1174],[101,1178],[102,1171],[103,1171],[117,1181],[105,1182],[106,1174],[107,1174],[108,1171],[109,1174],[110,1174],[112,1183],[113,1174],[114,1174],[115,1174],[116,1174],[118,1174],[119,1174],[120,1174],[121,1174],[123,1184],[124,1174],[125,1174],[126,1174],[127,1171],[128,1174],[129,1185],[130,1185],[131,1185],[132,1171],[133,1174],[134,1174],[135,1174],[140,1174],[136,1174],[137,1171],[138,1174],[139,1171],[141,1174],[142,1174],[143,1174],[144,1174],[145,1174],[146,1174],[147,1171],[148,1174],[149,1174],[150,1174],[151,1174],[152,1174],[153,1174],[154,1174],[155,1174],[156,1174],[157,1174],[158,1174],[159,1174],[160,1174],[161,1174],[162,1174],[163,1174],[164,1186],[165,1174],[166,1174],[167,1174],[168,1174],[169,1174],[170,1174],[171,1171],[172,1171],[173,1171],[174,1171],[175,1171],[176,1174],[177,1174],[178,1174],[179,1174],[197,1187],[245,1171],[182,1188],[181,1189],[205,1190],[204,1191],[200,1192],[199,1191],[201,1193],[190,1194],[188,1195],[203,1196],[202,1193],[189,2],[191,1197],[104,1198],[60,1199],[59,1174],[194,2],[186,1200],[187,1201],[184,2],[185,1202],[183,1174],[192,1203],[63,1204],[212,2],[213,2],[206,2],[209,1170],[208,2],[214,2],[215,2],[207,1205],[216,2],[217,2],[180,1206],[193,1207],[2243,885],[2560,1208],[2559,1209],[2561,1210],[2556,1211],[2563,1212],[2558,1213],[2566,1214],[2565,1215],[2562,1216],[2564,1217],[2557,1209],[4169,1218],[4042,1219],[4171,1220],[4170,1221],[4043,1222],[4044,1223],[4172,1224],[4045,1225],[4173,1226],[4046,1227],[4174,1228],[4175,1229],[4047,1230],[4176,1231],[4048,1232],[4178,1233],[4177,1234],[4049,1235],[4179,1236],[4181,1237],[4050,1238],[4180,1239],[4182,1240],[4051,1241],[4183,1242],[4052,1243],[4184,1244],[4053,1245],[4185,1246],[4054,1247],[4187,1248],[4055,1249],[4186,1250],[4188,1251],[4056,1252],[4189,1253],[4057,1254],[4190,1255],[4058,1256],[4191,1257],[4192,1258],[4193,1259],[4194,1260],[4059,1261],[4195,1262],[4060,1263],[4196,1264],[4061,1265],[4197,1266],[4062,1267],[4198,1268],[4063,1269],[4199,1270],[4064,1271],[4200,1272],[4201,1273],[4065,1274],[4202,1275],[4066,1276],[4203,1277],[4204,1278],[4067,1279],[4033,1280],[4205,1281],[4206,1282],[4209,1283],[4207,1284],[4068,1285],[4208,1286],[4069,1287],[4210,1288],[4211,1289],[4070,1290],[4213,1291],[4071,1292],[4212,1293],[4214,1294],[4072,1295],[4215,1296],[4073,1297],[4036,1298],[4216,1299],[4074,1300],[4039,1301],[4217,1302],[4075,1303],[4218,1304],[4219,1305],[4220,1306],[4076,1307],[4221,1308],[4077,1309],[4222,1310],[4223,1311],[4224,1312],[4078,1313],[4225,1314],[4079,1315],[4226,1316],[4080,1317],[4227,1318],[4081,1319],[4228,1320],[4229,1321],[4082,1322],[4230,1323],[4083,1324],[4314,1325],[4231,1326],[4084,1327],[4232,1328],[4233,1329],[4085,1330],[4234,1331],[4086,1332],[4235,1333],[4087,1334],[4236,1335],[4088,1336],[4237,1337],[4089,1338],[4238,1339],[4090,1340],[4239,1341],[4091,1342],[4240,1343],[4092,1344],[4241,1345],[4093,1346],[4242,1347],[4094,1348],[4243,1349],[4095,1350],[4032,1351],[4244,1352],[4245,1353],[4246,1354],[4096,1355],[4034,1356],[4037,1357],[4247,1358],[4097,1359],[4248,1360],[4098,1361],[4249,1362],[4099,1363],[4250,1364],[4100,1365],[4251,1366],[4101,1367],[4252,1368],[4102,1369],[4253,1370],[4254,1371],[4103,1372],[4255,1373],[4104,1374],[4256,1375],[4257,1376],[4105,1377],[4258,1378],[4106,1379],[4259,1380],[4107,1381],[4260,1382],[4108,1383],[4261,1384],[4109,1385],[4262,1386],[4110,1387],[4263,1388],[4111,1389],[4264,1390],[4112,1391],[4265,1392],[4113,1393],[4266,1394],[4267,1395],[4114,1396],[4268,1397],[4115,1398],[4269,1399],[4116,1400],[4035,1401],[4270,1402],[4117,1403],[4271,1404],[4118,1405],[4272,1406],[4274,1407],[4119,1408],[4273,1409],[4275,1410],[4120,1411],[4277,1412],[4276,1413],[4121,1414],[4122,1415],[4278,1416],[4279,1417],[4123,1418],[4280,1419],[4124,1420],[4281,1421],[4125,1422],[4282,1423],[4283,1424],[4126,1425],[4284,1426],[4127,1427],[4285,1428],[4286,1429],[4128,1430],[4287,1431],[4129,1432],[4288,1433],[4130,1434],[4289,1435],[4131,1436],[4290,1437],[4132,1438],[4291,1439],[4133,1440],[4134,1441],[4135,1442],[4136,1443],[4137,1444],[4138,1445],[4139,1446],[4140,1447],[4141,1448],[4292,1449],[4142,1450],[4143,1451],[4144,1452],[4145,1453],[4146,1454],[4147,1455],[4041,1456],[4293,1457],[4148,1458],[4294,1459],[4149,1460],[4295,1461],[4296,1462],[4150,1463],[4297,1464],[4151,1465],[4298,1466],[4152,1467],[4299,1468],[4300,1469],[4153,1470],[4301,1471],[4154,1472],[4302,1473],[4155,1474],[4303,1475],[4156,1476],[4304,1477],[4157,1478],[4305,1479],[4158,1480],[4306,1481],[4159,1482],[4307,1483],[4160,1484],[4040,1485],[4308,1486],[4161,1487],[4310,1488],[4162,1489],[4309,1490],[4164,1491],[4163,1492],[4165,1493],[4167,1494],[4311,1495],[4312,1496],[4166,1497],[4313,1498],[4168,1499],[4038,1500],[2869,1501],[2873,1502],[2871,2],[2872,2],[2870,1503],[2868,2],[720,1504],[719,2],[741,2],[659,1505],[721,2],[668,2],[658,2],[785,2],[872,2],[822,1506],[1028,1507],[869,1508],[1027,1509],[1026,1509],[871,2],[722,1510],[829,1511],[825,1512],[1023,1508],[993,2],[943,1513],[944,1514],[945,1514],[957,1514],[950,1515],[949,1516],[951,1514],[952,1514],[956,1517],[954,1518],[984,1519],[981,2],[980,1520],[982,1514],[996,1521],[994,2],[990,1522],[995,2],[989,1523],[958,2],[959,2],[962,2],[960,2],[961,2],[963,2],[964,2],[967,2],[965,2],[966,2],[968,2],[969,2],[664,1524],[937,2],[938,2],[939,2],[940,2],[665,1525],[941,2],[942,2],[971,1526],[696,1527],[970,2],[699,2],[700,1528],[701,1528],[948,1529],[946,1529],[947,2],[656,1527],[695,1530],[991,1531],[663,2],[955,1524],[983,586],[953,1532],[972,1528],[973,1533],[974,1534],[975,1534],[976,1534],[977,1534],[978,1535],[979,1535],[988,1536],[987,2],[985,2],[986,1537],[992,1538],[815,2],[816,1539],[819,1506],[820,1506],[821,1506],[790,1540],[791,1541],[810,1506],[727,1542],[814,1506],[732,2],[809,1543],[769,1544],[733,1545],[792,2],[793,1546],[813,1506],[807,2],[808,1547],[794,1540],[795,1548],[689,2],[812,1506],[817,2],[818,1549],[823,2],[824,1550],[690,1551],[796,1506],[811,1506],[798,2],[799,2],[800,2],[801,2],[802,2],[803,2],[797,2],[804,2],[1025,2],[805,1552],[806,1553],[662,2],[687,2],[718,2],[692,2],[694,2],[780,2],[688,1529],[723,2],[726,2],[786,1554],[775,1555],[826,1556],[715,1557],[706,2],[697,1558],[698,1559],[1032,1521],[707,2],[710,1558],[693,2],[708,1514],[714,1560],[709,1535],[702,1561],[705,1531],[875,1562],[898,1562],[879,1562],[882,1563],[884,1562],[933,1562],[910,1562],[874,1562],[902,1562],[930,1562],[881,1562],[911,1562],[896,1562],[899,1562],[887,1562],[920,1564],[916,1562],[909,1562],[891,1565],[890,1565],[907,1563],[917,1562],[935,1566],[936,1567],[921,1568],[913,1562],[894,1562],[880,1562],[883,1562],[915,1562],[900,1563],[908,1562],[905,1569],[922,1569],[906,1563],[892,1562],[901,1562],[934,1562],[924,1562],[912,1562],[932,1562],[914,1562],[893,1562],[928,1562],[918,1562],[895,1562],[923,1562],[931,1562],[897,1562],[919,1565],[903,1562],[927,1570],[878,1570],[889,1562],[888,1562],[886,1571],[873,2],[885,1562],[929,1569],[925,1569],[904,1569],[926,1569],[734,1572],[740,1573],[739,1574],[730,1575],[729,2],[738,1576],[737,1576],[736,1576],[1016,1577],[735,1578],[777,2],[728,2],[745,1579],[744,1580],[997,1572],[999,1572],[1000,1572],[1001,1572],[1002,1572],[1003,1572],[1004,1581],[1009,1572],[1005,1572],[1006,1572],[1015,1572],[1007,1572],[1008,1572],[1010,1572],[1011,1572],[1012,1572],[1013,1572],[998,1572],[1014,1582],[703,2],[870,1583],[1037,1584],[1017,1585],[1018,1586],[1021,1587],[1019,1586],[716,1588],[717,1589],[1020,1586],[762,2],[667,1590],[862,2],[676,2],[681,1591],[863,1592],[860,2],[766,2],[867,1593],[866,2],[832,2],[861,1514],[858,2],[859,1594],[868,1595],[857,2],[856,1535],[677,1535],[661,1596],[830,1597],[864,2],[865,2],[713,1536],[666,2],[683,1531],[763,1598],[686,1599],[685,1600],[682,1601],[831,1602],[767,1603],[674,1604],[833,1605],[679,1606],[678,1607],[675,1608],[712,1609],[653,2],[680,2],[654,2],[655,2],[657,2],[660,1592],[652,2],[704,2],[711,2],[684,1610],[789,1611],[1029,1612],[788,1588],[1030,1613],[1031,1614],[673,1615],[877,1616],[876,1617],[731,1618],[840,1619],[782,1620],[849,1621],[783,1622],[851,1623],[841,1624],[853,1625],[854,1626],[839,2],[847,1627],[770,1628],[843,1629],[842,1629],[828,1630],[827,1630],[852,1631],[774,1632],[772,1633],[773,1633],[844,2],[855,1634],[845,2],[850,1635],[779,1636],[848,1637],[846,2],[781,1638],[771,2],[838,1639],[1022,1640],[1024,1641],[1035,2],[776,1642],[743,2],[787,1643],[742,2],[778,1644],[784,1645],[761,2],[669,2],[765,2],[724,2],[834,2],[836,1646],[746,2],[671,586],[1033,1647],[691,1648],[837,1649],[764,1650],[670,1651],[768,1652],[725,1653],[835,1654],[747,1655],[672,1656],[760,1657],[748,2],[759,1658],[754,1659],[755,1660],[758,1556],[757,1661],[753,1660],[756,1661],[749,1556],[750,1556],[751,1556],[752,1662],[1034,1663],[1036,1664],[50,2],[51,2],[9,2],[11,2],[10,2],[2,2],[12,2],[13,2],[14,2],[15,2],[16,2],[17,2],[18,2],[19,2],[3,2],[20,2],[21,2],[4,2],[22,2],[26,2],[23,2],[24,2],[25,2],[27,2],[28,2],[29,2],[5,2],[30,2],[31,2],[32,2],[33,2],[6,2],[37,2],[34,2],[35,2],[36,2],[38,2],[7,2],[39,2],[44,2],[45,2],[40,2],[41,2],[42,2],[43,2],[8,2],[49,2],[46,2],[47,2],[48,2],[1,2],[567,1665],[577,1666],[566,1665],[587,1667],[558,1668],[557,1669],[586,25],[580,1670],[585,1671],[560,1672],[574,1673],[559,1674],[583,1675],[555,1676],[554,25],[584,1677],[556,1678],[561,1679],[562,2],[565,1679],[552,2],[588,1680],[578,1681],[569,1682],[570,1683],[572,1684],[568,1685],[571,1686],[581,25],[563,1687],[564,1688],[573,1689],[553,810],[576,1681],[575,1679],[579,2],[582,1690],[3365,1691],[3343,1692],[3353,1693],[3342,1692],[3363,1694],[3334,1695],[3333,1669],[3362,25],[3356,1696],[3361,1697],[3336,1698],[3350,1699],[3335,1700],[3359,1701],[3331,1702],[3330,25],[3360,1703],[3332,1704],[3337,1705],[3338,2],[3341,1705],[3328,2],[3364,1706],[3354,1707],[3345,1708],[3346,1709],[3348,1710],[3344,1711],[3347,1712],[3357,25],[3339,1713],[3340,1714],[3349,1715],[3329,810],[3352,1716],[3351,1705],[3355,2],[3358,1717],[2265,1718],[2249,2],[2250,2],[2252,1719],[2253,2],[2251,2],[2254,1719],[2255,1719],[2257,1720],[2256,1719],[2258,1719],[2259,1720],[2260,1719],[2261,2],[2262,1719],[2263,2],[2264,2],[3247,1721],[3248,1722],[3242,1723],[3244,1724],[3245,1725],[3240,1726],[3239,1727],[3241,1728],[3238,1729],[3243,1730],[3246,1731],[439,1732],[4351,1733],[438,258],[3153,1734],[3155,1735],[3152,1736],[4558,1737],[4559,1738],[3148,1739],[3149,1740],[3147,1741],[3146,1742],[3144,2],[3145,2],[3150,258],[3154,1743],[3151,1744],[3201,1745],[3191,1746],[3204,1747],[3192,1748],[4560,1749],[3190,2],[3202,1750],[3203,1751],[4561,1752],[3194,1753],[3205,1754],[3193,1755],[3127,258],[2575,1756],[3165,1757],[1703,1758],[2576,1759],[3169,1760],[4562,1761],[3197,1762],[2505,1763],[2507,1764],[2420,1765],[2498,1766],[2418,1767],[2500,1768],[2503,1737],[2504,1737],[2276,1769],[2278,1770],[2273,2],[2274,2],[2277,2],[2275,2],[2499,1771],[2419,1772],[2502,1773],[2506,1774],[2501,1775],[3314,1776],[3315,1777],[3124,2],[3316,1778],[3125,1779],[4563,258],[4564,1780],[3135,1781],[3134,1782],[3313,1783],[3812,1784],[3816,1785],[4024,1786],[3814,1787],[3815,1737],[1760,586],[3813,1788],[3810,1789],[4023,1790],[3811,1791],[2577,1792],[2578,1793],[2512,1794],[2537,2],[2508,1795],[2569,1796],[2511,258],[2510,1795],[2509,1795],[4565,1797],[3119,2],[1702,2],[1734,2],[1733,2],[4346,1798],[4347,1799],[4344,1800],[1614,1801],[4566,1802],[4567,1802],[4568,1802],[4328,588],[1611,588],[1612,1802],[1613,1803],[2589,258],[4569,1804],[3120,1805],[4349,1806],[4570,1807],[3128,1808],[2590,1809],[1726,1810],[2567,1811],[4571,2],[1731,1812],[4572,1813],[4551,1814],[4553,1815],[1609,1816],[4552,1817],[1730,1818],[4575,1819],[4574,1820],[4573,1821],[1728,258],[4345,1822],[1732,1823],[4577,1824],[4576,258],[1719,258],[1729,1788],[4348,1825],[4350,1826],[2240,2],[4332,1827],[1725,1828],[3784,1829],[3121,1830],[2624,2],[3122,2],[3118,1756],[2568,1831],[4578,1832],[3113,1833],[4554,1834],[4556,599],[3600,1835],[1690,1836],[1727,2],[4579,1837],[3142,1838],[3143,1839],[3136,1840],[3133,1841],[3131,1737],[3139,1737],[3137,1737],[3132,1842],[3141,1843],[2571,1844],[2573,1845],[2572,1846],[2570,1847],[4580,1848],[3140,1849],[3138,1850],[2584,1851],[2585,1852],[2579,1795],[2582,1795],[2583,1795],[2580,1795],[2581,1795],[3645,1853],[3623,1854],[3644,1855],[3624,1856],[3641,1857],[3637,1858],[3642,1737],[3622,1859],[3635,1860],[3625,1737],[3614,1861],[3617,1737],[4581,1862],[3627,1737],[3636,1863],[3626,1864],[3615,1865],[3618,1866],[3639,1867],[3649,1868],[3628,1869],[3603,1870],[3605,1871],[3632,1872],[3630,1873],[3634,1874],[3607,1875],[3609,1876],[3621,1877],[3647,1878],[3602,1878],[4582,1879],[3612,1880],[3613,1881],[3604,2],[3631,2],[3608,2],[3620,2],[4583,1882],[3606,2],[3611,2],[3610,2],[3629,2],[3633,2],[3648,1883],[3643,1884],[3616,1885],[3638,1886],[3619,1887],[3640,1888],[3646,1889],[3318,1890],[3322,1891],[3320,1892],[3323,1893],[3317,2],[3319,1890],[3321,1890],[3657,1894],[3659,1895],[3652,586],[3654,586],[3650,1896],[3653,1897],[3655,1898],[3651,1896],[3661,1899],[3662,1900],[3660,1901],[3658,1902],[3656,1903],[4584,1904],[4585,1904],[3221,1905],[3212,1906],[3210,1906],[4586,1907],[3209,1908],[4587,1909],[3217,1910],[3216,1911],[4588,1912],[3213,1913],[3211,1914],[3214,1915],[3215,1916],[3219,1917],[3220,473],[3208,1918],[3206,1919],[4589,1920],[3207,1921],[1720,1922],[1721,1923],[1718,1924],[4550,2],[3667,1925],[3668,1926],[3666,1927],[3665,258],[3664,258],[3663,258],[4334,1737],[4335,1737],[4336,1737],[4337,1859],[4338,1859],[4339,1737],[4330,586],[4331,1928],[4333,1929],[4340,2],[4342,1930],[4343,1931],[4341,1932],[4557,1933],[2268,1934],[2272,1935],[2266,1936],[2269,1937],[2247,1938],[2242,1939],[2271,1940],[2241,1941],[2244,2],[2245,1942],[2246,1943],[2248,1944],[2267,1945],[4594,1946],[4593,1947],[4592,1948],[4595,1949],[4591,1950],[4590,1951],[3223,1952],[3222,586],[3233,1953],[3236,1952],[3235,1954],[3237,1955],[3231,1956],[3234,1957],[3232,1958],[3225,1952],[3226,1952],[3227,1952],[3228,1952],[3229,1952],[3230,1952],[4596,1952],[3224,1952],[3774,1959],[3773,586],[3769,1960],[3771,1961],[3770,586],[3776,1962],[3775,1963],[3772,258],[1602,258],[1610,1964],[1591,1965],[1604,1966],[1608,1967],[1607,1968],[1606,258],[1605,1969],[1603,1970],[3168,1971],[3199,1972],[3188,1973],[3167,1896],[3166,1896],[3171,1974],[3198,1975],[3170,1976],[3200,1977],[3189,1978],[3172,1979],[3301,1980],[3249,2],[3251,1981],[3250,1982],[3300,1981],[3303,1983],[3304,1984],[3302,1985],[3299,1986],[3783,258],[3788,1987],[3787,1988],[3789,1989],[3785,1990],[3786,258],[4027,1991],[4031,1992],[4030,1993],[4606,1994],[4029,1995],[4025,1846],[4028,1996],[4026,1896],[4317,2],[4319,1997],[4329,1998],[4318,1999],[4607,2],[4316,2000],[4315,1788],[4327,2001],[4326,258],[4610,2002],[4320,586],[4608,2],[4609,2003],[4324,2004],[4321,2005],[4325,2006],[4323,2007],[4322,2008],[2621,2009],[2586,2],[2587,2010],[2593,2011],[2591,2012],[2625,2013],[2622,2014],[2626,2015],[2588,2016],[2623,2017],[2592,2018],[3309,1737],[3310,2019],[4613,2],[4612,2020],[3311,2021],[3307,2022],[3305,1988],[4611,2023],[3306,258],[3312,2024],[3308,2025],[3595,2026],[3596,2026],[3599,2026],[3598,2027],[3601,2028],[3597,2029],[3794,258],[3793,258],[3791,2030],[3795,2031],[3790,2032],[3792,258],[1722,2],[1724,2033],[1723,2034],[1737,2035],[1735,258],[1736,2036],[1738,2037],[1740,2038],[1739,2039],[3804,2040],[3801,2041],[3802,2042],[4614,258],[3807,258],[3800,2043],[3798,2044],[3797,2044],[3799,2044],[3796,586],[3808,2045],[3806,2046],[3805,2047],[3809,2048],[3803,2049],[3114,1757],[3123,2050],[3115,2051],[2574,2052],[3129,2053],[3130,2054],[3126,2055]],"semanticDiagnosticsPerFile":[[3193,[{"start":12318,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'JwtTokenPayload' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'JwtTokenPayload' is not assignable to parameter of type 'Buffer'.","category":1,"code":2345,"next":[{"messageText":"Type 'JwtTokenPayload' is missing the following properties from type 'Buffer': slice, subarray, write, toJSON, and 97 more.","category":1,"code":2740}]}]},{"messageText":"Overload 3 of 3, '(payload: JwtTokenPayload, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]}]],[3197,[{"start":2691,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type '{ sub: string; email: string; role: string; roles: string[]; sid: string; }' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'sub' does not exist in type 'Buffer'.","category":1,"code":2353}]},{"messageText":"Overload 3 of 3, '(payload: { sub: string; email: string; role: string; roles: string[]; sid: string; }, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]}]],[4562,[{"start":2585,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type '{ sub: string; email: string; role: UserRole; sid: string; }' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'sub' does not exist in type 'Buffer'.","category":1,"code":2353}]},{"messageText":"Overload 3 of 3, '(payload: { sub: string; email: string; role: UserRole; sid: string; }, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]}]]],"affectedFilesPendingEmit":[3247,3248,3242,3244,3245,3240,3239,3241,3238,3243,3246,439,4351,438,3153,3155,3152,4558,4559,3148,3149,3147,3146,3144,3145,3150,3154,3151,3201,3191,3204,3192,4560,3190,3202,3203,4561,3194,3205,3193,3127,2575,3165,1703,2576,3169,4562,3197,2505,2507,2420,2498,2418,2500,2503,2504,2276,2278,2273,2274,2277,2275,2499,2419,2502,2506,2501,3314,3315,3124,3316,3125,4563,4564,3135,3134,3313,3812,3816,4024,3814,3815,1760,3813,3810,4023,3811,2577,2578,2512,2537,2508,2569,2511,2510,2509,4565,3119,1702,1734,1733,4346,4347,4344,1614,4566,4567,4568,4328,1611,1612,1613,2589,4569,3120,4349,4570,3128,2590,1726,2567,4571,1731,4572,4551,4553,1609,4552,1730,4575,4574,4573,1728,4345,1732,4577,4576,1719,1729,4348,4350,2240,4332,1725,3784,3121,2624,3122,3118,2568,4578,3113,4554,4556,3600,1690,1727,4579,3142,3143,3136,3133,3131,3139,3137,3132,3141,2571,2573,2572,2570,4580,3140,3138,2584,2585,2579,2582,2583,2580,2581,3645,3623,3644,3624,3641,3637,3642,3622,3635,3625,3614,3617,4581,3627,3636,3626,3615,3618,3639,3649,3628,3603,3605,3632,3630,3634,3607,3609,3621,3647,3602,4582,3612,3613,3604,3631,3608,3620,4583,3606,3611,3610,3629,3633,3648,3643,3616,3638,3619,3640,3646,3318,3322,3320,3323,3317,3319,3321,3657,3659,3652,3654,3650,3653,3655,3651,3661,3662,3660,3658,3656,4584,4585,3221,3212,3210,4586,3209,4587,3217,3216,4588,3213,3211,3214,3215,3219,3220,3208,3206,4589,3207,1720,1721,1718,4550,3667,3668,3666,3665,3664,3663,4334,4335,4336,4337,4338,4339,4330,4331,4333,4340,4342,4343,4341,4557,2268,2272,2266,2269,2247,2242,2271,2241,2244,2245,2246,2248,2267,4594,4593,4592,4595,4591,4590,3223,3222,3233,3236,3235,3237,3231,3234,3232,3225,3226,3227,3228,3229,3230,4596,3224,3774,3773,3769,3771,3770,3776,3775,3772,1602,1610,1591,1604,1608,1607,1606,1605,1603,3168,3199,3188,3167,3166,3171,3198,3170,3200,3189,3172,3301,3249,3251,3250,3300,3303,3304,3302,3299,3783,3788,3787,3789,3785,3786,4027,4031,4030,4606,4029,4025,4028,4026,4317,4319,4329,4318,4607,4316,4315,4327,4326,4610,4320,4608,4609,4324,4321,4325,4323,4322,2621,2586,2587,2593,2591,2625,2622,2626,2588,2623,2592,3309,3310,4613,4612,3311,3307,3305,4611,3306,3312,3308,3595,3596,3599,3598,3601,3597,3794,3793,3791,3795,3790,3792,1722,1724,1723,1737,1735,1736,1738,1740,1739,3804,3801,3802,4614,3807,3800,3798,3797,3799,3796,3808,3806,3805,3809,3803,3114,3123,3115,2574,3129,3130,3126],"version":"6.0.3"} \ No newline at end of file +{"fileNames":["./node_modules/typescript/lib/lib.es5.d.ts","./node_modules/typescript/lib/lib.es2015.d.ts","./node_modules/typescript/lib/lib.es2016.d.ts","./node_modules/typescript/lib/lib.es2017.d.ts","./node_modules/typescript/lib/lib.es2018.d.ts","./node_modules/typescript/lib/lib.es2019.d.ts","./node_modules/typescript/lib/lib.es2020.d.ts","./node_modules/typescript/lib/lib.es2021.d.ts","./node_modules/typescript/lib/lib.dom.d.ts","./node_modules/typescript/lib/lib.es2015.core.d.ts","./node_modules/typescript/lib/lib.es2015.collection.d.ts","./node_modules/typescript/lib/lib.es2015.generator.d.ts","./node_modules/typescript/lib/lib.es2015.iterable.d.ts","./node_modules/typescript/lib/lib.es2015.promise.d.ts","./node_modules/typescript/lib/lib.es2015.proxy.d.ts","./node_modules/typescript/lib/lib.es2015.reflect.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2016.array.include.d.ts","./node_modules/typescript/lib/lib.es2016.intl.d.ts","./node_modules/typescript/lib/lib.es2017.arraybuffer.d.ts","./node_modules/typescript/lib/lib.es2017.date.d.ts","./node_modules/typescript/lib/lib.es2017.object.d.ts","./node_modules/typescript/lib/lib.es2017.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2017.string.d.ts","./node_modules/typescript/lib/lib.es2017.intl.d.ts","./node_modules/typescript/lib/lib.es2017.typedarrays.d.ts","./node_modules/typescript/lib/lib.es2018.asyncgenerator.d.ts","./node_modules/typescript/lib/lib.es2018.asynciterable.d.ts","./node_modules/typescript/lib/lib.es2018.intl.d.ts","./node_modules/typescript/lib/lib.es2018.promise.d.ts","./node_modules/typescript/lib/lib.es2018.regexp.d.ts","./node_modules/typescript/lib/lib.es2019.array.d.ts","./node_modules/typescript/lib/lib.es2019.object.d.ts","./node_modules/typescript/lib/lib.es2019.string.d.ts","./node_modules/typescript/lib/lib.es2019.symbol.d.ts","./node_modules/typescript/lib/lib.es2019.intl.d.ts","./node_modules/typescript/lib/lib.es2020.bigint.d.ts","./node_modules/typescript/lib/lib.es2020.date.d.ts","./node_modules/typescript/lib/lib.es2020.promise.d.ts","./node_modules/typescript/lib/lib.es2020.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2020.string.d.ts","./node_modules/typescript/lib/lib.es2020.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2020.intl.d.ts","./node_modules/typescript/lib/lib.es2020.number.d.ts","./node_modules/typescript/lib/lib.es2021.promise.d.ts","./node_modules/typescript/lib/lib.es2021.string.d.ts","./node_modules/typescript/lib/lib.es2021.weakref.d.ts","./node_modules/typescript/lib/lib.es2021.intl.d.ts","./node_modules/typescript/lib/lib.decorators.d.ts","./node_modules/typescript/lib/lib.decorators.legacy.d.ts","./node_modules/reflect-metadata/index.d.ts","./node_modules/@nestjs/common/decorators/core/bind.decorator.d.ts","./node_modules/@nestjs/common/interfaces/abstract.interface.d.ts","./node_modules/@nestjs/common/interfaces/controllers/controller-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/controllers/controller.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/arguments-host.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/exception-filter.interface.d.ts","./node_modules/rxjs/dist/types/internal/Subscription.d.ts","./node_modules/rxjs/dist/types/internal/Subscriber.d.ts","./node_modules/rxjs/dist/types/internal/Operator.d.ts","./node_modules/rxjs/dist/types/internal/Observable.d.ts","./node_modules/rxjs/dist/types/internal/types.d.ts","./node_modules/rxjs/dist/types/internal/operators/audit.d.ts","./node_modules/rxjs/dist/types/internal/operators/auditTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/buffer.d.ts","./node_modules/rxjs/dist/types/internal/operators/bufferCount.d.ts","./node_modules/rxjs/dist/types/internal/operators/bufferTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/bufferToggle.d.ts","./node_modules/rxjs/dist/types/internal/operators/bufferWhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/catchError.d.ts","./node_modules/rxjs/dist/types/internal/operators/combineLatestAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/combineAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/combineLatest.d.ts","./node_modules/rxjs/dist/types/internal/operators/combineLatestWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/concat.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatMap.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatMapTo.d.ts","./node_modules/rxjs/dist/types/internal/operators/concatWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/connect.d.ts","./node_modules/rxjs/dist/types/internal/operators/count.d.ts","./node_modules/rxjs/dist/types/internal/operators/debounce.d.ts","./node_modules/rxjs/dist/types/internal/operators/debounceTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/defaultIfEmpty.d.ts","./node_modules/rxjs/dist/types/internal/operators/delay.d.ts","./node_modules/rxjs/dist/types/internal/operators/delayWhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/dematerialize.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinct.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinctUntilChanged.d.ts","./node_modules/rxjs/dist/types/internal/operators/distinctUntilKeyChanged.d.ts","./node_modules/rxjs/dist/types/internal/operators/elementAt.d.ts","./node_modules/rxjs/dist/types/internal/operators/endWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/every.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaustAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaust.d.ts","./node_modules/rxjs/dist/types/internal/operators/exhaustMap.d.ts","./node_modules/rxjs/dist/types/internal/operators/expand.d.ts","./node_modules/rxjs/dist/types/internal/operators/filter.d.ts","./node_modules/rxjs/dist/types/internal/operators/finalize.d.ts","./node_modules/rxjs/dist/types/internal/operators/find.d.ts","./node_modules/rxjs/dist/types/internal/operators/findIndex.d.ts","./node_modules/rxjs/dist/types/internal/operators/first.d.ts","./node_modules/rxjs/dist/types/internal/Subject.d.ts","./node_modules/rxjs/dist/types/internal/operators/groupBy.d.ts","./node_modules/rxjs/dist/types/internal/operators/ignoreElements.d.ts","./node_modules/rxjs/dist/types/internal/operators/isEmpty.d.ts","./node_modules/rxjs/dist/types/internal/operators/last.d.ts","./node_modules/rxjs/dist/types/internal/operators/map.d.ts","./node_modules/rxjs/dist/types/internal/operators/mapTo.d.ts","./node_modules/rxjs/dist/types/internal/Notification.d.ts","./node_modules/rxjs/dist/types/internal/operators/materialize.d.ts","./node_modules/rxjs/dist/types/internal/operators/max.d.ts","./node_modules/rxjs/dist/types/internal/operators/merge.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeMap.d.ts","./node_modules/rxjs/dist/types/internal/operators/flatMap.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeMapTo.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeScan.d.ts","./node_modules/rxjs/dist/types/internal/operators/mergeWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/min.d.ts","./node_modules/rxjs/dist/types/internal/observable/ConnectableObservable.d.ts","./node_modules/rxjs/dist/types/internal/operators/multicast.d.ts","./node_modules/rxjs/dist/types/internal/operators/observeOn.d.ts","./node_modules/rxjs/dist/types/internal/operators/onErrorResumeNextWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/pairwise.d.ts","./node_modules/rxjs/dist/types/internal/operators/partition.d.ts","./node_modules/rxjs/dist/types/internal/operators/pluck.d.ts","./node_modules/rxjs/dist/types/internal/operators/publish.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishBehavior.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishLast.d.ts","./node_modules/rxjs/dist/types/internal/operators/publishReplay.d.ts","./node_modules/rxjs/dist/types/internal/operators/race.d.ts","./node_modules/rxjs/dist/types/internal/operators/raceWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/reduce.d.ts","./node_modules/rxjs/dist/types/internal/operators/repeat.d.ts","./node_modules/rxjs/dist/types/internal/operators/repeatWhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/retry.d.ts","./node_modules/rxjs/dist/types/internal/operators/retryWhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/refCount.d.ts","./node_modules/rxjs/dist/types/internal/operators/sample.d.ts","./node_modules/rxjs/dist/types/internal/operators/sampleTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/scan.d.ts","./node_modules/rxjs/dist/types/internal/operators/sequenceEqual.d.ts","./node_modules/rxjs/dist/types/internal/operators/share.d.ts","./node_modules/rxjs/dist/types/internal/operators/shareReplay.d.ts","./node_modules/rxjs/dist/types/internal/operators/single.d.ts","./node_modules/rxjs/dist/types/internal/operators/skip.d.ts","./node_modules/rxjs/dist/types/internal/operators/skipLast.d.ts","./node_modules/rxjs/dist/types/internal/operators/skipUntil.d.ts","./node_modules/rxjs/dist/types/internal/operators/skipWhile.d.ts","./node_modules/rxjs/dist/types/internal/operators/startWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/subscribeOn.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchMap.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchMapTo.d.ts","./node_modules/rxjs/dist/types/internal/operators/switchScan.d.ts","./node_modules/rxjs/dist/types/internal/operators/take.d.ts","./node_modules/rxjs/dist/types/internal/operators/takeLast.d.ts","./node_modules/rxjs/dist/types/internal/operators/takeUntil.d.ts","./node_modules/rxjs/dist/types/internal/operators/takeWhile.d.ts","./node_modules/rxjs/dist/types/internal/operators/tap.d.ts","./node_modules/rxjs/dist/types/internal/operators/throttle.d.ts","./node_modules/rxjs/dist/types/internal/operators/throttleTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/throwIfEmpty.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeInterval.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeout.d.ts","./node_modules/rxjs/dist/types/internal/operators/timeoutWith.d.ts","./node_modules/rxjs/dist/types/internal/operators/timestamp.d.ts","./node_modules/rxjs/dist/types/internal/operators/toArray.d.ts","./node_modules/rxjs/dist/types/internal/operators/window.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowCount.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowTime.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowToggle.d.ts","./node_modules/rxjs/dist/types/internal/operators/windowWhen.d.ts","./node_modules/rxjs/dist/types/internal/operators/withLatestFrom.d.ts","./node_modules/rxjs/dist/types/internal/operators/zip.d.ts","./node_modules/rxjs/dist/types/internal/operators/zipAll.d.ts","./node_modules/rxjs/dist/types/internal/operators/zipWith.d.ts","./node_modules/rxjs/dist/types/operators/index.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/Action.d.ts","./node_modules/rxjs/dist/types/internal/Scheduler.d.ts","./node_modules/rxjs/dist/types/internal/testing/TestMessage.d.ts","./node_modules/rxjs/dist/types/internal/testing/SubscriptionLog.d.ts","./node_modules/rxjs/dist/types/internal/testing/SubscriptionLoggable.d.ts","./node_modules/rxjs/dist/types/internal/testing/ColdObservable.d.ts","./node_modules/rxjs/dist/types/internal/testing/HotObservable.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/AsyncScheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/timerHandle.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/AsyncAction.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/VirtualTimeScheduler.d.ts","./node_modules/rxjs/dist/types/internal/testing/TestScheduler.d.ts","./node_modules/rxjs/dist/types/testing/index.d.ts","./node_modules/rxjs/dist/types/internal/symbol/observable.d.ts","./node_modules/rxjs/dist/types/internal/observable/dom/animationFrames.d.ts","./node_modules/rxjs/dist/types/internal/BehaviorSubject.d.ts","./node_modules/rxjs/dist/types/internal/ReplaySubject.d.ts","./node_modules/rxjs/dist/types/internal/AsyncSubject.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/AsapScheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/asap.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/async.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/QueueScheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/queue.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/AnimationFrameScheduler.d.ts","./node_modules/rxjs/dist/types/internal/scheduler/animationFrame.d.ts","./node_modules/rxjs/dist/types/internal/util/identity.d.ts","./node_modules/rxjs/dist/types/internal/util/pipe.d.ts","./node_modules/rxjs/dist/types/internal/util/noop.d.ts","./node_modules/rxjs/dist/types/internal/util/isObservable.d.ts","./node_modules/rxjs/dist/types/internal/lastValueFrom.d.ts","./node_modules/rxjs/dist/types/internal/firstValueFrom.d.ts","./node_modules/rxjs/dist/types/internal/util/ArgumentOutOfRangeError.d.ts","./node_modules/rxjs/dist/types/internal/util/EmptyError.d.ts","./node_modules/rxjs/dist/types/internal/util/NotFoundError.d.ts","./node_modules/rxjs/dist/types/internal/util/ObjectUnsubscribedError.d.ts","./node_modules/rxjs/dist/types/internal/util/SequenceError.d.ts","./node_modules/rxjs/dist/types/internal/util/UnsubscriptionError.d.ts","./node_modules/rxjs/dist/types/internal/observable/bindCallback.d.ts","./node_modules/rxjs/dist/types/internal/observable/bindNodeCallback.d.ts","./node_modules/rxjs/dist/types/internal/AnyCatcher.d.ts","./node_modules/rxjs/dist/types/internal/observable/combineLatest.d.ts","./node_modules/rxjs/dist/types/internal/observable/concat.d.ts","./node_modules/rxjs/dist/types/internal/observable/connectable.d.ts","./node_modules/rxjs/dist/types/internal/observable/defer.d.ts","./node_modules/rxjs/dist/types/internal/observable/empty.d.ts","./node_modules/rxjs/dist/types/internal/observable/forkJoin.d.ts","./node_modules/rxjs/dist/types/internal/observable/from.d.ts","./node_modules/rxjs/dist/types/internal/observable/fromEvent.d.ts","./node_modules/rxjs/dist/types/internal/observable/fromEventPattern.d.ts","./node_modules/rxjs/dist/types/internal/observable/generate.d.ts","./node_modules/rxjs/dist/types/internal/observable/iif.d.ts","./node_modules/rxjs/dist/types/internal/observable/interval.d.ts","./node_modules/rxjs/dist/types/internal/observable/merge.d.ts","./node_modules/rxjs/dist/types/internal/observable/never.d.ts","./node_modules/rxjs/dist/types/internal/observable/of.d.ts","./node_modules/rxjs/dist/types/internal/observable/onErrorResumeNext.d.ts","./node_modules/rxjs/dist/types/internal/observable/pairs.d.ts","./node_modules/rxjs/dist/types/internal/observable/partition.d.ts","./node_modules/rxjs/dist/types/internal/observable/race.d.ts","./node_modules/rxjs/dist/types/internal/observable/range.d.ts","./node_modules/rxjs/dist/types/internal/observable/throwError.d.ts","./node_modules/rxjs/dist/types/internal/observable/timer.d.ts","./node_modules/rxjs/dist/types/internal/observable/using.d.ts","./node_modules/rxjs/dist/types/internal/observable/zip.d.ts","./node_modules/rxjs/dist/types/internal/scheduled/scheduled.d.ts","./node_modules/rxjs/dist/types/internal/config.d.ts","./node_modules/rxjs/dist/types/index.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/rpc-exception-filter.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/ws-exception-filter.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validation-error.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/execution-context.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/can-activate.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/custom-route-param-factory.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/nest-interceptor.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/paramtype.interface.d.ts","./node_modules/@nestjs/common/interfaces/type.interface.d.ts","./node_modules/@nestjs/common/interfaces/features/pipe-transform.interface.d.ts","./node_modules/@nestjs/common/enums/request-method.enum.d.ts","./node_modules/@nestjs/common/enums/http-status.enum.d.ts","./node_modules/@nestjs/common/enums/shutdown-signal.enum.d.ts","./node_modules/@nestjs/common/enums/version-type.enum.d.ts","./node_modules/@nestjs/common/enums/index.d.ts","./node_modules/@nestjs/common/interfaces/version-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-configuration.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-consumer.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/middleware-config-proxy.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/nest-middleware.interface.d.ts","./node_modules/@nestjs/common/interfaces/middleware/index.d.ts","./node_modules/@nestjs/common/interfaces/global-prefix-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/before-application-shutdown.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-application-bootstrap.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-application-shutdown.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-destroy.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/on-init.interface.d.ts","./node_modules/@nestjs/common/interfaces/hooks/index.d.ts","./node_modules/@nestjs/common/interfaces/http/http-exception-body.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/http-redirect-response.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/cors-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/https-options.interface.d.ts","./node_modules/@nestjs/common/services/logger.service.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-context-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/http-server.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/message-event.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/raw-body-request.interface.d.ts","./node_modules/@nestjs/common/interfaces/http/index.d.ts","./node_modules/@nestjs/common/interfaces/injectable.interface.d.ts","./node_modules/@nestjs/common/interfaces/microservices/nest-hybrid-application-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/forward-reference.interface.d.ts","./node_modules/@nestjs/common/interfaces/scope-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/injection-token.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/optional-factory-dependency.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/provider.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/module-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/dynamic-module.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/introspection-result.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/nest-module.interface.d.ts","./node_modules/@nestjs/common/interfaces/modules/index.d.ts","./node_modules/@nestjs/common/interfaces/nest-application-context.interface.d.ts","./node_modules/@nestjs/common/interfaces/websockets/web-socket-adapter.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-application.interface.d.ts","./node_modules/@nestjs/common/interfaces/nest-microservice.interface.d.ts","./node_modules/@nestjs/common/interfaces/index.d.ts","./node_modules/@nestjs/common/decorators/core/catch.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/controller.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/dependencies.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/exception-filters.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/inject.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/injectable.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/optional.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/set-metadata.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-guards.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-interceptors.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/use-pipes.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/apply-decorators.d.ts","./node_modules/@nestjs/common/decorators/core/version.decorator.d.ts","./node_modules/@nestjs/common/decorators/core/index.d.ts","./node_modules/@nestjs/common/decorators/modules/global.decorator.d.ts","./node_modules/@nestjs/common/decorators/modules/module.decorator.d.ts","./node_modules/@nestjs/common/decorators/modules/index.d.ts","./node_modules/@nestjs/common/decorators/http/request-mapping.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/route-params.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/http-code.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/create-route-param-metadata.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/render.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/header.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/redirect.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/sse.decorator.d.ts","./node_modules/@nestjs/common/decorators/http/index.d.ts","./node_modules/@nestjs/common/decorators/index.d.ts","./node_modules/@nestjs/common/exceptions/http.exception.d.ts","./node_modules/@nestjs/common/exceptions/bad-request.exception.d.ts","./node_modules/@nestjs/common/exceptions/unauthorized.exception.d.ts","./node_modules/@nestjs/common/exceptions/method-not-allowed.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-found.exception.d.ts","./node_modules/@nestjs/common/exceptions/forbidden.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-acceptable.exception.d.ts","./node_modules/@nestjs/common/exceptions/request-timeout.exception.d.ts","./node_modules/@nestjs/common/exceptions/conflict.exception.d.ts","./node_modules/@nestjs/common/exceptions/gone.exception.d.ts","./node_modules/@nestjs/common/exceptions/payload-too-large.exception.d.ts","./node_modules/@nestjs/common/exceptions/unsupported-media-type.exception.d.ts","./node_modules/@nestjs/common/exceptions/unprocessable-entity.exception.d.ts","./node_modules/@nestjs/common/exceptions/internal-server-error.exception.d.ts","./node_modules/@nestjs/common/exceptions/not-implemented.exception.d.ts","./node_modules/@nestjs/common/exceptions/http-version-not-supported.exception.d.ts","./node_modules/@nestjs/common/exceptions/bad-gateway.exception.d.ts","./node_modules/@nestjs/common/exceptions/service-unavailable.exception.d.ts","./node_modules/@nestjs/common/exceptions/gateway-timeout.exception.d.ts","./node_modules/@nestjs/common/exceptions/im-a-teapot.exception.d.ts","./node_modules/@nestjs/common/exceptions/precondition-failed.exception.d.ts","./node_modules/@nestjs/common/exceptions/misdirected.exception.d.ts","./node_modules/@nestjs/common/exceptions/index.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/streamable-options.interface.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/streamable-handler-response.interface.d.ts","./node_modules/@nestjs/common/file-stream/interfaces/index.d.ts","./node_modules/@nestjs/common/services/console-logger.service.d.ts","./node_modules/@nestjs/common/services/index.d.ts","./node_modules/@nestjs/common/file-stream/streamable-file.d.ts","./node_modules/@nestjs/common/file-stream/index.d.ts","./node_modules/@nestjs/common/module-utils/constants.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-async-options.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-cls.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/configurable-module-host.interface.d.ts","./node_modules/@nestjs/common/module-utils/interfaces/index.d.ts","./node_modules/@nestjs/common/module-utils/configurable-module.builder.d.ts","./node_modules/@nestjs/common/module-utils/index.d.ts","./node_modules/@nestjs/common/pipes/default-value.pipe.d.ts","./node_modules/@nestjs/common/interfaces/external/class-transform-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/transformer-package.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validator-options.interface.d.ts","./node_modules/@nestjs/common/interfaces/external/validator-package.interface.d.ts","./node_modules/@nestjs/common/utils/http-error-by-code.util.d.ts","./node_modules/@nestjs/common/pipes/validation.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-array.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-bool.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-int.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-float.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-enum.pipe.d.ts","./node_modules/@nestjs/common/pipes/parse-uuid.pipe.d.ts","./node_modules/@nestjs/common/pipes/file/interfaces/file.interface.d.ts","./node_modules/@nestjs/common/pipes/file/interfaces/index.d.ts","./node_modules/@nestjs/common/pipes/file/file-validator.interface.d.ts","./node_modules/@nestjs/common/pipes/file/file-type.validator.d.ts","./node_modules/@nestjs/common/pipes/file/max-file-size.validator.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file-options.interface.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file.pipe.d.ts","./node_modules/@nestjs/common/pipes/file/parse-file-pipe.builder.d.ts","./node_modules/@nestjs/common/pipes/file/index.d.ts","./node_modules/@nestjs/common/pipes/index.d.ts","./node_modules/@nestjs/common/serializer/class-serializer.interfaces.d.ts","./node_modules/@nestjs/common/serializer/class-serializer.interceptor.d.ts","./node_modules/@nestjs/common/serializer/decorators/serialize-options.decorator.d.ts","./node_modules/@nestjs/common/serializer/decorators/index.d.ts","./node_modules/@nestjs/common/serializer/index.d.ts","./node_modules/@nestjs/common/utils/forward-ref.util.d.ts","./node_modules/@nestjs/common/utils/index.d.ts","./node_modules/@nestjs/common/index.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-basic.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-bearer.decorator.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/open-api-spec.interface.d.ts","./node_modules/@nestjs/swagger/dist/types/swagger-enum.type.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-body.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-consumes.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-cookie.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-exclude-endpoint.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-exclude-controller.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-extra-models.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-header.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-hide-property.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-oauth2.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-operation.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-param.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-produces.decorator.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/schema-object-metadata.interface.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-property.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-query.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-response.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-security.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-use-tags.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/api-extension.decorator.d.ts","./node_modules/@nestjs/swagger/dist/decorators/index.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-ui-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-custom-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/swagger-document-options.interface.d.ts","./node_modules/@nestjs/swagger/dist/interfaces/index.d.ts","./node_modules/@nestjs/swagger/dist/document-builder.d.ts","./node_modules/@nestjs/swagger/dist/swagger-module.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/intersection-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/omit-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/partial-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/pick-type.helper.d.ts","./node_modules/@nestjs/swagger/dist/type-helpers/index.d.ts","./node_modules/@nestjs/swagger/dist/utils/get-schema-path.util.d.ts","./node_modules/@nestjs/swagger/dist/utils/index.d.ts","./node_modules/@nestjs/swagger/dist/index.d.ts","./node_modules/@nestjs/swagger/index.d.ts","./src/app.service.ts","./src/app.controller.ts","./node_modules/@nestjs/core/adapters/http-adapter.d.ts","./node_modules/@nestjs/core/adapters/index.d.ts","./node_modules/@nestjs/common/constants.d.ts","./node_modules/@nestjs/core/inspector/interfaces/edge.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/entrypoint.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/extras.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/node.interface.d.ts","./node_modules/@nestjs/core/injector/settlement-signal.d.ts","./node_modules/@nestjs/core/injector/injector.d.ts","./node_modules/@nestjs/core/inspector/interfaces/serialized-graph-metadata.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/serialized-graph-json.interface.d.ts","./node_modules/@nestjs/core/inspector/serialized-graph.d.ts","./node_modules/@nestjs/core/injector/module-token-factory.d.ts","./node_modules/@nestjs/core/injector/compiler.d.ts","./node_modules/@nestjs/core/injector/modules-container.d.ts","./node_modules/@nestjs/core/injector/container.d.ts","./node_modules/@nestjs/core/injector/instance-links-host.d.ts","./node_modules/@nestjs/core/injector/abstract-instance-resolver.d.ts","./node_modules/@nestjs/core/injector/module-ref.d.ts","./node_modules/@nestjs/core/injector/module.d.ts","./node_modules/@nestjs/core/injector/instance-wrapper.d.ts","./node_modules/@nestjs/core/router/interfaces/exclude-route-metadata.interface.d.ts","./node_modules/@nestjs/core/application-config.d.ts","./node_modules/@nestjs/core/constants.d.ts","./node_modules/@nestjs/core/discovery/discovery-module.d.ts","./node_modules/@nestjs/core/discovery/discovery-service.d.ts","./node_modules/@nestjs/core/discovery/index.d.ts","./node_modules/@nestjs/core/helpers/http-adapter-host.d.ts","./node_modules/@nestjs/core/exceptions/base-exception-filter.d.ts","./node_modules/@nestjs/core/exceptions/index.d.ts","./node_modules/@nestjs/core/helpers/context-id-factory.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/exception-filter-metadata.interface.d.ts","./node_modules/@nestjs/core/exceptions/exceptions-handler.d.ts","./node_modules/@nestjs/core/router/router-proxy.d.ts","./node_modules/@nestjs/core/helpers/context-creator.d.ts","./node_modules/@nestjs/core/exceptions/base-exception-filter-context.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/rpc-exception-filter-metadata.interface.d.ts","./node_modules/@nestjs/common/interfaces/exceptions/index.d.ts","./node_modules/@nestjs/core/exceptions/external-exception-filter.d.ts","./node_modules/@nestjs/core/exceptions/external-exceptions-handler.d.ts","./node_modules/@nestjs/core/exceptions/external-exception-filter-context.d.ts","./node_modules/@nestjs/core/guards/constants.d.ts","./node_modules/@nestjs/core/helpers/execution-context-host.d.ts","./node_modules/@nestjs/core/guards/guards-consumer.d.ts","./node_modules/@nestjs/core/guards/guards-context-creator.d.ts","./node_modules/@nestjs/core/guards/index.d.ts","./node_modules/@nestjs/core/interceptors/interceptors-consumer.d.ts","./node_modules/@nestjs/core/interceptors/interceptors-context-creator.d.ts","./node_modules/@nestjs/core/interceptors/index.d.ts","./node_modules/@nestjs/common/enums/route-paramtypes.enum.d.ts","./node_modules/@nestjs/core/pipes/params-token-factory.d.ts","./node_modules/@nestjs/core/pipes/pipes-consumer.d.ts","./node_modules/@nestjs/core/pipes/pipes-context-creator.d.ts","./node_modules/@nestjs/core/pipes/index.d.ts","./node_modules/@nestjs/core/helpers/context-utils.d.ts","./node_modules/@nestjs/core/injector/inquirer/inquirer-constants.d.ts","./node_modules/@nestjs/core/injector/inquirer/index.d.ts","./node_modules/@nestjs/core/interfaces/module-definition.interface.d.ts","./node_modules/@nestjs/core/interfaces/module-override.interface.d.ts","./node_modules/@nestjs/core/inspector/interfaces/enhancer-metadata-cache-entry.interface.d.ts","./node_modules/@nestjs/core/inspector/graph-inspector.d.ts","./node_modules/@nestjs/core/metadata-scanner.d.ts","./node_modules/@nestjs/core/scanner.d.ts","./node_modules/@nestjs/core/injector/instance-loader.d.ts","./node_modules/@nestjs/core/injector/lazy-module-loader/lazy-module-loader-options.interface.d.ts","./node_modules/@nestjs/core/injector/lazy-module-loader/lazy-module-loader.d.ts","./node_modules/@nestjs/core/injector/index.d.ts","./node_modules/@nestjs/core/helpers/interfaces/external-handler-metadata.interface.d.ts","./node_modules/@nestjs/core/helpers/interfaces/params-metadata.interface.d.ts","./node_modules/@nestjs/core/helpers/external-context-creator.d.ts","./node_modules/@nestjs/core/helpers/index.d.ts","./node_modules/@nestjs/core/inspector/initialize-on-preview.allowlist.d.ts","./node_modules/@nestjs/core/inspector/partial-graph.host.d.ts","./node_modules/@nestjs/core/inspector/index.d.ts","./node_modules/@nestjs/core/middleware/route-info-path-extractor.d.ts","./node_modules/@nestjs/core/middleware/routes-mapper.d.ts","./node_modules/@nestjs/core/middleware/builder.d.ts","./node_modules/@nestjs/core/middleware/index.d.ts","./node_modules/@nestjs/core/nest-application-context.d.ts","./node_modules/@nestjs/core/nest-application.d.ts","./node_modules/@nestjs/common/interfaces/microservices/nest-microservice-options.interface.d.ts","./node_modules/@nestjs/core/nest-factory.d.ts","./node_modules/@nestjs/core/repl/repl.d.ts","./node_modules/@nestjs/core/repl/index.d.ts","./node_modules/@nestjs/core/router/interfaces/routes.interface.d.ts","./node_modules/@nestjs/core/router/interfaces/index.d.ts","./node_modules/@nestjs/core/router/request/request-constants.d.ts","./node_modules/@nestjs/core/router/request/index.d.ts","./node_modules/@nestjs/core/router/router-module.d.ts","./node_modules/@nestjs/core/router/index.d.ts","./node_modules/@nestjs/core/services/reflector.service.d.ts","./node_modules/@nestjs/core/services/index.d.ts","./node_modules/@nestjs/core/index.d.ts","./node_modules/@nestjs/config/dist/conditional.module.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-change-event.interface.d.ts","./node_modules/@nestjs/config/dist/types/config-object.type.d.ts","./node_modules/@nestjs/config/dist/types/config.type.d.ts","./node_modules/@nestjs/config/dist/types/no-infer.type.d.ts","./node_modules/@nestjs/config/dist/types/path-value.type.d.ts","./node_modules/@nestjs/config/dist/types/index.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-factory.interface.d.ts","./node_modules/@types/node/compatibility/disposable.d.ts","./node_modules/@types/node/compatibility/indexable.d.ts","./node_modules/@types/node/compatibility/iterators.d.ts","./node_modules/@types/node/compatibility/index.d.ts","./node_modules/@types/node/globals.typedarray.d.ts","./node_modules/@types/node/buffer.buffer.d.ts","./node_modules/@types/node/globals.d.ts","./node_modules/@types/node/web-globals/abortcontroller.d.ts","./node_modules/@types/node/web-globals/domexception.d.ts","./node_modules/@types/node/web-globals/events.d.ts","./node_modules/buffer/index.d.ts","./node_modules/undici-types/header.d.ts","./node_modules/undici-types/readable.d.ts","./node_modules/undici-types/file.d.ts","./node_modules/undici-types/fetch.d.ts","./node_modules/undici-types/formdata.d.ts","./node_modules/undici-types/connector.d.ts","./node_modules/undici-types/client.d.ts","./node_modules/undici-types/errors.d.ts","./node_modules/undici-types/dispatcher.d.ts","./node_modules/undici-types/global-dispatcher.d.ts","./node_modules/undici-types/global-origin.d.ts","./node_modules/undici-types/pool-stats.d.ts","./node_modules/undici-types/pool.d.ts","./node_modules/undici-types/handlers.d.ts","./node_modules/undici-types/balanced-pool.d.ts","./node_modules/undici-types/agent.d.ts","./node_modules/undici-types/mock-interceptor.d.ts","./node_modules/undici-types/mock-agent.d.ts","./node_modules/undici-types/mock-client.d.ts","./node_modules/undici-types/mock-pool.d.ts","./node_modules/undici-types/mock-errors.d.ts","./node_modules/undici-types/proxy-agent.d.ts","./node_modules/undici-types/env-http-proxy-agent.d.ts","./node_modules/undici-types/retry-handler.d.ts","./node_modules/undici-types/retry-agent.d.ts","./node_modules/undici-types/api.d.ts","./node_modules/undici-types/interceptors.d.ts","./node_modules/undici-types/util.d.ts","./node_modules/undici-types/cookies.d.ts","./node_modules/undici-types/patch.d.ts","./node_modules/undici-types/websocket.d.ts","./node_modules/undici-types/eventsource.d.ts","./node_modules/undici-types/filereader.d.ts","./node_modules/undici-types/diagnostics-channel.d.ts","./node_modules/undici-types/content-type.d.ts","./node_modules/undici-types/cache.d.ts","./node_modules/undici-types/index.d.ts","./node_modules/@types/node/web-globals/fetch.d.ts","./node_modules/@types/node/assert.d.ts","./node_modules/@types/node/assert/strict.d.ts","./node_modules/@types/node/async_hooks.d.ts","./node_modules/@types/node/buffer.d.ts","./node_modules/@types/node/child_process.d.ts","./node_modules/@types/node/cluster.d.ts","./node_modules/@types/node/console.d.ts","./node_modules/@types/node/constants.d.ts","./node_modules/@types/node/crypto.d.ts","./node_modules/@types/node/dgram.d.ts","./node_modules/@types/node/diagnostics_channel.d.ts","./node_modules/@types/node/dns.d.ts","./node_modules/@types/node/dns/promises.d.ts","./node_modules/@types/node/domain.d.ts","./node_modules/@types/node/events.d.ts","./node_modules/@types/node/fs.d.ts","./node_modules/@types/node/fs/promises.d.ts","./node_modules/@types/node/http.d.ts","./node_modules/@types/node/http2.d.ts","./node_modules/@types/node/https.d.ts","./node_modules/@types/node/inspector.generated.d.ts","./node_modules/@types/node/module.d.ts","./node_modules/@types/node/net.d.ts","./node_modules/@types/node/os.d.ts","./node_modules/@types/node/path.d.ts","./node_modules/@types/node/perf_hooks.d.ts","./node_modules/@types/node/process.d.ts","./node_modules/@types/node/punycode.d.ts","./node_modules/@types/node/querystring.d.ts","./node_modules/@types/node/readline.d.ts","./node_modules/@types/node/readline/promises.d.ts","./node_modules/@types/node/repl.d.ts","./node_modules/@types/node/sea.d.ts","./node_modules/@types/node/stream.d.ts","./node_modules/@types/node/stream/promises.d.ts","./node_modules/@types/node/stream/consumers.d.ts","./node_modules/@types/node/stream/web.d.ts","./node_modules/@types/node/string_decoder.d.ts","./node_modules/@types/node/test.d.ts","./node_modules/@types/node/timers.d.ts","./node_modules/@types/node/timers/promises.d.ts","./node_modules/@types/node/tls.d.ts","./node_modules/@types/node/trace_events.d.ts","./node_modules/@types/node/tty.d.ts","./node_modules/@types/node/url.d.ts","./node_modules/@types/node/util.d.ts","./node_modules/@types/node/v8.d.ts","./node_modules/@types/node/vm.d.ts","./node_modules/@types/node/wasi.d.ts","./node_modules/@types/node/worker_threads.d.ts","./node_modules/@types/node/zlib.d.ts","./node_modules/@types/node/index.d.ts","./node_modules/dotenv-expand/lib/main.d.ts","./node_modules/@nestjs/config/dist/interfaces/config-module-options.interface.d.ts","./node_modules/@nestjs/config/dist/interfaces/index.d.ts","./node_modules/@nestjs/config/dist/config.module.d.ts","./node_modules/@nestjs/config/dist/config.service.d.ts","./node_modules/@nestjs/config/dist/utils/register-as.util.d.ts","./node_modules/@nestjs/config/dist/utils/get-config-token.util.d.ts","./node_modules/@nestjs/config/dist/utils/index.d.ts","./node_modules/@nestjs/config/dist/index.d.ts","./node_modules/@nestjs/config/index.d.ts","./node_modules/typeorm/metadata/types/RelationTypes.d.ts","./node_modules/typeorm/metadata/types/DeferrableType.d.ts","./node_modules/typeorm/metadata/types/OnDeleteType.d.ts","./node_modules/typeorm/metadata/types/OnUpdateType.d.ts","./node_modules/typeorm/decorator/options/RelationOptions.d.ts","./node_modules/typeorm/metadata/types/PropertyTypeInFunction.d.ts","./node_modules/typeorm/common/ObjectType.d.ts","./node_modules/typeorm/common/EntityTarget.d.ts","./node_modules/typeorm/metadata/types/RelationTypeInFunction.d.ts","./node_modules/typeorm/metadata-args/RelationMetadataArgs.d.ts","./node_modules/typeorm/driver/types/ColumnTypes.d.ts","./node_modules/typeorm/decorator/options/ValueTransformer.d.ts","./node_modules/typeorm/decorator/options/ColumnCommonOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnOptions.d.ts","./node_modules/typeorm/metadata-args/types/ColumnMode.d.ts","./node_modules/typeorm/metadata-args/ColumnMetadataArgs.d.ts","./node_modules/typeorm/common/ObjectLiteral.d.ts","./node_modules/typeorm/schema-builder/options/TableColumnOptions.d.ts","./node_modules/typeorm/schema-builder/table/TableColumn.d.ts","./node_modules/typeorm/schema-builder/options/ViewOptions.d.ts","./node_modules/typeorm/schema-builder/view/View.d.ts","./node_modules/typeorm/naming-strategy/NamingStrategyInterface.d.ts","./node_modules/typeorm/metadata/ForeignKeyMetadata.d.ts","./node_modules/typeorm/metadata/RelationMetadata.d.ts","./node_modules/typeorm/metadata-args/EmbeddedMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/RelationIdMetadataArgs.d.ts","./node_modules/typeorm/metadata/RelationIdMetadata.d.ts","./node_modules/typeorm/metadata/RelationCountMetadata.d.ts","./node_modules/typeorm/metadata/types/EventListenerTypes.d.ts","./node_modules/typeorm/metadata-args/EntityListenerMetadataArgs.d.ts","./node_modules/typeorm/metadata/EntityListenerMetadata.d.ts","./node_modules/typeorm/metadata-args/UniqueMetadataArgs.d.ts","./node_modules/typeorm/metadata/UniqueMetadata.d.ts","./node_modules/typeorm/metadata/EmbeddedMetadata.d.ts","./node_modules/typeorm/metadata/ColumnMetadata.d.ts","./node_modules/typeorm/driver/types/CteCapabilities.d.ts","./node_modules/typeorm/driver/types/MappedColumnTypes.d.ts","./node_modules/typeorm/driver/Query.d.ts","./node_modules/typeorm/driver/SqlInMemory.d.ts","./node_modules/typeorm/schema-builder/SchemaBuilder.d.ts","./node_modules/typeorm/driver/types/DataTypeDefaults.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaIndexOptions.d.ts","./node_modules/typeorm/driver/types/GeoJsonTypes.d.ts","./node_modules/typeorm/decorator/options/SpatialColumnOptions.d.ts","./node_modules/typeorm/decorator/options/ForeignKeyOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaColumnForeignKeyOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaColumnOptions.d.ts","./node_modules/typeorm/decorator/options/JoinColumnOptions.d.ts","./node_modules/typeorm/decorator/options/JoinTableMultipleColumnsOptions.d.ts","./node_modules/typeorm/decorator/options/JoinTableOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaRelationOptions.d.ts","./node_modules/typeorm/find-options/OrderByCondition.d.ts","./node_modules/typeorm/metadata/types/TableTypes.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaUniqueOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaCheckOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaExclusionOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaInheritanceOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaRelationIdOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaForeignKeyOptions.d.ts","./node_modules/typeorm/metadata/types/TreeTypes.d.ts","./node_modules/typeorm/metadata/types/ClosureTreeOptions.d.ts","./node_modules/typeorm/metadata-args/TreeMetadataArgs.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaOptions.d.ts","./node_modules/typeorm/entity-schema/EntitySchema.d.ts","./node_modules/typeorm/logger/Logger.d.ts","./node_modules/typeorm/logger/LoggerOptions.d.ts","./node_modules/typeorm/driver/types/DatabaseType.d.ts","./node_modules/typeorm/cache/QueryResultCacheOptions.d.ts","./node_modules/typeorm/cache/QueryResultCache.d.ts","./node_modules/typeorm/common/MixedList.d.ts","./node_modules/typeorm/data-source/BaseDataSourceOptions.d.ts","./node_modules/typeorm/driver/types/ReplicationMode.d.ts","./node_modules/typeorm/schema-builder/options/TableForeignKeyOptions.d.ts","./node_modules/typeorm/schema-builder/table/TableForeignKey.d.ts","./node_modules/typeorm/driver/types/UpsertType.d.ts","./node_modules/typeorm/driver/Driver.d.ts","./node_modules/typeorm/find-options/JoinOptions.d.ts","./node_modules/typeorm/find-options/FindOperatorType.d.ts","./node_modules/typeorm/find-options/FindOperator.d.ts","./node_modules/typeorm/platform/PlatformTools.d.ts","./node_modules/typeorm/driver/mongodb/bson.typings.d.ts","./node_modules/typeorm/driver/mongodb/typings.d.ts","./node_modules/typeorm/find-options/EqualOperator.d.ts","./node_modules/typeorm/find-options/FindOptionsWhere.d.ts","./node_modules/typeorm/find-options/FindOptionsSelect.d.ts","./node_modules/typeorm/find-options/FindOptionsRelations.d.ts","./node_modules/typeorm/find-options/FindOptionsOrder.d.ts","./node_modules/typeorm/find-options/FindOneOptions.d.ts","./node_modules/typeorm/find-options/FindManyOptions.d.ts","./node_modules/typeorm/common/DeepPartial.d.ts","./node_modules/typeorm/repository/SaveOptions.d.ts","./node_modules/typeorm/repository/RemoveOptions.d.ts","./node_modules/typeorm/find-options/mongodb/MongoFindOneOptions.d.ts","./node_modules/typeorm/find-options/mongodb/MongoFindManyOptions.d.ts","./node_modules/typeorm/schema-builder/options/TableUniqueOptions.d.ts","./node_modules/typeorm/schema-builder/table/TableUnique.d.ts","./node_modules/typeorm/subscriber/BroadcasterResult.d.ts","./node_modules/typeorm/subscriber/event/TransactionCommitEvent.d.ts","./node_modules/typeorm/subscriber/event/TransactionRollbackEvent.d.ts","./node_modules/typeorm/subscriber/event/TransactionStartEvent.d.ts","./node_modules/typeorm/subscriber/event/UpdateEvent.d.ts","./node_modules/typeorm/subscriber/event/RemoveEvent.d.ts","./node_modules/typeorm/subscriber/event/InsertEvent.d.ts","./node_modules/typeorm/subscriber/event/LoadEvent.d.ts","./node_modules/typeorm/subscriber/event/SoftRemoveEvent.d.ts","./node_modules/typeorm/subscriber/event/RecoverEvent.d.ts","./node_modules/typeorm/subscriber/event/QueryEvent.d.ts","./node_modules/typeorm/subscriber/EntitySubscriberInterface.d.ts","./node_modules/typeorm/subscriber/Broadcaster.d.ts","./node_modules/typeorm/schema-builder/options/TableCheckOptions.d.ts","./node_modules/typeorm/metadata-args/CheckMetadataArgs.d.ts","./node_modules/typeorm/metadata/CheckMetadata.d.ts","./node_modules/typeorm/schema-builder/table/TableCheck.d.ts","./node_modules/typeorm/schema-builder/options/TableExclusionOptions.d.ts","./node_modules/typeorm/metadata-args/ExclusionMetadataArgs.d.ts","./node_modules/typeorm/metadata/ExclusionMetadata.d.ts","./node_modules/typeorm/schema-builder/table/TableExclusion.d.ts","./node_modules/typeorm/driver/mongodb/MongoQueryRunner.d.ts","./node_modules/typeorm/query-builder/QueryPartialEntity.d.ts","./node_modules/typeorm/query-runner/QueryResult.d.ts","./node_modules/typeorm/query-builder/result/InsertResult.d.ts","./node_modules/typeorm/query-builder/result/UpdateResult.d.ts","./node_modules/typeorm/query-builder/result/DeleteResult.d.ts","./node_modules/typeorm/entity-manager/MongoEntityManager.d.ts","./node_modules/typeorm/repository/MongoRepository.d.ts","./node_modules/typeorm/find-options/FindTreeOptions.d.ts","./node_modules/typeorm/repository/TreeRepository.d.ts","./node_modules/typeorm/query-builder/transformer/PlainObjectToNewEntityTransformer.d.ts","./node_modules/typeorm/driver/types/IsolationLevel.d.ts","./node_modules/typeorm/query-builder/WhereExpressionBuilder.d.ts","./node_modules/typeorm/query-builder/Brackets.d.ts","./node_modules/typeorm/query-builder/InsertOrUpdateOptions.d.ts","./node_modules/typeorm/repository/UpsertOptions.d.ts","./node_modules/typeorm/common/PickKeysByType.d.ts","./node_modules/typeorm/entity-manager/EntityManager.d.ts","./node_modules/typeorm/repository/Repository.d.ts","./node_modules/typeorm/migration/MigrationInterface.d.ts","./node_modules/typeorm/migration/Migration.d.ts","./node_modules/typeorm/driver/cockroachdb/CockroachConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/cockroachdb/CockroachConnectionOptions.d.ts","./node_modules/typeorm/driver/mysql/MysqlConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/mysql/MysqlConnectionOptions.d.ts","./node_modules/typeorm/driver/postgres/PostgresConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/postgres/PostgresConnectionOptions.d.ts","./node_modules/typeorm/driver/sqlite/SqliteConnectionOptions.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/DefaultAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryAccessTokenAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryDefaultAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryMsiAppServiceAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryMsiVmAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryPasswordAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/AzureActiveDirectoryServicePrincipalSecret.d.ts","./node_modules/typeorm/driver/sqlserver/authentication/NtlmAuthentication.d.ts","./node_modules/typeorm/driver/sqlserver/SqlServerConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/sqlserver/SqlServerConnectionOptions.d.ts","./node_modules/typeorm/driver/oracle/OracleConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/oracle/OracleConnectionOptions.d.ts","./node_modules/typeorm/driver/mongodb/MongoConnectionOptions.d.ts","./node_modules/typeorm/driver/cordova/CordovaConnectionOptions.d.ts","./node_modules/typeorm/driver/sqljs/SqljsConnectionOptions.d.ts","./node_modules/typeorm/driver/react-native/ReactNativeConnectionOptions.d.ts","./node_modules/typeorm/driver/nativescript/NativescriptConnectionOptions.d.ts","./node_modules/typeorm/driver/expo/ExpoConnectionOptions.d.ts","./node_modules/typeorm/driver/aurora-mysql/AuroraMysqlConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/aurora-mysql/AuroraMysqlConnectionOptions.d.ts","./node_modules/typeorm/driver/sap/SapConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/sap/SapConnectionOptions.d.ts","./node_modules/typeorm/driver/aurora-postgres/AuroraPostgresConnectionOptions.d.ts","./node_modules/typeorm/driver/better-sqlite3/BetterSqlite3ConnectionOptions.d.ts","./node_modules/typeorm/driver/capacitor/CapacitorConnectionOptions.d.ts","./node_modules/typeorm/connection/BaseConnectionOptions.d.ts","./node_modules/typeorm/driver/spanner/SpannerConnectionCredentialsOptions.d.ts","./node_modules/typeorm/driver/spanner/SpannerConnectionOptions.d.ts","./node_modules/typeorm/data-source/DataSourceOptions.d.ts","./node_modules/typeorm/entity-manager/SqljsEntityManager.d.ts","./node_modules/typeorm/query-builder/RelationLoader.d.ts","./node_modules/typeorm/query-builder/RelationIdLoader.d.ts","./node_modules/typeorm/data-source/DataSource.d.ts","./node_modules/typeorm/metadata-args/TableMetadataArgs.d.ts","./node_modules/typeorm/metadata/EntityMetadata.d.ts","./node_modules/typeorm/metadata-args/IndexMetadataArgs.d.ts","./node_modules/typeorm/metadata/IndexMetadata.d.ts","./node_modules/typeorm/schema-builder/options/TableIndexOptions.d.ts","./node_modules/typeorm/schema-builder/table/TableIndex.d.ts","./node_modules/typeorm/schema-builder/options/TableOptions.d.ts","./node_modules/typeorm/schema-builder/table/Table.d.ts","./node_modules/typeorm/query-runner/QueryRunner.d.ts","./node_modules/typeorm/query-builder/QueryBuilderCte.d.ts","./node_modules/typeorm/query-builder/Alias.d.ts","./node_modules/typeorm/query-builder/JoinAttribute.d.ts","./node_modules/typeorm/query-builder/relation-id/RelationIdAttribute.d.ts","./node_modules/typeorm/query-builder/relation-count/RelationCountAttribute.d.ts","./node_modules/typeorm/query-builder/SelectQuery.d.ts","./node_modules/typeorm/query-builder/SelectQueryBuilderOption.d.ts","./node_modules/typeorm/query-builder/WhereClause.d.ts","./node_modules/typeorm/query-builder/QueryExpressionMap.d.ts","./node_modules/typeorm/query-builder/UpdateQueryBuilder.d.ts","./node_modules/typeorm/query-builder/DeleteQueryBuilder.d.ts","./node_modules/typeorm/query-builder/SoftDeleteQueryBuilder.d.ts","./node_modules/typeorm/query-builder/InsertQueryBuilder.d.ts","./node_modules/typeorm/query-builder/RelationQueryBuilder.d.ts","./node_modules/typeorm/query-builder/NotBrackets.d.ts","./node_modules/typeorm/query-builder/QueryBuilder.d.ts","./node_modules/typeorm/query-builder/SelectQueryBuilder.d.ts","./node_modules/typeorm/metadata-args/RelationCountMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/NamingStrategyMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/JoinColumnMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/JoinTableMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/EntitySubscriberMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/InheritanceMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/DiscriminatorValueMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/EntityRepositoryMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/TransactionEntityMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/TransactionRepositoryMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/GeneratedMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/ForeignKeyMetadataArgs.d.ts","./node_modules/typeorm/metadata-args/MetadataArgsStorage.d.ts","./node_modules/typeorm/connection/ConnectionManager.d.ts","./node_modules/typeorm/globals.d.ts","./node_modules/typeorm/container.d.ts","./node_modules/typeorm/common/RelationType.d.ts","./node_modules/typeorm/error/TypeORMError.d.ts","./node_modules/typeorm/error/CannotReflectMethodParameterTypeError.d.ts","./node_modules/typeorm/error/AlreadyHasActiveConnectionError.d.ts","./node_modules/typeorm/persistence/SubjectChangeMap.d.ts","./node_modules/typeorm/persistence/Subject.d.ts","./node_modules/typeorm/error/SubjectWithoutIdentifierError.d.ts","./node_modules/typeorm/error/CannotConnectAlreadyConnectedError.d.ts","./node_modules/typeorm/error/LockNotSupportedOnGivenDriverError.d.ts","./node_modules/typeorm/error/ConnectionIsNotSetError.d.ts","./node_modules/typeorm/error/CannotCreateEntityIdMapError.d.ts","./node_modules/typeorm/error/MetadataAlreadyExistsError.d.ts","./node_modules/typeorm/error/CannotDetermineEntityError.d.ts","./node_modules/typeorm/error/UpdateValuesMissingError.d.ts","./node_modules/typeorm/error/TreeRepositoryNotSupportedError.d.ts","./node_modules/typeorm/error/CustomRepositoryNotFoundError.d.ts","./node_modules/typeorm/error/TransactionNotStartedError.d.ts","./node_modules/typeorm/error/TransactionAlreadyStartedError.d.ts","./node_modules/typeorm/error/EntityNotFoundError.d.ts","./node_modules/typeorm/error/EntityMetadataNotFoundError.d.ts","./node_modules/typeorm/error/MustBeEntityError.d.ts","./node_modules/typeorm/error/OptimisticLockVersionMismatchError.d.ts","./node_modules/typeorm/error/LimitOnUpdateNotSupportedError.d.ts","./node_modules/typeorm/error/PrimaryColumnCannotBeNullableError.d.ts","./node_modules/typeorm/error/CustomRepositoryCannotInheritRepositoryError.d.ts","./node_modules/typeorm/error/QueryRunnerProviderAlreadyReleasedError.d.ts","./node_modules/typeorm/error/CannotAttachTreeChildrenEntityError.d.ts","./node_modules/typeorm/error/CustomRepositoryDoesNotHaveEntityError.d.ts","./node_modules/typeorm/error/MissingDeleteDateColumnError.d.ts","./node_modules/typeorm/error/NoConnectionForRepositoryError.d.ts","./node_modules/typeorm/error/CircularRelationsError.d.ts","./node_modules/typeorm/error/ReturningStatementNotSupportedError.d.ts","./node_modules/typeorm/error/UsingJoinTableIsNotAllowedError.d.ts","./node_modules/typeorm/error/MissingJoinColumnError.d.ts","./node_modules/typeorm/error/MissingPrimaryColumnError.d.ts","./node_modules/typeorm/error/EntityPropertyNotFoundError.d.ts","./node_modules/typeorm/error/MissingDriverError.d.ts","./node_modules/typeorm/error/DriverPackageNotInstalledError.d.ts","./node_modules/typeorm/error/CannotGetEntityManagerNotConnectedError.d.ts","./node_modules/typeorm/error/ConnectionNotFoundError.d.ts","./node_modules/typeorm/error/NoVersionOrUpdateDateColumnError.d.ts","./node_modules/typeorm/error/InsertValuesMissingError.d.ts","./node_modules/typeorm/error/OptimisticLockCanNotBeUsedError.d.ts","./node_modules/typeorm/error/MetadataWithSuchNameAlreadyExistsError.d.ts","./node_modules/typeorm/error/DriverOptionNotSetError.d.ts","./node_modules/typeorm/error/FindRelationsNotFoundError.d.ts","./node_modules/typeorm/error/PessimisticLockTransactionRequiredError.d.ts","./node_modules/typeorm/error/RepositoryNotTreeError.d.ts","./node_modules/typeorm/error/DataTypeNotSupportedError.d.ts","./node_modules/typeorm/error/InitializedRelationError.d.ts","./node_modules/typeorm/error/MissingJoinTableError.d.ts","./node_modules/typeorm/error/QueryFailedError.d.ts","./node_modules/typeorm/error/NoNeedToReleaseEntityManagerError.d.ts","./node_modules/typeorm/error/UsingJoinColumnOnlyOnOneSideAllowedError.d.ts","./node_modules/typeorm/error/UsingJoinTableOnlyOnOneSideAllowedError.d.ts","./node_modules/typeorm/error/SubjectRemovedAndUpdatedError.d.ts","./node_modules/typeorm/error/PersistedEntityNotFoundError.d.ts","./node_modules/typeorm/error/UsingJoinColumnIsNotAllowedError.d.ts","./node_modules/typeorm/error/ColumnTypeUndefinedError.d.ts","./node_modules/typeorm/error/QueryRunnerAlreadyReleasedError.d.ts","./node_modules/typeorm/error/OffsetWithoutLimitNotSupportedError.d.ts","./node_modules/typeorm/error/CannotExecuteNotConnectedError.d.ts","./node_modules/typeorm/error/NoConnectionOptionError.d.ts","./node_modules/typeorm/error/ForbiddenTransactionModeOverrideError.d.ts","./node_modules/typeorm/error/index.d.ts","./node_modules/typeorm/decorator/options/ColumnEmbeddedOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnEnumOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnHstoreOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnNumericOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnUnsignedOptions.d.ts","./node_modules/typeorm/decorator/options/ColumnWithLengthOptions.d.ts","./node_modules/typeorm/decorator/columns/Column.d.ts","./node_modules/typeorm/decorator/columns/CreateDateColumn.d.ts","./node_modules/typeorm/decorator/columns/DeleteDateColumn.d.ts","./node_modules/typeorm/decorator/options/PrimaryGeneratedColumnNumericOptions.d.ts","./node_modules/typeorm/decorator/options/PrimaryGeneratedColumnUUIDOptions.d.ts","./node_modules/typeorm/decorator/options/PrimaryGeneratedColumnIdentityOptions.d.ts","./node_modules/typeorm/decorator/columns/PrimaryGeneratedColumn.d.ts","./node_modules/typeorm/decorator/columns/PrimaryColumn.d.ts","./node_modules/typeorm/decorator/columns/UpdateDateColumn.d.ts","./node_modules/typeorm/decorator/columns/VersionColumn.d.ts","./node_modules/typeorm/decorator/options/VirtualColumnOptions.d.ts","./node_modules/typeorm/decorator/columns/VirtualColumn.d.ts","./node_modules/typeorm/decorator/options/ViewColumnOptions.d.ts","./node_modules/typeorm/decorator/columns/ViewColumn.d.ts","./node_modules/typeorm/decorator/columns/ObjectIdColumn.d.ts","./node_modules/typeorm/decorator/listeners/AfterInsert.d.ts","./node_modules/typeorm/decorator/listeners/AfterLoad.d.ts","./node_modules/typeorm/decorator/listeners/AfterRemove.d.ts","./node_modules/typeorm/decorator/listeners/AfterSoftRemove.d.ts","./node_modules/typeorm/decorator/listeners/AfterRecover.d.ts","./node_modules/typeorm/decorator/listeners/AfterUpdate.d.ts","./node_modules/typeorm/decorator/listeners/BeforeInsert.d.ts","./node_modules/typeorm/decorator/listeners/BeforeRemove.d.ts","./node_modules/typeorm/decorator/listeners/BeforeSoftRemove.d.ts","./node_modules/typeorm/decorator/listeners/BeforeRecover.d.ts","./node_modules/typeorm/decorator/listeners/BeforeUpdate.d.ts","./node_modules/typeorm/decorator/listeners/EventSubscriber.d.ts","./node_modules/typeorm/decorator/options/IndexOptions.d.ts","./node_modules/typeorm/decorator/options/EntityOptions.d.ts","./node_modules/typeorm/decorator/relations/JoinColumn.d.ts","./node_modules/typeorm/decorator/relations/JoinTable.d.ts","./node_modules/typeorm/decorator/relations/ManyToMany.d.ts","./node_modules/typeorm/decorator/relations/ManyToOne.d.ts","./node_modules/typeorm/decorator/relations/OneToMany.d.ts","./node_modules/typeorm/decorator/relations/OneToOne.d.ts","./node_modules/typeorm/decorator/relations/RelationCount.d.ts","./node_modules/typeorm/decorator/relations/RelationId.d.ts","./node_modules/typeorm/decorator/entity/Entity.d.ts","./node_modules/typeorm/decorator/entity/ChildEntity.d.ts","./node_modules/typeorm/decorator/entity/TableInheritance.d.ts","./node_modules/typeorm/decorator/options/ViewEntityOptions.d.ts","./node_modules/typeorm/decorator/entity-view/ViewEntity.d.ts","./node_modules/typeorm/decorator/tree/TreeLevelColumn.d.ts","./node_modules/typeorm/decorator/tree/TreeParent.d.ts","./node_modules/typeorm/decorator/tree/TreeChildren.d.ts","./node_modules/typeorm/decorator/tree/Tree.d.ts","./node_modules/typeorm/decorator/Index.d.ts","./node_modules/typeorm/decorator/ForeignKey.d.ts","./node_modules/typeorm/decorator/options/UniqueOptions.d.ts","./node_modules/typeorm/decorator/Unique.d.ts","./node_modules/typeorm/decorator/Check.d.ts","./node_modules/typeorm/decorator/Exclusion.d.ts","./node_modules/typeorm/decorator/Generated.d.ts","./node_modules/typeorm/decorator/EntityRepository.d.ts","./node_modules/typeorm/find-options/operator/And.d.ts","./node_modules/typeorm/find-options/operator/Or.d.ts","./node_modules/typeorm/find-options/operator/Any.d.ts","./node_modules/typeorm/find-options/operator/ArrayContainedBy.d.ts","./node_modules/typeorm/find-options/operator/ArrayContains.d.ts","./node_modules/typeorm/find-options/operator/ArrayOverlap.d.ts","./node_modules/typeorm/find-options/operator/Between.d.ts","./node_modules/typeorm/find-options/operator/Equal.d.ts","./node_modules/typeorm/find-options/operator/In.d.ts","./node_modules/typeorm/find-options/operator/IsNull.d.ts","./node_modules/typeorm/find-options/operator/LessThan.d.ts","./node_modules/typeorm/find-options/operator/LessThanOrEqual.d.ts","./node_modules/typeorm/find-options/operator/ILike.d.ts","./node_modules/typeorm/find-options/operator/Like.d.ts","./node_modules/typeorm/find-options/operator/MoreThan.d.ts","./node_modules/typeorm/find-options/operator/MoreThanOrEqual.d.ts","./node_modules/typeorm/find-options/operator/Not.d.ts","./node_modules/typeorm/find-options/operator/Raw.d.ts","./node_modules/typeorm/find-options/operator/JsonContains.d.ts","./node_modules/typeorm/find-options/FindOptionsUtils.d.ts","./node_modules/typeorm/logger/AbstractLogger.d.ts","./node_modules/typeorm/logger/AdvancedConsoleLogger.d.ts","./node_modules/typeorm/logger/FormattedConsoleLogger.d.ts","./node_modules/typeorm/logger/SimpleConsoleLogger.d.ts","./node_modules/typeorm/logger/FileLogger.d.ts","./node_modules/typeorm/repository/AbstractRepository.d.ts","./node_modules/typeorm/data-source/index.d.ts","./node_modules/typeorm/repository/BaseEntity.d.ts","./node_modules/typeorm/driver/sqlserver/MssqlParameter.d.ts","./node_modules/typeorm/connection/ConnectionOptionsReader.d.ts","./node_modules/typeorm/connection/ConnectionOptions.d.ts","./node_modules/typeorm/connection/Connection.d.ts","./node_modules/typeorm/migration/MigrationExecutor.d.ts","./node_modules/typeorm/naming-strategy/DefaultNamingStrategy.d.ts","./node_modules/typeorm/naming-strategy/LegacyOracleNamingStrategy.d.ts","./node_modules/typeorm/entity-schema/EntitySchemaEmbeddedColumnOptions.d.ts","./node_modules/typeorm/schema-builder/RdbmsSchemaBuilder.d.ts","./node_modules/typeorm/util/InstanceChecker.d.ts","./node_modules/typeorm/repository/FindTreesOptions.d.ts","./node_modules/typeorm/util/TreeRepositoryUtils.d.ts","./node_modules/typeorm/index.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/entity-class-or-schema.type.d.ts","./node_modules/@nestjs/typeorm/dist/common/typeorm.decorators.d.ts","./node_modules/@nestjs/typeorm/dist/common/typeorm.utils.d.ts","./node_modules/@nestjs/typeorm/dist/common/index.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/typeorm-options.interface.d.ts","./node_modules/@nestjs/typeorm/dist/interfaces/index.d.ts","./node_modules/@nestjs/typeorm/dist/typeorm.module.d.ts","./node_modules/@nestjs/typeorm/dist/index.d.ts","./node_modules/@nestjs/typeorm/index.d.ts","./node_modules/keyv/dist/index.d.cts","./node_modules/cache-manager/dist/index.d.cts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/command-options.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/lua-script.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_CAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_DELUSER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_DRYRUN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_GENPASS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_GETUSER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_LOAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_LOG_RESET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_LOG.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_SAVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_SETUSER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_USERS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ACL_WHOAMI.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ASKING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/AUTH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BGREWRITEAOF.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BGSAVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_CACHING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_GETNAME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_GETREDIR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_ID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_KILL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_NO-EVICT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_NO-TOUCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_PAUSE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_SETNAME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_TRACKING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_TRACKINGINFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLIENT_UNPAUSE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_ADDSLOTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/generic-transformers.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_ADDSLOTSRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_BUMPEPOCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_COUNT-FAILURE-REPORTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_COUNTKEYSINSLOT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_DELSLOTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_DELSLOTSRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_FAILOVER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_FLUSHSLOTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_FORGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_GETKEYSINSLOT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_KEYSLOT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_LINKS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_MEET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_MYID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_MYSHARDID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_NODES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_REPLICAS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_REPLICATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_RESET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_SAVECONFIG.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_SET-CONFIG-EPOCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_SETSLOT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CLUSTER_SLOTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND_GETKEYS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND_GETKEYSANDFLAGS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND_INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COMMAND.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CONFIG_GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CONFIG_RESETSTAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CONFIG_REWRITE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/CONFIG_SET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DBSIZE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DISCARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ECHO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FAILOVER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FLUSHALL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FLUSHDB.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_DELETE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_DUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_FLUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_KILL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_LIST_WITHCODE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_LOAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_RESTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FUNCTION_STATS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HELLO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/KEYS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LASTSAVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LATENCY_DOCTOR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LATENCY_GRAPH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LATENCY_HISTORY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LATENCY_LATEST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LOLWUT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MEMORY_DOCTOR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MEMORY_MALLOC-STATS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MEMORY_PURGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MEMORY_STATS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MEMORY_USAGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MODULE_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MODULE_LOAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MODULE_UNLOAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MOVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBSUB_CHANNELS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBSUB_NUMPAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBSUB_NUMSUB.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBSUB_SHARDCHANNELS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBSUB_SHARDNUMSUB.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RANDOMKEY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/READONLY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/READWRITE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/REPLICAOF.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RESTORE-ASKING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ROLE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SAVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCAN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCRIPT_DEBUG.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCRIPT_EXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCRIPT_FLUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCRIPT_KILL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCRIPT_LOAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SHUTDOWN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SWAPDB.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/TIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/UNWATCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/WAIT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/APPEND.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BITCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BITFIELD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BITFIELD_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BITOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BITPOS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BLMOVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LMPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BLMPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BLPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BRPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BRPOPLPUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZMPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BZMPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BZPOPMAX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/BZPOPMIN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/COPY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DECR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DECRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/DUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EVAL_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EVAL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EVALSHA.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EVALSHA_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EXPIRE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EXPIREAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/EXPIRETIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FCALL_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/FCALL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEODIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOHASH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOPOS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUS_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUS_RO_WITH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUS_WITH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBER_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBER_RO_WITH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBER_WITH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBERSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEORADIUSSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOSEARCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOSEARCH_WITH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GEOSEARCHSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GETBIT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GETDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GETEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GETRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/GETSET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HEXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HEXPIRE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HEXPIREAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HEXPIRETIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HGETALL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HINCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HINCRBYFLOAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HKEYS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HMGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HPERSIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HPEXPIRE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HPEXPIREAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HPEXPIRETIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HPTTL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HRANDFIELD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HRANDFIELD_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HRANDFIELD_COUNT_WITHVALUES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HSCAN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HSCAN_NOVALUES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HSET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HSETNX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HSTRLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HTTL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/HVALS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/INCR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/INCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/INCRBYFLOAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LCS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LCS_IDX_WITHMATCHLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LCS_IDX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LCS_LEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LINDEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LINSERT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LMOVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPOP_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPOS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPOS_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LPUSHX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LREM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LSET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/LTRIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MIGRATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MSET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/MSETNX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/OBJECT_ENCODING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/OBJECT_FREQ.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/OBJECT_IDLETIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/OBJECT_REFCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PERSIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PEXPIRE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PEXPIREAT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PEXPIRETIME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PFADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PFCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PFMERGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PSETEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PTTL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/PUBLISH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RENAME.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RENAMENX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RESTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RPOP_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RPOPLPUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RPUSH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/RPUSHX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SCARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SDIFF.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SDIFFSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SINTER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SINTERCARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SINTERSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SETBIT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SETEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SETNX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SETRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SISMEMBER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SMEMBERS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SMISMEMBER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SMOVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SORT_RO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SORT_STORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SORT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SPUBLISH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SRANDMEMBER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SRANDMEMBER_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SREM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SSCAN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/STRLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SUNION.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/SUNIONSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/TOUCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/TTL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/TYPE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/UNLINK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/WATCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XACK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XAUTOCLAIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XAUTOCLAIM_JUSTID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XCLAIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XCLAIM_JUSTID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XGROUP_CREATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XGROUP_CREATECONSUMER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XGROUP_DELCONSUMER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XGROUP_DESTROY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XGROUP_SETID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XINFO_CONSUMERS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XINFO_GROUPS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XINFO_STREAM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XPENDING_RANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XPENDING.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XREAD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XREADGROUP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XREVRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XSETID.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/XTRIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZCARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZDIFF.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZDIFF_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZDIFFSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZINCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZINTER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZINTER_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZINTERCARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZINTERSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZLEXCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZMSCORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZPOPMAX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZPOPMAX_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZPOPMIN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZPOPMIN_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANDMEMBER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANDMEMBER_COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANDMEMBER_COUNT_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGE_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGEBYLEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGEBYSCORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGEBYSCORE_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANGESTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZREM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZREMRANGEBYLEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZREMRANGEBYRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZREMRANGEBYSCORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZREVRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZSCAN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZSCORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZUNION.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZUNION_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/commands/ZUNIONSTORE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/commands.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/socket.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/pub-sub.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/commands-queue.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/errors.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/multi-command.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/multi-command.d.ts","./node_modules/generic-pool/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/client/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/commands.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/cluster-slots.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/multi-command.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/lib/cluster/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/client/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/ADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/CARD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/EXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/INSERT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/LOADCHUNK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/MADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/MEXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/RESERVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/bloom/SCANDUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/INCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/INITBYDIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/INITBYPROB.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/MERGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/count-min-sketch/QUERY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/ADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/ADDNX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/DEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/EXISTS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/INSERTNX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/LOADCHUNK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/RESERVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/SCANDUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/cuckoo/INSERT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/ADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/BYREVRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/CDF.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/CREATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/MAX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/MERGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/MIN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/QUANTILE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/RANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/RESET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/REVRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/TRIMMED_MEAN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/t-digest/BYRANK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/ADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/COUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/INCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/LIST_WITHCOUNT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/QUERY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/top-k/RESERVE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/bloom/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/CONFIG_GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/CONFIG_SET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/DELETE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/EXPLAIN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/PROFILE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/QUERY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/RO_QUERY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/SLOWLOG.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/graph.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/graph/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRAPPEND.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRINDEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRINSERT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRPOP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/ARRTRIM.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/DEBUG_MEMORY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/DEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/FORGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/MERGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/MGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/MSET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/NUMINCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/NUMMULTBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/OBJKEYS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/OBJLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/RESP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/SET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/STRAPPEND.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/STRLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/TYPE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/json/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/_LIST.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/ALTER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/AGGREGATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/AGGREGATE_WITHCURSOR.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/ALIASADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/ALIASDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/ALIASUPDATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/CONFIG_GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/CONFIG_SET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/CREATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/CURSOR_DEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/CURSOR_READ.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/DICTADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/DICTDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/DICTDUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/DROPINDEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/EXPLAIN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/EXPLAINCLI.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SEARCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/PROFILE_SEARCH.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/PROFILE_AGGREGATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SEARCH_NOCONTENT.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SPELLCHECK.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGDEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGGET_WITHPAYLOADS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGGET_WITHSCORES.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGGET_WITHSCORES_WITHPAYLOADS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SUGLEN.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SYNDUMP.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/SYNUPDATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/TAGVALS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/search/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/ADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/ALTER.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/CREATE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/CREATERULE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/DECRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/DEL.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/DELETERULE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/GET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/INCRBY.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/INFO.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/INFO_DEBUG.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MADD.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MGET.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MGET_WITHLABELS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/QUERYINDEX.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/RANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/REVRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MRANGE_WITHLABELS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MREVRANGE.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/MREVRANGE_WITHLABELS.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/commands/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/@redis/time-series/dist/index.d.ts","./node_modules/cache-manager-redis-store/node_modules/redis/dist/index.d.ts","./node_modules/cache-manager-redis-store/dist/index.d.ts","./node_modules/@nestjs/schedule/dist/enums/cron-expression.enum.d.ts","./node_modules/@nestjs/schedule/dist/enums/index.d.ts","./node_modules/@types/luxon/src/zone.d.ts","./node_modules/@types/luxon/src/settings.d.ts","./node_modules/@types/luxon/src/_util.d.ts","./node_modules/@types/luxon/src/misc.d.ts","./node_modules/@types/luxon/src/duration.d.ts","./node_modules/@types/luxon/src/interval.d.ts","./node_modules/@types/luxon/src/datetime.d.ts","./node_modules/@types/luxon/src/info.d.ts","./node_modules/@types/luxon/src/luxon.d.ts","./node_modules/@types/luxon/index.d.ts","./node_modules/cron/dist/errors.d.ts","./node_modules/cron/dist/constants.d.ts","./node_modules/cron/dist/job.d.ts","./node_modules/cron/dist/types/utils.d.ts","./node_modules/cron/dist/types/cron.types.d.ts","./node_modules/cron/dist/time.d.ts","./node_modules/cron/dist/index.d.ts","./node_modules/@nestjs/schedule/dist/decorators/cron.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/interval.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/timeout.decorator.d.ts","./node_modules/@nestjs/schedule/dist/decorators/index.d.ts","./node_modules/@nestjs/schedule/dist/interfaces/schedule-module-options.interface.d.ts","./node_modules/@nestjs/schedule/dist/schedule.module.d.ts","./node_modules/@nestjs/schedule/dist/scheduler.registry.d.ts","./node_modules/@nestjs/schedule/dist/index.d.ts","./node_modules/@nestjs/schedule/index.d.ts","./node_modules/prom-client/index.d.ts","./src/monitoring/metrics/metrics-collection.service.ts","./node_modules/@types/send/index.d.ts","./node_modules/@types/qs/index.d.ts","./node_modules/@types/range-parser/index.d.ts","./node_modules/@types/express-serve-static-core/index.d.ts","./node_modules/@types/http-errors/index.d.ts","./node_modules/@types/serve-static/index.d.ts","./node_modules/@types/connect/index.d.ts","./node_modules/@types/body-parser/index.d.ts","./node_modules/@types/express/node_modules/@types/express-serve-static-core/index.d.ts","./node_modules/@types/express/index.d.ts","./src/monitoring/alerting/alerting.service.ts","./src/monitoring/scheduled-task-monitoring.service.ts","./src/monitoring/monitoring.controller.ts","./src/monitoring/performance/performance-analysis.service.ts","./src/monitoring/optimization/optimization.service.ts","./src/monitoring/monitoring.service.ts","./src/monitoring/monitoring.module.ts","./src/common/interceptors/monitoring.interceptor.ts","./src/monitoring/logging/typeorm-logger.ts","./src/common/database/transaction.service.ts","./src/common/database/transactional.decorator.ts","./src/common/database/transactional.interceptor.ts","./src/common/database/database.module.ts","./node_modules/@nestjs/bull-shared/dist/bull.messages.d.ts","./node_modules/@nestjs/bull-shared/dist/bull.tokens.d.ts","./node_modules/@nestjs/bull-shared/dist/errors/missing-shared-bull-config.error.d.ts","./node_modules/@nestjs/bull-shared/dist/errors/index.d.ts","./node_modules/@nestjs/bull-shared/dist/helpers/create-conditional-dep-holder.helper.d.ts","./node_modules/@nestjs/bull-shared/dist/helpers/index.d.ts","./node_modules/@nestjs/bull-shared/dist/utils/get-queue-token.util.d.ts","./node_modules/@nestjs/bull-shared/dist/utils/index.d.ts","./node_modules/@nestjs/bull-shared/dist/index.d.ts","./node_modules/ioredis/built/types.d.ts","./node_modules/ioredis/built/Command.d.ts","./node_modules/ioredis/built/ScanStream.d.ts","./node_modules/ioredis/built/utils/RedisCommander.d.ts","./node_modules/ioredis/built/transaction.d.ts","./node_modules/ioredis/built/utils/Commander.d.ts","./node_modules/ioredis/built/connectors/AbstractConnector.d.ts","./node_modules/ioredis/built/connectors/ConnectorConstructor.d.ts","./node_modules/ioredis/built/connectors/SentinelConnector/types.d.ts","./node_modules/ioredis/built/connectors/SentinelConnector/SentinelIterator.d.ts","./node_modules/ioredis/built/connectors/SentinelConnector/index.d.ts","./node_modules/ioredis/built/connectors/StandaloneConnector.d.ts","./node_modules/ioredis/built/redis/RedisOptions.d.ts","./node_modules/ioredis/built/cluster/util.d.ts","./node_modules/ioredis/built/cluster/ClusterOptions.d.ts","./node_modules/ioredis/built/cluster/index.d.ts","./node_modules/denque/index.d.ts","./node_modules/ioredis/built/SubscriptionSet.d.ts","./node_modules/ioredis/built/DataHandler.d.ts","./node_modules/ioredis/built/Redis.d.ts","./node_modules/ioredis/built/Pipeline.d.ts","./node_modules/ioredis/built/index.d.ts","./node_modules/bull/index.d.ts","./node_modules/@nestjs/bull/dist/interfaces/bull.interfaces.d.ts","./node_modules/@nestjs/bull/dist/bull.types.d.ts","./node_modules/@nestjs/bull/dist/interfaces/bull-module-options.interface.d.ts","./node_modules/@nestjs/bull/dist/interfaces/shared-bull-config.interface.d.ts","./node_modules/@nestjs/bull/dist/interfaces/index.d.ts","./node_modules/@nestjs/bull/dist/bull.module.d.ts","./node_modules/@nestjs/bull/dist/decorators/inject-queue.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/process.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/processor.decorator.d.ts","./node_modules/@nestjs/bull/dist/decorators/queue-hooks.decorators.d.ts","./node_modules/@nestjs/bull/dist/decorators/index.d.ts","./node_modules/@nestjs/bull/dist/enums/bull-queue-events.enum.d.ts","./node_modules/@nestjs/bull/dist/enums/bull-queue-global-events.enum.d.ts","./node_modules/@nestjs/bull/dist/enums/index.d.ts","./node_modules/@nestjs/bull/dist/utils/get-queue-options-token.util.d.ts","./node_modules/@nestjs/bull/dist/utils/get-shared-config-token.util.d.ts","./node_modules/@nestjs/bull/dist/utils/index.d.ts","./node_modules/@nestjs/bull/dist/index.d.ts","./node_modules/eventemitter2/eventemitter2.d.ts","./node_modules/@nestjs/event-emitter/dist/constants.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/event-emitter-options.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/on-event-options.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/event-payload-host.interface.d.ts","./node_modules/@nestjs/event-emitter/dist/interfaces/index.d.ts","./node_modules/@nestjs/event-emitter/dist/decorators/on-event.decorator.d.ts","./node_modules/@nestjs/event-emitter/dist/decorators/index.d.ts","./node_modules/@nestjs/event-emitter/dist/event-emitter-readiness.watcher.d.ts","./node_modules/@nestjs/event-emitter/dist/event-emitter.module.d.ts","./node_modules/@nestjs/event-emitter/dist/index.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.constants.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/cache-manager.interface.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/cache-module.interface.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.module-definition.d.ts","./node_modules/@nestjs/cache-manager/dist/cache.module.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/cache-key.decorator.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/cache-ttl.decorator.d.ts","./node_modules/@nestjs/cache-manager/dist/decorators/index.d.ts","./node_modules/@nestjs/cache-manager/dist/interceptors/cache.interceptor.d.ts","./node_modules/@nestjs/cache-manager/dist/interceptors/index.d.ts","./node_modules/@nestjs/cache-manager/dist/interfaces/index.d.ts","./node_modules/@nestjs/cache-manager/dist/index.d.ts","./node_modules/@nestjs/cache-manager/index.d.ts","./node_modules/joi/lib/index.d.ts","./src/config/env.validation.ts","./node_modules/@nestjs/passport/dist/abstract.strategy.d.ts","./node_modules/@nestjs/passport/dist/interfaces/auth-module.options.d.ts","./node_modules/@nestjs/passport/dist/interfaces/type.interface.d.ts","./node_modules/@nestjs/passport/dist/interfaces/index.d.ts","./node_modules/@nestjs/passport/dist/auth.guard.d.ts","./node_modules/@nestjs/passport/dist/passport.module.d.ts","./node_modules/@types/passport/index.d.ts","./node_modules/@nestjs/passport/dist/passport/passport.serializer.d.ts","./node_modules/@nestjs/passport/dist/passport/passport.strategy.d.ts","./node_modules/@nestjs/passport/dist/index.d.ts","./node_modules/@nestjs/passport/index.d.ts","./src/common/constants/auth.constants.ts","./src/auth/guards/jwt-auth.guard.ts","./node_modules/@nestjs/throttler/dist/throttler-storage-record.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler-storage.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.guard.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler-module-options.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.decorator.d.ts","./node_modules/@nestjs/throttler/dist/throttler.exception.d.ts","./node_modules/@nestjs/throttler/dist/throttler.guard.d.ts","./node_modules/@nestjs/throttler/dist/throttler.module.d.ts","./node_modules/@nestjs/throttler/dist/throttler.providers.d.ts","./node_modules/@nestjs/throttler/dist/throttler-storage-options.interface.d.ts","./node_modules/@nestjs/throttler/dist/throttler.service.d.ts","./node_modules/@nestjs/throttler/dist/utilities.d.ts","./node_modules/@nestjs/throttler/dist/index.d.ts","./node_modules/axios/index.d.cts","./src/health/health.service.ts","./src/common/services/shutdown-state.service.ts","./src/health/health.controller.ts","./src/health/health.module.ts","./src/session/session.constants.ts","./src/session/session.service.ts","./src/session/session.module.ts","./src/common/utils/bull-redis.util.ts","./src/common/guards/throttle.guard.ts","./src/config/feature-flags.config.ts","./src/common/lazy-loading/startup-logger.service.ts","./src/common/timeout/timeout-config.service.ts","./src/common/interceptors/timeout.interceptor.ts","./src/common/interceptors/api-version.interceptor.ts","./src/common/modules/api-versioning.module.ts","./src/common/constants/queue.constants.ts","./src/common/constants/event.constants.ts","./src/sync/conflicts/conflict-resolution.service.ts","./src/sync/consistency/data-consistency.service.ts","./src/sync/cache/cache-invalidation.service.ts","./src/sync/replication/replication.service.ts","./src/sync/sync.service.ts","./src/sync/sync.module.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-body-parser-options.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-body-parser.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/serve-static-options.interface.d.ts","./node_modules/@nestjs/platform-express/adapters/express-adapter.d.ts","./node_modules/@nestjs/platform-express/adapters/index.d.ts","./node_modules/@nestjs/platform-express/interfaces/nest-express-application.interface.d.ts","./node_modules/@nestjs/platform-express/interfaces/index.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/multer-options.interface.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/any-files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/file-fields.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/file.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/no-files.interceptor.d.ts","./node_modules/@nestjs/platform-express/multer/interceptors/index.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/files-upload-module.interface.d.ts","./node_modules/@nestjs/platform-express/multer/interfaces/index.d.ts","./node_modules/@nestjs/platform-express/multer/multer.module.d.ts","./node_modules/@nestjs/platform-express/multer/index.d.ts","./node_modules/@nestjs/platform-express/index.d.ts","./src/cdn/entities/content-metadata.entity.ts","./node_modules/@smithy/types/dist-types/abort-handler.d.ts","./node_modules/@smithy/types/dist-types/abort.d.ts","./node_modules/@smithy/types/dist-types/auth/auth.d.ts","./node_modules/@smithy/types/dist-types/auth/HttpApiKeyAuth.d.ts","./node_modules/@smithy/types/dist-types/identity/identity.d.ts","./node_modules/@smithy/types/dist-types/response.d.ts","./node_modules/@smithy/types/dist-types/command.d.ts","./node_modules/@smithy/types/dist-types/endpoint.d.ts","./node_modules/@smithy/types/dist-types/feature-ids.d.ts","./node_modules/@smithy/types/dist-types/logger.d.ts","./node_modules/@smithy/types/dist-types/uri.d.ts","./node_modules/@smithy/types/dist-types/http.d.ts","./node_modules/@smithy/types/dist-types/util.d.ts","./node_modules/@smithy/types/dist-types/middleware.d.ts","./node_modules/@smithy/types/dist-types/auth/HttpSigner.d.ts","./node_modules/@smithy/types/dist-types/auth/IdentityProviderConfig.d.ts","./node_modules/@smithy/types/dist-types/auth/HttpAuthScheme.d.ts","./node_modules/@smithy/types/dist-types/auth/HttpAuthSchemeProvider.d.ts","./node_modules/@smithy/types/dist-types/auth/index.d.ts","./node_modules/@smithy/types/dist-types/transform/exact.d.ts","./node_modules/@smithy/types/dist-types/externals-check/browser-externals-check.d.ts","./node_modules/@smithy/types/dist-types/blob/blob-payload-input-types.d.ts","./node_modules/@smithy/types/dist-types/crypto.d.ts","./node_modules/@smithy/types/dist-types/checksum.d.ts","./node_modules/@smithy/types/dist-types/client.d.ts","./node_modules/@smithy/types/dist-types/connection/config.d.ts","./node_modules/@smithy/types/dist-types/transfer.d.ts","./node_modules/@smithy/types/dist-types/connection/manager.d.ts","./node_modules/@smithy/types/dist-types/connection/pool.d.ts","./node_modules/@smithy/types/dist-types/connection/index.d.ts","./node_modules/@smithy/types/dist-types/eventStream.d.ts","./node_modules/@smithy/types/dist-types/encode.d.ts","./node_modules/@smithy/types/dist-types/endpoints/shared.d.ts","./node_modules/@smithy/types/dist-types/endpoints/EndpointRuleObject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/ErrorRuleObject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/TreeRuleObject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/RuleSetObject.d.ts","./node_modules/@smithy/types/dist-types/endpoints/index.d.ts","./node_modules/@smithy/types/dist-types/extensions/checksum.d.ts","./node_modules/@smithy/types/dist-types/extensions/defaultClientConfiguration.d.ts","./node_modules/@smithy/types/dist-types/shapes.d.ts","./node_modules/@smithy/types/dist-types/retry.d.ts","./node_modules/@smithy/types/dist-types/extensions/retry.d.ts","./node_modules/@smithy/types/dist-types/extensions/defaultExtensionConfiguration.d.ts","./node_modules/@smithy/types/dist-types/extensions/index.d.ts","./node_modules/@smithy/types/dist-types/http/httpHandlerInitialization.d.ts","./node_modules/@smithy/types/dist-types/identity/apiKeyIdentity.d.ts","./node_modules/@smithy/types/dist-types/identity/awsCredentialIdentity.d.ts","./node_modules/@smithy/types/dist-types/identity/tokenIdentity.d.ts","./node_modules/@smithy/types/dist-types/identity/index.d.ts","./node_modules/@smithy/types/dist-types/pagination.d.ts","./node_modules/@smithy/types/dist-types/profile.d.ts","./node_modules/@smithy/types/dist-types/serde.d.ts","./node_modules/@smithy/types/dist-types/schema/sentinels.d.ts","./node_modules/@smithy/types/dist-types/schema/static-schemas.d.ts","./node_modules/@smithy/types/dist-types/schema/traits.d.ts","./node_modules/@smithy/types/dist-types/schema/schema.d.ts","./node_modules/@smithy/types/dist-types/schema/schema-deprecated.d.ts","./node_modules/@smithy/types/dist-types/signature.d.ts","./node_modules/@smithy/types/dist-types/stream.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-common-types.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-payload-input-types.d.ts","./node_modules/@smithy/types/dist-types/streaming-payload/streaming-blob-payload-output-types.d.ts","./node_modules/@smithy/types/dist-types/transform/type-transform.d.ts","./node_modules/@smithy/types/dist-types/transform/client-method-transforms.d.ts","./node_modules/@smithy/types/dist-types/transform/client-payload-blob-type-narrow.d.ts","./node_modules/@smithy/types/dist-types/transform/mutable.d.ts","./node_modules/@smithy/types/dist-types/transform/no-undefined.d.ts","./node_modules/@smithy/types/dist-types/waiter.d.ts","./node_modules/@smithy/types/dist-types/index.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromEnv.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getHomeDir.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getProfileName.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getSSOTokenFilepath.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/getSSOTokenFromFile.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/constants.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/loadSharedConfigFiles.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/loadSsoSessionData.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/parseKnownFiles.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/externalDataInterceptor.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/types.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/readFile.d.ts","./node_modules/@smithy/shared-ini-file-loader/dist-types/index.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromSharedConfigFiles.d.ts","./node_modules/@smithy/node-config-provider/dist-types/fromStatic.d.ts","./node_modules/@smithy/node-config-provider/dist-types/configLoader.d.ts","./node_modules/@smithy/node-config-provider/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/constants.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/NODE_REQUEST_CHECKSUM_CALCULATION_CONFIG_OPTIONS.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/NODE_RESPONSE_CHECKSUM_VALIDATION_CONFIG_OPTIONS.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/resolveFlexibleChecksumsConfig.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/configuration.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexibleChecksumsMiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexibleChecksumsInputMiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/flexibleChecksumsResponseMiddleware.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/getFlexibleChecksumsPlugin.d.ts","./node_modules/@aws-sdk/middleware-flexible-checksums/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-host-header/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/check-content-length-header.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/region-redirect-middleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/region-redirect-endpoint-middleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-expires-middleware.d.ts","./node_modules/@aws-sdk/types/dist-types/abort.d.ts","./node_modules/@aws-sdk/types/dist-types/auth.d.ts","./node_modules/@aws-sdk/types/dist-types/blob/blob-types.d.ts","./node_modules/@aws-sdk/types/dist-types/checksum.d.ts","./node_modules/@aws-sdk/types/dist-types/client.d.ts","./node_modules/@aws-sdk/types/dist-types/command.d.ts","./node_modules/@aws-sdk/types/dist-types/connection.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/Identity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/AnonymousIdentity.d.ts","./node_modules/@aws-sdk/types/dist-types/feature-ids.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/AwsCredentialIdentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/LoginIdentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/TokenIdentity.d.ts","./node_modules/@aws-sdk/types/dist-types/identity/index.d.ts","./node_modules/@aws-sdk/types/dist-types/util.d.ts","./node_modules/@aws-sdk/types/dist-types/credentials.d.ts","./node_modules/@aws-sdk/types/dist-types/crypto.d.ts","./node_modules/@aws-sdk/types/dist-types/dns.d.ts","./node_modules/@aws-sdk/types/dist-types/encode.d.ts","./node_modules/@aws-sdk/types/dist-types/endpoint.d.ts","./node_modules/@aws-sdk/types/dist-types/eventStream.d.ts","./node_modules/@aws-sdk/types/dist-types/extensions/index.d.ts","./node_modules/@aws-sdk/types/dist-types/function.d.ts","./node_modules/@aws-sdk/types/dist-types/http.d.ts","./node_modules/@aws-sdk/types/dist-types/logger.d.ts","./node_modules/@aws-sdk/types/dist-types/middleware.d.ts","./node_modules/@aws-sdk/types/dist-types/pagination.d.ts","./node_modules/@aws-sdk/types/dist-types/profile.d.ts","./node_modules/@aws-sdk/types/dist-types/request.d.ts","./node_modules/@aws-sdk/types/dist-types/response.d.ts","./node_modules/@aws-sdk/types/dist-types/retry.d.ts","./node_modules/@aws-sdk/types/dist-types/serde.d.ts","./node_modules/@aws-sdk/types/dist-types/shapes.d.ts","./node_modules/@aws-sdk/types/dist-types/signature.d.ts","./node_modules/@aws-sdk/types/dist-types/stream.d.ts","./node_modules/@aws-sdk/types/dist-types/token.d.ts","./node_modules/@aws-sdk/types/dist-types/transfer.d.ts","./node_modules/@aws-sdk/types/dist-types/uri.d.ts","./node_modules/@aws-sdk/types/dist-types/waiter.d.ts","./node_modules/@aws-sdk/types/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/interfaces/S3ExpressIdentity.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/S3ExpressIdentityCacheEntry.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/S3ExpressIdentityCache.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/interfaces/S3ExpressIdentityProvider.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/S3ExpressIdentityProviderImpl.d.ts","./node_modules/@smithy/signature-v4/dist-types/SignatureV4Base.d.ts","./node_modules/@smithy/signature-v4/dist-types/SignatureV4.d.ts","./node_modules/@smithy/signature-v4/dist-types/constants.d.ts","./node_modules/@smithy/signature-v4/dist-types/getCanonicalHeaders.d.ts","./node_modules/@smithy/signature-v4/dist-types/getCanonicalQuery.d.ts","./node_modules/@smithy/signature-v4/dist-types/getPayloadHash.d.ts","./node_modules/@smithy/signature-v4/dist-types/moveHeadersToQuery.d.ts","./node_modules/@smithy/signature-v4/dist-types/prepareRequest.d.ts","./node_modules/@smithy/signature-v4/dist-types/credentialDerivation.d.ts","./node_modules/@smithy/signature-v4/dist-types/headerUtil.d.ts","./node_modules/@smithy/signature-v4/dist-types/signature-v4a-container.d.ts","./node_modules/@smithy/signature-v4/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/classes/SignatureV4S3Express.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/constants.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/functions/s3ExpressMiddleware.d.ts","./node_modules/@smithy/protocol-http/dist-types/httpRequest.d.ts","./node_modules/@smithy/protocol-http/dist-types/httpResponse.d.ts","./node_modules/@smithy/protocol-http/dist-types/httpHandler.d.ts","./node_modules/@smithy/protocol-http/dist-types/extensions/httpExtensionConfiguration.d.ts","./node_modules/@smithy/protocol-http/dist-types/extensions/index.d.ts","./node_modules/@smithy/protocol-http/dist-types/Field.d.ts","./node_modules/@smithy/protocol-http/dist-types/Fields.d.ts","./node_modules/@smithy/protocol-http/dist-types/isValidHostname.d.ts","./node_modules/@smithy/protocol-http/dist-types/types.d.ts","./node_modules/@smithy/protocol-http/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/functions/s3ExpressHttpSigningMiddleware.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3-express/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/s3Configuration.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/throw-200-exceptions.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/validate-bucket-name.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/cbor.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/cbor-types.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/parseCborBody.d.ts","./node_modules/@smithy/util-stream/dist-types/blob/Uint8ArrayBlobAdapter.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/ChecksumStream.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/ChecksumStream.browser.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/createChecksumStream.browser.d.ts","./node_modules/@smithy/util-stream/dist-types/checksum/createChecksumStream.d.ts","./node_modules/@smithy/util-stream/dist-types/createBufferedReadable.d.ts","./node_modules/@smithy/util-stream/dist-types/getAwsChunkedEncodingStream.d.ts","./node_modules/@smithy/util-stream/dist-types/headStream.d.ts","./node_modules/@smithy/util-stream/dist-types/sdk-stream-mixin.d.ts","./node_modules/@smithy/util-stream/dist-types/splitStream.d.ts","./node_modules/@smithy/util-stream/dist-types/stream-type-check.d.ts","./node_modules/@smithy/util-stream/dist-types/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/collect-stream-body.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/extended-encode-uri-component.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/deref.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/middleware/schema-middleware-types.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/middleware/getSchemaSerdePlugin.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/Schema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/ListSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/MapSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/OperationSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/operation.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/StructureSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/ErrorSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/NormalizedSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/SimpleSchema.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/sentinels.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/schemas/translateTraits.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/TypeRegistry.d.ts","./node_modules/@smithy/core/dist-types/submodules/schema/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/event-streams/EventStreamSerde.d.ts","./node_modules/@smithy/core/dist-types/submodules/event-streams/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/SerdeContext.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/HttpProtocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/HttpBindingProtocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/RpcProtocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/requestBuilder.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/resolve-path.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/FromStringShapeDeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/HttpInterceptingShapeDeserializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/ToStringShapeSerializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/HttpInterceptingShapeSerializer.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/serde/determineTimestampFormat.d.ts","./node_modules/@smithy/core/dist-types/submodules/protocols/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/CborCodec.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/SmithyRpcV2CborProtocol.d.ts","./node_modules/@smithy/core/dist-types/submodules/cbor/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/cbor/AwsSmithyRpcV2CborProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/coercing-serializers.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/ConfigurableSerdeContext.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/JsonShapeDeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/JsonShapeSerializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/JsonCodec.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/AwsJsonRpcProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/AwsJson1_0Protocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/AwsJson1_1Protocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/AwsRestJsonProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/awsExpectUnion.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/json/parseJsonBody.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/XmlShapeSerializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/XmlCodec.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/XmlShapeDeserializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/QuerySerializerSettings.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/QueryShapeSerializer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/AwsQueryProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/query/AwsEc2QueryProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/AwsRestXmlProtocol.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/xml/parseXmlBody.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/protocols/index.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/protocol/S3RestXmlProtocol.d.ts","./node_modules/@aws-sdk/middleware-sdk-s3/dist-types/index.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/configurations.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/user-agent-middleware.d.ts","./node_modules/@aws-sdk/middleware-user-agent/dist-types/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsConfig/NodeUseDualstackEndpointConfigOptions.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsConfig/NodeUseFipsEndpointConfigOptions.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsConfig/resolveEndpointsConfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsConfig/resolveCustomEndpointsConfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/endpointsConfig/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionConfig/config.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionConfig/resolveRegionConfig.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionConfig/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/EndpointVariantTag.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/EndpointVariant.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/PartitionHash.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/RegionHash.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/getRegionInfo.d.ts","./node_modules/@smithy/config-resolver/dist-types/regionInfo/index.d.ts","./node_modules/@smithy/config-resolver/dist-types/index.d.ts","./node_modules/@smithy/eventstream-serde-config-resolver/dist-types/EventStreamSerdeConfig.d.ts","./node_modules/@smithy/eventstream-serde-config-resolver/dist-types/index.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/resolveEndpointConfig.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/types.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/getEndpointFromInstructions.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/toEndpointV1.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/adaptors/index.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/endpointMiddleware.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/getEndpointPlugin.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/resolveEndpointRequiredConfig.d.ts","./node_modules/@smithy/middleware-endpoint/dist-types/index.d.ts","./node_modules/@smithy/util-retry/dist-types/StandardRetryStrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/types.d.ts","./node_modules/@smithy/util-retry/dist-types/AdaptiveRetryStrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/ConfiguredRetryStrategy.d.ts","./node_modules/@smithy/util-retry/dist-types/DefaultRateLimiter.d.ts","./node_modules/@smithy/util-retry/dist-types/config.d.ts","./node_modules/@smithy/util-retry/dist-types/constants.d.ts","./node_modules/@smithy/util-retry/dist-types/retries-2026-config.d.ts","./node_modules/@smithy/util-retry/dist-types/index.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/types.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/StandardRetryStrategy.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/AdaptiveRetryStrategy.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/delayDecider.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retry-pre-sra-deprecated/retryDecider.d.ts","./node_modules/@smithy/middleware-retry/dist-types/configurations.d.ts","./node_modules/@smithy/middleware-retry/dist-types/omitRetryHeadersMiddleware.d.ts","./node_modules/@smithy/middleware-retry/dist-types/retryMiddleware.d.ts","./node_modules/@smithy/middleware-retry/dist-types/parseRetryAfterHeader.d.ts","./node_modules/@smithy/middleware-retry/dist-types/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/client.d.ts","./node_modules/@smithy/smithy-client/dist-types/collect-stream-body.d.ts","./node_modules/@smithy/smithy-client/dist-types/command.d.ts","./node_modules/@smithy/smithy-client/dist-types/constants.d.ts","./node_modules/@smithy/smithy-client/dist-types/create-aggregated-client.d.ts","./node_modules/@smithy/smithy-client/dist-types/default-error-handler.d.ts","./node_modules/@smithy/smithy-client/dist-types/defaults-mode.d.ts","./node_modules/@smithy/smithy-client/dist-types/emitWarningIfUnsupportedVersion.d.ts","./node_modules/@smithy/smithy-client/dist-types/exceptions.d.ts","./node_modules/@smithy/smithy-client/dist-types/extended-encode-uri-component.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/checksum.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/retry.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/defaultExtensionConfiguration.d.ts","./node_modules/@smithy/smithy-client/dist-types/extensions/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/get-array-if-single-item.d.ts","./node_modules/@smithy/smithy-client/dist-types/get-value-from-text-node.d.ts","./node_modules/@smithy/smithy-client/dist-types/is-serializable-header-value.d.ts","./node_modules/@smithy/smithy-client/dist-types/NoOpLogger.d.ts","./node_modules/@smithy/smithy-client/dist-types/object-mapping.d.ts","./node_modules/@smithy/smithy-client/dist-types/resolve-path.d.ts","./node_modules/@smithy/smithy-client/dist-types/ser-utils.d.ts","./node_modules/@smithy/smithy-client/dist-types/serde-json.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/copyDocumentWithTransform.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/date-utils.d.ts","./node_modules/@smithy/uuid/dist-types/v4.d.ts","./node_modules/@smithy/uuid/dist-types/index.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/generateIdempotencyToken.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/lazy-json.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/parse-utils.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/quote-header.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/schema-serde-lib/schema-date-utils.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/split-every.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/split-header.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/value/NumericValue.d.ts","./node_modules/@smithy/core/dist-types/submodules/serde/index.d.ts","./node_modules/@smithy/smithy-client/dist-types/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/resolveAwsSdkSigV4AConfig.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/AwsSdkSigV4Signer.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/AwsSdkSigV4ASigner.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/NODE_AUTH_SCHEME_PREFERENCE_OPTIONS.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/resolveAwsSdkSigV4Config.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/aws_sdk/index.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/utils/getBearerTokenEnvKey.d.ts","./node_modules/@aws-sdk/core/dist-types/submodules/httpAuthSchemes/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/endpoint/EndpointParameters.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/auth/httpAuthSchemeProvider.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/AbortMultipartUploadCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CompleteMultipartUploadCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CopyObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CreateBucketCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CreateBucketMetadataConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CreateBucketMetadataTableConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CreateMultipartUploadCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/CreateSessionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketAnalyticsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketCorsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketEncryptionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketIntelligentTieringConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketInventoryConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketLifecycleCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketMetadataConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketMetadataTableConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketMetricsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketOwnershipControlsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketPolicyCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketReplicationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteBucketWebsiteCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteObjectsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeleteObjectTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/DeletePublicAccessBlockCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketAbacCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketAccelerateConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketAclCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketAnalyticsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketCorsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketEncryptionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketIntelligentTieringConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketInventoryConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketLifecycleConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketLocationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketLoggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketMetadataConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketMetadataTableConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketMetricsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketNotificationConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketOwnershipControlsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketPolicyCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketPolicyStatusCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketReplicationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketRequestPaymentCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketVersioningCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetBucketWebsiteCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectAclCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectAttributesCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectLegalHoldCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectLockConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectRetentionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetObjectTorrentCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/GetPublicAccessBlockCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/HeadBucketCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/HeadObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListBucketAnalyticsConfigurationsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListBucketIntelligentTieringConfigurationsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListBucketInventoryConfigurationsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListBucketMetricsConfigurationsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListBucketsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListDirectoryBucketsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListMultipartUploadsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListObjectsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListObjectsV2Command.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListObjectVersionsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/ListPartsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketAbacCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketAccelerateConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketAclCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketAnalyticsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketCorsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketEncryptionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketIntelligentTieringConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketInventoryConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketLifecycleConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketLoggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketMetricsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketNotificationConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketOwnershipControlsCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketPolicyCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketReplicationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketRequestPaymentCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketVersioningCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutBucketWebsiteCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectAclCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectLegalHoldCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectLockConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectRetentionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutObjectTaggingCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/PutPublicAccessBlockCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/RenameObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/models_1.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/RestoreObjectCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/SelectObjectContentCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/UpdateBucketMetadataInventoryTableConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/UpdateBucketMetadataJournalTableConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/UpdateObjectEncryptionCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/UploadPartCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/UploadPartCopyCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/WriteGetObjectResponseCommand.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/auth/httpAuthExtensionConfiguration.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/extensionConfiguration.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/runtimeExtensions.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/S3Client.d.ts","./node_modules/@smithy/util-waiter/dist-types/waiter.d.ts","./node_modules/@smithy/util-waiter/dist-types/createWaiter.d.ts","./node_modules/@smithy/util-waiter/dist-types/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/S3.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/Interfaces.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/ListBucketsPaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/ListDirectoryBucketsPaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/ListObjectsV2Paginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/ListPartsPaginator.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitForBucketExists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitForBucketNotExists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitForObjectExists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/waitForObjectNotExists.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/waiters/index.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/S3ServiceException.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-s3/dist-types/index.d.ts","./src/common/types/file.types.ts","./src/media/storage/file-storage.service.ts","./src/media/processing/video-processing.service.ts","./node_modules/sharp/lib/index.d.ts","./src/media/validation/file-validation.constants.ts","./src/media/validation/file-validation.service.ts","./src/media/validation/malware-scanning.service.ts","./src/media/processing/image-processing.service.ts","./src/media/validation/upload-progress.service.ts","./node_modules/uuid/dist/max.d.ts","./node_modules/uuid/dist/nil.d.ts","./node_modules/uuid/dist/types.d.ts","./node_modules/uuid/dist/parse.d.ts","./node_modules/uuid/dist/stringify.d.ts","./node_modules/uuid/dist/v1.d.ts","./node_modules/uuid/dist/v1ToV6.d.ts","./node_modules/uuid/dist/v35.d.ts","./node_modules/uuid/dist/v3.d.ts","./node_modules/uuid/dist/v4.d.ts","./node_modules/uuid/dist/v5.d.ts","./node_modules/uuid/dist/v6.d.ts","./node_modules/uuid/dist/v6ToV1.d.ts","./node_modules/uuid/dist/v7.d.ts","./node_modules/uuid/dist/validate.d.ts","./node_modules/uuid/dist/version.d.ts","./node_modules/uuid/dist/index.d.ts","./src/media/media.service.ts","./src/media/validation/upload-validation.util.ts","./src/media/media.controller.ts","./src/media/processing/document-processing.service.ts","./node_modules/@types/fluent-ffmpeg/index.d.ts","./src/media/processing/video.processor.ts","./src/media/media.module.ts","./src/backup/enums/backup-status.enum.ts","./src/backup/enums/backup-type.enum.ts","./src/backup/enums/region.enum.ts","./src/backup/entities/backup-record.entity.ts","./src/backup/enums/recovery-test-status.enum.ts","./src/backup/entities/recovery-test.entity.ts","./node_modules/class-validator/types/validation/ValidationError.d.ts","./node_modules/class-validator/types/validation/ValidatorOptions.d.ts","./node_modules/class-validator/types/validation-schema/ValidationSchema.d.ts","./node_modules/class-validator/types/container.d.ts","./node_modules/class-validator/types/validation/ValidationArguments.d.ts","./node_modules/class-validator/types/decorator/ValidationOptions.d.ts","./node_modules/class-validator/types/decorator/common/Allow.d.ts","./node_modules/class-validator/types/decorator/common/IsDefined.d.ts","./node_modules/class-validator/types/decorator/common/IsOptional.d.ts","./node_modules/class-validator/types/decorator/common/Validate.d.ts","./node_modules/class-validator/types/validation/ValidatorConstraintInterface.d.ts","./node_modules/class-validator/types/decorator/common/ValidateBy.d.ts","./node_modules/class-validator/types/decorator/common/ValidateIf.d.ts","./node_modules/class-validator/types/decorator/common/ValidateNested.d.ts","./node_modules/class-validator/types/decorator/common/ValidatePromise.d.ts","./node_modules/class-validator/types/decorator/common/IsLatLong.d.ts","./node_modules/class-validator/types/decorator/common/IsLatitude.d.ts","./node_modules/class-validator/types/decorator/common/IsLongitude.d.ts","./node_modules/class-validator/types/decorator/common/Equals.d.ts","./node_modules/class-validator/types/decorator/common/NotEquals.d.ts","./node_modules/class-validator/types/decorator/common/IsEmpty.d.ts","./node_modules/class-validator/types/decorator/common/IsNotEmpty.d.ts","./node_modules/class-validator/types/decorator/common/IsIn.d.ts","./node_modules/class-validator/types/decorator/common/IsNotIn.d.ts","./node_modules/class-validator/types/decorator/number/IsDivisibleBy.d.ts","./node_modules/class-validator/types/decorator/number/IsPositive.d.ts","./node_modules/class-validator/types/decorator/number/IsNegative.d.ts","./node_modules/class-validator/types/decorator/number/Max.d.ts","./node_modules/class-validator/types/decorator/number/Min.d.ts","./node_modules/class-validator/types/decorator/date/MinDate.d.ts","./node_modules/class-validator/types/decorator/date/MaxDate.d.ts","./node_modules/class-validator/types/decorator/string/Contains.d.ts","./node_modules/class-validator/types/decorator/string/NotContains.d.ts","./node_modules/@types/validator/lib/isBoolean.d.ts","./node_modules/@types/validator/lib/isEmail.d.ts","./node_modules/@types/validator/lib/isFQDN.d.ts","./node_modules/@types/validator/lib/isIBAN.d.ts","./node_modules/@types/validator/lib/isISO31661Alpha2.d.ts","./node_modules/@types/validator/lib/isISO4217.d.ts","./node_modules/@types/validator/lib/isISO6391.d.ts","./node_modules/@types/validator/lib/isTaxID.d.ts","./node_modules/@types/validator/lib/isURL.d.ts","./node_modules/@types/validator/index.d.ts","./node_modules/class-validator/types/decorator/string/IsAlpha.d.ts","./node_modules/class-validator/types/decorator/string/IsAlphanumeric.d.ts","./node_modules/class-validator/types/decorator/string/IsDecimal.d.ts","./node_modules/class-validator/types/decorator/string/IsAscii.d.ts","./node_modules/class-validator/types/decorator/string/IsBase64.d.ts","./node_modules/class-validator/types/decorator/string/IsByteLength.d.ts","./node_modules/class-validator/types/decorator/string/IsCreditCard.d.ts","./node_modules/class-validator/types/decorator/string/IsCurrency.d.ts","./node_modules/class-validator/types/decorator/string/IsEmail.d.ts","./node_modules/class-validator/types/decorator/string/IsFQDN.d.ts","./node_modules/class-validator/types/decorator/string/IsFullWidth.d.ts","./node_modules/class-validator/types/decorator/string/IsHalfWidth.d.ts","./node_modules/class-validator/types/decorator/string/IsVariableWidth.d.ts","./node_modules/class-validator/types/decorator/string/IsHexColor.d.ts","./node_modules/class-validator/types/decorator/string/IsHexadecimal.d.ts","./node_modules/class-validator/types/decorator/string/IsMacAddress.d.ts","./node_modules/class-validator/types/decorator/string/IsIP.d.ts","./node_modules/class-validator/types/decorator/string/IsPort.d.ts","./node_modules/class-validator/types/decorator/string/IsISBN.d.ts","./node_modules/class-validator/types/decorator/string/IsISIN.d.ts","./node_modules/class-validator/types/decorator/string/IsISO8601.d.ts","./node_modules/class-validator/types/decorator/string/IsJSON.d.ts","./node_modules/class-validator/types/decorator/string/IsJWT.d.ts","./node_modules/class-validator/types/decorator/string/IsLowercase.d.ts","./node_modules/class-validator/types/decorator/string/IsMobilePhone.d.ts","./node_modules/class-validator/types/decorator/string/IsISO31661Alpha2.d.ts","./node_modules/class-validator/types/decorator/string/IsISO31661Alpha3.d.ts","./node_modules/class-validator/types/decorator/string/IsMongoId.d.ts","./node_modules/class-validator/types/decorator/string/IsMultibyte.d.ts","./node_modules/class-validator/types/decorator/string/IsSurrogatePair.d.ts","./node_modules/class-validator/types/decorator/string/IsUrl.d.ts","./node_modules/class-validator/types/decorator/string/IsUUID.d.ts","./node_modules/class-validator/types/decorator/string/IsFirebasePushId.d.ts","./node_modules/class-validator/types/decorator/string/IsUppercase.d.ts","./node_modules/class-validator/types/decorator/string/Length.d.ts","./node_modules/class-validator/types/decorator/string/MaxLength.d.ts","./node_modules/class-validator/types/decorator/string/MinLength.d.ts","./node_modules/class-validator/types/decorator/string/Matches.d.ts","./node_modules/libphonenumber-js/types.d.cts","./node_modules/libphonenumber-js/max/index.d.cts","./node_modules/class-validator/types/decorator/string/IsPhoneNumber.d.ts","./node_modules/class-validator/types/decorator/string/IsMilitaryTime.d.ts","./node_modules/class-validator/types/decorator/string/IsHash.d.ts","./node_modules/class-validator/types/decorator/string/IsISSN.d.ts","./node_modules/class-validator/types/decorator/string/IsDateString.d.ts","./node_modules/class-validator/types/decorator/string/IsBooleanString.d.ts","./node_modules/class-validator/types/decorator/string/IsNumberString.d.ts","./node_modules/class-validator/types/decorator/string/IsBase32.d.ts","./node_modules/class-validator/types/decorator/string/IsBIC.d.ts","./node_modules/class-validator/types/decorator/string/IsBtcAddress.d.ts","./node_modules/class-validator/types/decorator/string/IsDataURI.d.ts","./node_modules/class-validator/types/decorator/string/IsEAN.d.ts","./node_modules/class-validator/types/decorator/string/IsEthereumAddress.d.ts","./node_modules/class-validator/types/decorator/string/IsHSL.d.ts","./node_modules/class-validator/types/decorator/string/IsIBAN.d.ts","./node_modules/class-validator/types/decorator/string/IsIdentityCard.d.ts","./node_modules/class-validator/types/decorator/string/IsISRC.d.ts","./node_modules/class-validator/types/decorator/string/IsLocale.d.ts","./node_modules/class-validator/types/decorator/string/IsMagnetURI.d.ts","./node_modules/class-validator/types/decorator/string/IsMimeType.d.ts","./node_modules/class-validator/types/decorator/string/IsOctal.d.ts","./node_modules/class-validator/types/decorator/string/IsPassportNumber.d.ts","./node_modules/class-validator/types/decorator/string/IsPostalCode.d.ts","./node_modules/class-validator/types/decorator/string/IsRFC3339.d.ts","./node_modules/class-validator/types/decorator/string/IsRgbColor.d.ts","./node_modules/class-validator/types/decorator/string/IsSemVer.d.ts","./node_modules/class-validator/types/decorator/string/IsStrongPassword.d.ts","./node_modules/class-validator/types/decorator/string/IsTimeZone.d.ts","./node_modules/class-validator/types/decorator/string/IsBase58.d.ts","./node_modules/class-validator/types/decorator/string/is-tax-id.d.ts","./node_modules/class-validator/types/decorator/string/is-iso4217-currency-code.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsBoolean.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsDate.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsNumber.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsEnum.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsInt.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsString.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsArray.d.ts","./node_modules/class-validator/types/decorator/typechecker/IsObject.d.ts","./node_modules/class-validator/types/decorator/array/ArrayContains.d.ts","./node_modules/class-validator/types/decorator/array/ArrayNotContains.d.ts","./node_modules/class-validator/types/decorator/array/ArrayNotEmpty.d.ts","./node_modules/class-validator/types/decorator/array/ArrayMinSize.d.ts","./node_modules/class-validator/types/decorator/array/ArrayMaxSize.d.ts","./node_modules/class-validator/types/decorator/array/ArrayUnique.d.ts","./node_modules/class-validator/types/decorator/object/IsNotEmptyObject.d.ts","./node_modules/class-validator/types/decorator/object/IsInstance.d.ts","./node_modules/class-validator/types/decorator/decorators.d.ts","./node_modules/class-validator/types/validation/ValidationTypes.d.ts","./node_modules/class-validator/types/validation/Validator.d.ts","./node_modules/class-validator/types/register-decorator.d.ts","./node_modules/class-validator/types/metadata/ValidationMetadataArgs.d.ts","./node_modules/class-validator/types/metadata/ValidationMetadata.d.ts","./node_modules/class-validator/types/metadata/ConstraintMetadata.d.ts","./node_modules/class-validator/types/metadata/MetadataStorage.d.ts","./node_modules/class-validator/types/index.d.ts","./src/backup/dto/backup-response.dto.ts","./src/backup/interfaces/backup.interfaces.ts","./src/backup/backup.service.ts","./node_modules/@aws-sdk/client-kms/dist-types/auth/httpAuthSchemeProvider.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/CancelKeyDeletionCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ConnectCustomKeyStoreCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/CreateAliasCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/CreateCustomKeyStoreCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/CreateGrantCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/CreateKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DecryptCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DeleteAliasCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DeleteCustomKeyStoreCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DeleteImportedKeyMaterialCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DeriveSharedSecretCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DescribeCustomKeyStoresCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DescribeKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DisableKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DisableKeyRotationCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/DisconnectCustomKeyStoreCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/EnableKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/EnableKeyRotationCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/EncryptCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateDataKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateDataKeyPairCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateDataKeyPairWithoutPlaintextCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateDataKeyWithoutPlaintextCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateMacCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GenerateRandomCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GetKeyPolicyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GetKeyRotationStatusCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GetParametersForImportCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/GetPublicKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ImportKeyMaterialCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListAliasesCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListGrantsCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListKeyPoliciesCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListKeyRotationsCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListKeysCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListResourceTagsCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ListRetirableGrantsCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/PutKeyPolicyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ReEncryptCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ReplicateKeyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/RetireGrantCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/RevokeGrantCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/RotateKeyOnDemandCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/ScheduleKeyDeletionCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/SignCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/TagResourceCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/UntagResourceCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/UpdateAliasCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/UpdateCustomKeyStoreCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/UpdateKeyDescriptionCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/UpdatePrimaryRegionCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/VerifyCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/VerifyMacCommand.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/endpoint/EndpointParameters.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/auth/httpAuthExtensionConfiguration.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/extensionConfiguration.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/runtimeExtensions.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/KMSClient.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/KMS.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/Interfaces.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/DescribeCustomKeyStoresPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListAliasesPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListGrantsPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListKeyPoliciesPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListKeyRotationsPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListKeysPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListResourceTagsPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/ListRetirableGrantsPaginator.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/KMSServiceException.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-kms/dist-types/index.d.ts","./src/backup/disaster-recovery/disaster-recovery.service.ts","./src/backup/integrity/data-integrity.service.ts","./src/backup/dto/recovery-test-response.dto.ts","./src/backup/testing/recovery-testing.service.ts","./src/backup/monitoring/backup-monitoring.service.ts","./src/backup/dto/restore-backup.dto.ts","./src/backup/dto/trigger-recovery-test.dto.ts","./src/backup/backup.controller.ts","./src/backup/processing/backup-queue.processor.ts","./src/backup/backup.module.ts","./src/collaboration/documents/shared-document.service.ts","./src/collaboration/whiteboard/whiteboard.service.ts","./src/collaboration/versioning/version-control.service.ts","./src/collaboration/permissions/collaboration-permissions.service.ts","./src/collaboration/collaboration.service.ts","./node_modules/@nestjs/websockets/adapters/ws-adapter.d.ts","./node_modules/@nestjs/websockets/adapters/index.d.ts","./node_modules/@nestjs/websockets/decorators/connected-socket.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/gateway-server.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/message-body.decorator.d.ts","./node_modules/@nestjs/websockets/interfaces/gateway-metadata.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-connection.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-disconnect.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/on-gateway-init.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/hooks/index.d.ts","./node_modules/@nestjs/websockets/interfaces/server-and-event-streams-host.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/web-socket-server.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/ws-response.interface.d.ts","./node_modules/@nestjs/websockets/interfaces/index.d.ts","./node_modules/@nestjs/websockets/decorators/socket-gateway.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/subscribe-message.decorator.d.ts","./node_modules/@nestjs/websockets/decorators/index.d.ts","./node_modules/@nestjs/websockets/errors/ws-exception.d.ts","./node_modules/@nestjs/websockets/errors/index.d.ts","./node_modules/@nestjs/websockets/exceptions/base-ws-exception-filter.d.ts","./node_modules/@nestjs/websockets/exceptions/index.d.ts","./node_modules/@nestjs/websockets/interfaces/nest-gateway.interface.d.ts","./node_modules/@nestjs/websockets/gateway-metadata-explorer.d.ts","./node_modules/@nestjs/websockets/index.d.ts","./src/collaboration/constants/collaboration-events.constants.ts","./node_modules/engine.io-parser/build/cjs/commons.d.ts","./node_modules/engine.io-parser/build/cjs/encodePacket.d.ts","./node_modules/engine.io-parser/build/cjs/decodePacket.d.ts","./node_modules/engine.io-parser/build/cjs/index.d.ts","./node_modules/engine.io/build/parser-v3/index.d.ts","./node_modules/@types/ws/index.d.ts","./node_modules/engine.io/build/transport.d.ts","./node_modules/engine.io/build/socket.d.ts","./node_modules/@types/cors/index.d.ts","./node_modules/engine.io/build/contrib/types.cookie.d.ts","./node_modules/engine.io/build/server.d.ts","./node_modules/engine.io/build/transports/polling.d.ts","./node_modules/engine.io/build/transports/websocket.d.ts","./node_modules/engine.io/build/transports/webtransport.d.ts","./node_modules/engine.io/build/transports/index.d.ts","./node_modules/engine.io/build/userver.d.ts","./node_modules/engine.io/build/engine.io.d.ts","./node_modules/@socket.io/component-emitter/lib/cjs/index.d.ts","./node_modules/socket.io-parser/build/cjs/index.d.ts","./node_modules/socket.io/dist/typed-events.d.ts","./node_modules/socket.io/dist/client.d.ts","./node_modules/socket.io-adapter/dist/in-memory-adapter.d.ts","./node_modules/socket.io-adapter/dist/cluster-adapter.d.ts","./node_modules/socket.io-adapter/dist/index.d.ts","./node_modules/socket.io/dist/socket-types.d.ts","./node_modules/socket.io/dist/broadcast-operator.d.ts","./node_modules/socket.io/dist/socket.d.ts","./node_modules/socket.io/dist/namespace.d.ts","./node_modules/socket.io/dist/index.d.ts","./src/common/guards/ws-throttler.guard.ts","./src/common/utils/websocket.utils.ts","./src/collaboration/gateway/collaboration.gateway.ts","./src/courses/entities/lesson.entity.ts","./src/courses/entities/course-module.entity.ts","./src/courses/entities/enrollment.entity.ts","./src/courses/entities/course.entity.ts","./src/users/entities/user.entity.ts","./src/auth/decorators/roles.decorator.ts","./src/auth/guards/roles.guard.ts","./src/collaboration/collaboration.controller.ts","./src/collaboration/collaboration.module.ts","./src/data-warehouse/etl/etl-pipeline.service.ts","./src/data-warehouse/modeling/dimensional-modeling.service.ts","./src/data-warehouse/quality/data-quality.service.ts","./src/data-warehouse/lineage/data-lineage.service.ts","./src/data-warehouse/loading/incremental-loader.service.ts","./src/data-warehouse/data-warehouse.controller.ts","./src/data-warehouse/data-warehouse.module.ts","./src/queues/enums/job-priority.enum.ts","./src/queues/interfaces/queue.interfaces.ts","./src/queues/queue.service.ts","./src/common/decorators/roles.decorator.ts","./src/common/guards/roles.guard.ts","./src/queues/prioritization/prioritization.service.ts","./src/queues/scheduler/job-scheduler.service.ts","./src/queues/monitoring/queue-monitoring.service.ts","./node_modules/class-transformer/types/interfaces/decorator-options/expose-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/exclude-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/transform-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/type-discriminator-descriptor.interface.d.ts","./node_modules/class-transformer/types/interfaces/decorator-options/type-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/exclude-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/expose-metadata.interface.d.ts","./node_modules/class-transformer/types/enums/transformation-type.enum.d.ts","./node_modules/class-transformer/types/enums/index.d.ts","./node_modules/class-transformer/types/interfaces/target-map.interface.d.ts","./node_modules/class-transformer/types/interfaces/class-transformer-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/transform-fn-params.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/transform-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/metadata/type-metadata.interface.d.ts","./node_modules/class-transformer/types/interfaces/class-constructor.type.d.ts","./node_modules/class-transformer/types/interfaces/type-help-options.interface.d.ts","./node_modules/class-transformer/types/interfaces/index.d.ts","./node_modules/class-transformer/types/ClassTransformer.d.ts","./node_modules/class-transformer/types/decorators/exclude.decorator.d.ts","./node_modules/class-transformer/types/decorators/expose.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-instance-to-instance.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-instance-to-plain.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform-plain-to-instance.decorator.d.ts","./node_modules/class-transformer/types/decorators/transform.decorator.d.ts","./node_modules/class-transformer/types/decorators/type.decorator.d.ts","./node_modules/class-transformer/types/decorators/index.d.ts","./node_modules/class-transformer/types/index.d.ts","./src/queues/dto/queue.dto.ts","./src/queues/queue.controller.ts","./src/queues/retry/retry-logic.service.ts","./src/common/utils/pii-sanitizer.utils.ts","./src/queues/processors/default-queue.processor.ts","./src/queues/queue.module.ts","./node_modules/@nestjs/graphql/dist/decorators/args-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/base-type-options.interface.d.ts","./node_modules/graphql/version.d.ts","./node_modules/graphql/jsutils/Maybe.d.ts","./node_modules/graphql/language/source.d.ts","./node_modules/graphql/jsutils/ObjMap.d.ts","./node_modules/graphql/jsutils/Path.d.ts","./node_modules/graphql/jsutils/PromiseOrValue.d.ts","./node_modules/graphql/language/kinds.d.ts","./node_modules/graphql/language/tokenKind.d.ts","./node_modules/graphql/language/ast.d.ts","./node_modules/graphql/language/location.d.ts","./node_modules/graphql/error/GraphQLError.d.ts","./node_modules/graphql/language/directiveLocation.d.ts","./node_modules/graphql/type/directives.d.ts","./node_modules/graphql/type/schema.d.ts","./node_modules/graphql/type/definition.d.ts","./node_modules/graphql/execution/execute.d.ts","./node_modules/graphql/graphql.d.ts","./node_modules/graphql/type/scalars.d.ts","./node_modules/graphql/type/introspection.d.ts","./node_modules/graphql/type/validate.d.ts","./node_modules/graphql/type/assertName.d.ts","./node_modules/graphql/type/index.d.ts","./node_modules/graphql/language/printLocation.d.ts","./node_modules/graphql/language/lexer.d.ts","./node_modules/graphql/language/parser.d.ts","./node_modules/graphql/language/printer.d.ts","./node_modules/graphql/language/visitor.d.ts","./node_modules/graphql/language/predicates.d.ts","./node_modules/graphql/language/index.d.ts","./node_modules/graphql/execution/subscribe.d.ts","./node_modules/graphql/execution/values.d.ts","./node_modules/graphql/execution/index.d.ts","./node_modules/graphql/subscription/index.d.ts","./node_modules/graphql/utilities/TypeInfo.d.ts","./node_modules/graphql/validation/ValidationContext.d.ts","./node_modules/graphql/validation/validate.d.ts","./node_modules/graphql/validation/rules/MaxIntrospectionDepthRule.d.ts","./node_modules/graphql/validation/specifiedRules.d.ts","./node_modules/graphql/validation/rules/ExecutableDefinitionsRule.d.ts","./node_modules/graphql/validation/rules/FieldsOnCorrectTypeRule.d.ts","./node_modules/graphql/validation/rules/FragmentsOnCompositeTypesRule.d.ts","./node_modules/graphql/validation/rules/KnownArgumentNamesRule.d.ts","./node_modules/graphql/validation/rules/KnownDirectivesRule.d.ts","./node_modules/graphql/validation/rules/KnownFragmentNamesRule.d.ts","./node_modules/graphql/validation/rules/KnownTypeNamesRule.d.ts","./node_modules/graphql/validation/rules/LoneAnonymousOperationRule.d.ts","./node_modules/graphql/validation/rules/NoFragmentCyclesRule.d.ts","./node_modules/graphql/validation/rules/NoUndefinedVariablesRule.d.ts","./node_modules/graphql/validation/rules/NoUnusedFragmentsRule.d.ts","./node_modules/graphql/validation/rules/NoUnusedVariablesRule.d.ts","./node_modules/graphql/validation/rules/OverlappingFieldsCanBeMergedRule.d.ts","./node_modules/graphql/validation/rules/PossibleFragmentSpreadsRule.d.ts","./node_modules/graphql/validation/rules/ProvidedRequiredArgumentsRule.d.ts","./node_modules/graphql/validation/rules/ScalarLeafsRule.d.ts","./node_modules/graphql/validation/rules/SingleFieldSubscriptionsRule.d.ts","./node_modules/graphql/validation/rules/UniqueArgumentNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueDirectivesPerLocationRule.d.ts","./node_modules/graphql/validation/rules/UniqueFragmentNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueInputFieldNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueOperationNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueVariableNamesRule.d.ts","./node_modules/graphql/validation/rules/ValuesOfCorrectTypeRule.d.ts","./node_modules/graphql/validation/rules/VariablesAreInputTypesRule.d.ts","./node_modules/graphql/validation/rules/VariablesInAllowedPositionRule.d.ts","./node_modules/graphql/validation/rules/LoneSchemaDefinitionRule.d.ts","./node_modules/graphql/validation/rules/UniqueOperationTypesRule.d.ts","./node_modules/graphql/validation/rules/UniqueTypeNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueEnumValueNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueFieldDefinitionNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueArgumentDefinitionNamesRule.d.ts","./node_modules/graphql/validation/rules/UniqueDirectiveNamesRule.d.ts","./node_modules/graphql/validation/rules/PossibleTypeExtensionsRule.d.ts","./node_modules/graphql/validation/rules/custom/NoDeprecatedCustomRule.d.ts","./node_modules/graphql/validation/rules/custom/NoSchemaIntrospectionCustomRule.d.ts","./node_modules/graphql/validation/index.d.ts","./node_modules/graphql/error/syntaxError.d.ts","./node_modules/graphql/error/locatedError.d.ts","./node_modules/graphql/error/index.d.ts","./node_modules/graphql/utilities/getIntrospectionQuery.d.ts","./node_modules/graphql/utilities/getOperationAST.d.ts","./node_modules/graphql/utilities/getOperationRootType.d.ts","./node_modules/graphql/utilities/introspectionFromSchema.d.ts","./node_modules/graphql/utilities/buildClientSchema.d.ts","./node_modules/graphql/utilities/buildASTSchema.d.ts","./node_modules/graphql/utilities/extendSchema.d.ts","./node_modules/graphql/utilities/lexicographicSortSchema.d.ts","./node_modules/graphql/utilities/printSchema.d.ts","./node_modules/graphql/utilities/typeFromAST.d.ts","./node_modules/graphql/utilities/valueFromAST.d.ts","./node_modules/graphql/utilities/valueFromASTUntyped.d.ts","./node_modules/graphql/utilities/astFromValue.d.ts","./node_modules/graphql/utilities/coerceInputValue.d.ts","./node_modules/graphql/utilities/concatAST.d.ts","./node_modules/graphql/utilities/separateOperations.d.ts","./node_modules/graphql/utilities/stripIgnoredCharacters.d.ts","./node_modules/graphql/utilities/typeComparators.d.ts","./node_modules/graphql/utilities/assertValidName.d.ts","./node_modules/graphql/utilities/findBreakingChanges.d.ts","./node_modules/graphql/utilities/typedQueryDocumentNode.d.ts","./node_modules/graphql/utilities/resolveSchemaCoordinate.d.ts","./node_modules/graphql/utilities/index.d.ts","./node_modules/graphql/index.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/field-middleware.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/build-schema-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/complexity.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/custom-scalar.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/gql-exception-filter.interface.d.ts","./node_modules/@graphql-typed-document-node/core/typings/index.d.ts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/Interfaces.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/loaders.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/helpers.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getDirectiveExtensions.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/types.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/get-implementing-types.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/validate-documents.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/filterSchema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/heal.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getResolversFromSchema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/forEachField.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/forEachDefaultValue.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mapSchema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/addTypes.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/rewire.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/prune.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mergeDeep.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/stub.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/selectionSets.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getResponseKeyFromInfo.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fields.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/renameType.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/transformInputValue.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/executor.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mapAsyncIterator.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/updateArgument.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/implementsAbstractType.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/errors.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/observableToAsyncIterable.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/visitResult.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getArgumentValues.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/valueMatchesCriteria.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/isAsyncIterable.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/isDocumentNode.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/astFromValueUntyped.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/withCancel.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/rootTypes.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/comments.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/collectFields.d.cts","./node_modules/cross-inspect/typings/index.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/memoize.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fixSchemaAst.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/getOperationASTFromRequest.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/extractExtensionsFromSchema.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/Path.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/jsutils.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/directives.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/mergeIncrementalResult.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/debugTimer.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/map-maybe-promise.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/fakePromise.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/createDeferred.d.cts","./node_modules/@nestjs/graphql/node_modules/@graphql-tools/utils/typings/index.d.cts","./node_modules/@nestjs/graphql/dist/graphql-ast.explorer.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/schema-file-config.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/gql-module-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/graphql-driver.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/resolve-type-fn.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/return-type-func.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/build-federated-schema-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/index.d.ts","./node_modules/@nestjs/graphql/dist/decorators/args.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/context.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/directive.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/extensions.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/hide-field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/info.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/input-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/interface-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/mutation.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/object-type.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/parent.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/query.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-field.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-property.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolve-reference.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/resolver.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/root.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/scalar.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/subscription.decorator.d.ts","./node_modules/@nestjs/graphql/dist/decorators/index.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/type-options.interface.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/directive.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/param.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/resolver.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/orphaned-reference.registry.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/property.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/class.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/enum.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/extensions.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/union.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/type-mapper.service.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/enum-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/services/type-fields.accessor.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/ast-definition-node.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/input-type-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/interface.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/output-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/resolve-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/interface-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/metadata/object-type.metadata.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/object-type-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/union-definition.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/type-definitions.storage.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/input-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/args.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/root-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/mutation-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/orphaned-types.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/query-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/factories/subscription-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/type-definitions.generator.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/graphql-schema.factory.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/helpers/file-system.helper.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/resolver-metadata.interface.d.ts","./node_modules/@nestjs/graphql/dist/services/base-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-arguments-host.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-execution-context.d.ts","./node_modules/graphql-ws/lib/common.d.ts","./node_modules/graphql-ws/lib/client.d.ts","./node_modules/graphql-ws/lib/server.d.ts","./node_modules/graphql-ws/lib/index.d.ts","./node_modules/subscriptions-transport-ws/node_modules/eventemitter3/index.d.ts","./node_modules/subscriptions-transport-ws/dist/client.d.ts","./node_modules/subscriptions-transport-ws/dist/server.d.ts","./node_modules/subscriptions-transport-ws/dist/message-types.d.ts","./node_modules/subscriptions-transport-ws/dist/protocol.d.ts","./node_modules/subscriptions-transport-ws/dist/index.d.ts","./node_modules/@nestjs/graphql/dist/services/gql-subscription.service.d.ts","./node_modules/@nestjs/graphql/dist/services/resolvers-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/scalars-explorer.service.d.ts","./node_modules/@nestjs/graphql/dist/services/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-schema.builder.d.ts","./node_modules/@nestjs/graphql/dist/graphql.factory.d.ts","./node_modules/@nestjs/graphql/dist/drivers/abstract-graphql.driver.d.ts","./node_modules/@nestjs/graphql/dist/drivers/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-types.loader.d.ts","./node_modules/@nestjs/graphql/dist/graphql-definitions.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/graphql-federation-definitions.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/type-defs-decorator.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/graphql-federation.factory.d.ts","./node_modules/@nestjs/graphql/dist/federation/index.d.ts","./node_modules/@nestjs/graphql/dist/graphql-schema.host.d.ts","./node_modules/@nestjs/graphql/dist/graphql.constants.d.ts","./node_modules/@nestjs/graphql/dist/graphql.module.d.ts","./node_modules/@nestjs/graphql/dist/scalars/iso-date.scalar.d.ts","./node_modules/@nestjs/graphql/dist/scalars/timestamp.scalar.d.ts","./node_modules/@nestjs/graphql/dist/scalars/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/type-metadata.storage.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/storages/index.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/schema-builder.module.d.ts","./node_modules/@nestjs/graphql/dist/schema-builder/index.d.ts","./node_modules/@nestjs/graphql/dist/tokens.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/create-union-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/register-enum-type.factory.d.ts","./node_modules/@nestjs/graphql/dist/type-factories/index.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/field-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/interfaces/class-decorator-factory.interface.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/intersection-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/omit-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/partial-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/pick-type.helper.d.ts","./node_modules/@nestjs/graphql/dist/type-helpers/index.d.ts","./node_modules/@nestjs/graphql/dist/utils/extend.util.d.ts","./node_modules/@nestjs/graphql/dist/utils/transform-schema.util.d.ts","./node_modules/@nestjs/graphql/dist/index.d.ts","./node_modules/@nestjs/apollo/dist/decorators/plugin.decorator.d.ts","./node_modules/@nestjs/apollo/dist/decorators/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/KeyValueCache.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/PrefixingKeyValueCache.d.ts","./node_modules/lru-cache/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/InMemoryLRUCache.d.ts","./node_modules/@apollo/utils.logger/dist/index.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/ErrorsAreMissesCache.d.ts","./node_modules/@apollo/utils.keyvaluecache/dist/index.d.ts","./node_modules/@apollo/protobufjs/index.d.ts","./node_modules/@apollo/usage-reporting-protobuf/generated/cjs/protobuf.d.ts","./node_modules/@apollo/utils.fetcher/dist/index.d.ts","./node_modules/@apollo/server-gateway-interface/dist/cjs/index.d.ts","./node_modules/@apollo/utils.withrequired/dist/index.d.ts","./node_modules/@apollo/server/dist/cjs/utils/resolvable.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/context.d.ts","./node_modules/@apollo/server/dist/cjs/utils/HeaderMap.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/http.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/incrementalDeliveryPolyfill.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/graphql.d.ts","./node_modules/@graphql-tools/utils/typings/Interfaces.d.cts","./node_modules/@graphql-tools/utils/typings/loaders.d.cts","./node_modules/@graphql-tools/utils/typings/helpers.d.cts","./node_modules/@graphql-tools/utils/typings/get-directives.d.cts","./node_modules/@graphql-tools/utils/typings/types.d.cts","./node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/get-implementing-types.d.cts","./node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.cts","./node_modules/@graphql-tools/utils/typings/validate-documents.d.cts","./node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.cts","./node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.cts","./node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.cts","./node_modules/@graphql-tools/utils/typings/filterSchema.d.cts","./node_modules/@graphql-tools/utils/typings/heal.d.cts","./node_modules/@graphql-tools/utils/typings/getResolversFromSchema.d.cts","./node_modules/@graphql-tools/utils/typings/forEachField.d.cts","./node_modules/@graphql-tools/utils/typings/forEachDefaultValue.d.cts","./node_modules/@graphql-tools/utils/typings/mapSchema.d.cts","./node_modules/@graphql-tools/utils/typings/addTypes.d.cts","./node_modules/@graphql-tools/utils/typings/rewire.d.cts","./node_modules/@graphql-tools/utils/typings/prune.d.cts","./node_modules/@graphql-tools/utils/typings/mergeDeep.d.cts","./node_modules/@graphql-tools/utils/typings/stub.d.cts","./node_modules/@graphql-tools/utils/typings/selectionSets.d.cts","./node_modules/@graphql-tools/utils/typings/getResponseKeyFromInfo.d.cts","./node_modules/@graphql-tools/utils/typings/fields.d.cts","./node_modules/@graphql-tools/utils/typings/renameType.d.cts","./node_modules/@graphql-tools/utils/typings/transformInputValue.d.cts","./node_modules/@graphql-tools/utils/typings/mapAsyncIterator.d.cts","./node_modules/@graphql-tools/utils/typings/updateArgument.d.cts","./node_modules/@graphql-tools/utils/typings/implementsAbstractType.d.cts","./node_modules/@graphql-tools/utils/typings/errors.d.cts","./node_modules/@graphql-tools/utils/typings/observableToAsyncIterable.d.cts","./node_modules/@graphql-tools/utils/typings/visitResult.d.cts","./node_modules/@graphql-tools/utils/typings/getArgumentValues.d.cts","./node_modules/@graphql-tools/utils/typings/valueMatchesCriteria.d.cts","./node_modules/@graphql-tools/utils/typings/isAsyncIterable.d.cts","./node_modules/@graphql-tools/utils/typings/isDocumentNode.d.cts","./node_modules/@graphql-tools/utils/typings/astFromValueUntyped.d.cts","./node_modules/@graphql-tools/utils/typings/executor.d.cts","./node_modules/@graphql-tools/utils/typings/withCancel.d.cts","./node_modules/@graphql-tools/utils/typings/AggregateError.d.cts","./node_modules/@graphql-tools/utils/typings/rootTypes.d.cts","./node_modules/@graphql-tools/utils/typings/comments.d.cts","./node_modules/@graphql-tools/utils/typings/collectFields.d.cts","./node_modules/@graphql-tools/utils/typings/inspect.d.cts","./node_modules/@graphql-tools/utils/typings/memoize.d.cts","./node_modules/@graphql-tools/utils/typings/fixSchemaAst.d.cts","./node_modules/@graphql-tools/utils/typings/getOperationASTFromRequest.d.cts","./node_modules/@graphql-tools/utils/typings/extractExtensionsFromSchema.d.cts","./node_modules/@graphql-tools/utils/typings/Path.d.cts","./node_modules/@graphql-tools/utils/typings/jsutils.d.cts","./node_modules/@graphql-tools/utils/typings/directives.d.cts","./node_modules/@graphql-tools/utils/typings/index.d.cts","./node_modules/@graphql-tools/schema/typings/assertResolversPresent.d.cts","./node_modules/@graphql-tools/schema/typings/chainResolvers.d.cts","./node_modules/@graphql-tools/schema/typings/addResolversToSchema.d.cts","./node_modules/@graphql-tools/schema/typings/checkForResolveTypeResolver.d.cts","./node_modules/@graphql-tools/schema/typings/extendResolversFromInterfaces.d.cts","./node_modules/@graphql-tools/schema/typings/types.d.cts","./node_modules/@graphql-tools/schema/typings/makeExecutableSchema.d.cts","./node_modules/@graphql-tools/schema/typings/merge-schemas.d.cts","./node_modules/@graphql-tools/schema/typings/index.d.cts","./node_modules/@apollo/server/dist/cjs/incrementalDeliveryPolyfill.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/constructor.d.ts","./node_modules/@apollo/cache-control-types/dist/cjs/index.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/requestPipeline.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/plugins.d.ts","./node_modules/@apollo/server/dist/cjs/externalTypes/index.d.ts","./node_modules/@apollo/server/dist/cjs/utils/schemaManager.d.ts","./node_modules/@apollo/server/dist/cjs/ApolloServer.d.ts","./node_modules/@apollo/server/dist/cjs/index.d.ts","./node_modules/@apollographql/graphql-playground-html/dist/render-playground-page.d.ts","./node_modules/@apollographql/graphql-playground-html/dist/index.d.ts","./node_modules/@apollo/usage-reporting-protobuf/generated/esm/protobuf.d.ts","./node_modules/@apollo/server-gateway-interface/dist/esm/index.d.ts","./node_modules/@apollo/server/dist/esm/utils/resolvable.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/context.d.ts","./node_modules/@apollo/server/dist/esm/utils/HeaderMap.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/http.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/incrementalDeliveryPolyfill.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/graphql.d.ts","./node_modules/@graphql-tools/utils/typings/Interfaces.d.ts","./node_modules/@graphql-tools/utils/typings/loaders.d.ts","./node_modules/@graphql-tools/utils/typings/helpers.d.ts","./node_modules/@graphql-tools/utils/typings/get-directives.d.ts","./node_modules/@graphql-tools/utils/typings/types.d.ts","./node_modules/@graphql-tools/utils/typings/get-fields-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/get-arguments-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/get-implementing-types.d.ts","./node_modules/@graphql-tools/utils/typings/print-schema-with-directives.d.ts","./node_modules/@graphql-tools/utils/typings/validate-documents.d.ts","./node_modules/@graphql-tools/utils/typings/parse-graphql-json.d.ts","./node_modules/@graphql-tools/utils/typings/parse-graphql-sdl.d.ts","./node_modules/@graphql-tools/utils/typings/build-operation-for-field.d.ts","./node_modules/@graphql-tools/utils/typings/filterSchema.d.ts","./node_modules/@graphql-tools/utils/typings/heal.d.ts","./node_modules/@graphql-tools/utils/typings/getResolversFromSchema.d.ts","./node_modules/@graphql-tools/utils/typings/forEachField.d.ts","./node_modules/@graphql-tools/utils/typings/forEachDefaultValue.d.ts","./node_modules/@graphql-tools/utils/typings/mapSchema.d.ts","./node_modules/@graphql-tools/utils/typings/addTypes.d.ts","./node_modules/@graphql-tools/utils/typings/rewire.d.ts","./node_modules/@graphql-tools/utils/typings/prune.d.ts","./node_modules/@graphql-tools/utils/typings/mergeDeep.d.ts","./node_modules/@graphql-tools/utils/typings/stub.d.ts","./node_modules/@graphql-tools/utils/typings/selectionSets.d.ts","./node_modules/@graphql-tools/utils/typings/getResponseKeyFromInfo.d.ts","./node_modules/@graphql-tools/utils/typings/fields.d.ts","./node_modules/@graphql-tools/utils/typings/renameType.d.ts","./node_modules/@graphql-tools/utils/typings/transformInputValue.d.ts","./node_modules/@graphql-tools/utils/typings/mapAsyncIterator.d.ts","./node_modules/@graphql-tools/utils/typings/updateArgument.d.ts","./node_modules/@graphql-tools/utils/typings/implementsAbstractType.d.ts","./node_modules/@graphql-tools/utils/typings/errors.d.ts","./node_modules/@graphql-tools/utils/typings/observableToAsyncIterable.d.ts","./node_modules/@graphql-tools/utils/typings/visitResult.d.ts","./node_modules/@graphql-tools/utils/typings/getArgumentValues.d.ts","./node_modules/@graphql-tools/utils/typings/valueMatchesCriteria.d.ts","./node_modules/@graphql-tools/utils/typings/isAsyncIterable.d.ts","./node_modules/@graphql-tools/utils/typings/isDocumentNode.d.ts","./node_modules/@graphql-tools/utils/typings/astFromValueUntyped.d.ts","./node_modules/@graphql-tools/utils/typings/executor.d.ts","./node_modules/@graphql-tools/utils/typings/withCancel.d.ts","./node_modules/@graphql-tools/utils/typings/AggregateError.d.ts","./node_modules/@graphql-tools/utils/typings/rootTypes.d.ts","./node_modules/@graphql-tools/utils/typings/comments.d.ts","./node_modules/@graphql-tools/utils/typings/collectFields.d.ts","./node_modules/@graphql-tools/utils/typings/inspect.d.ts","./node_modules/@graphql-tools/utils/typings/memoize.d.ts","./node_modules/@graphql-tools/utils/typings/fixSchemaAst.d.ts","./node_modules/@graphql-tools/utils/typings/getOperationASTFromRequest.d.ts","./node_modules/@graphql-tools/utils/typings/extractExtensionsFromSchema.d.ts","./node_modules/@graphql-tools/utils/typings/Path.d.ts","./node_modules/@graphql-tools/utils/typings/jsutils.d.ts","./node_modules/@graphql-tools/utils/typings/directives.d.ts","./node_modules/@graphql-tools/utils/typings/index.d.ts","./node_modules/@graphql-tools/schema/typings/assertResolversPresent.d.ts","./node_modules/@graphql-tools/schema/typings/chainResolvers.d.ts","./node_modules/@graphql-tools/schema/typings/addResolversToSchema.d.ts","./node_modules/@graphql-tools/schema/typings/checkForResolveTypeResolver.d.ts","./node_modules/@graphql-tools/schema/typings/extendResolversFromInterfaces.d.ts","./node_modules/@graphql-tools/schema/typings/types.d.ts","./node_modules/@graphql-tools/schema/typings/makeExecutableSchema.d.ts","./node_modules/@graphql-tools/schema/typings/merge-schemas.d.ts","./node_modules/@graphql-tools/schema/typings/index.d.ts","./node_modules/@apollo/server/dist/esm/incrementalDeliveryPolyfill.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/constructor.d.ts","./node_modules/@apollo/cache-control-types/dist/esm/index.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/requestPipeline.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/plugins.d.ts","./node_modules/@apollo/server/dist/esm/externalTypes/index.d.ts","./node_modules/@apollo/server/dist/esm/utils/schemaManager.d.ts","./node_modules/@apollo/server/dist/esm/ApolloServer.d.ts","./node_modules/@apollo/server/dist/esm/index.d.ts","./node_modules/@apollo/server-plugin-landing-page-graphql-playground/dist/esm/index.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-federation-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/apollo-gateway-driver-config.interface.d.ts","./node_modules/@nestjs/apollo/dist/interfaces/index.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-base.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-federation.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo-gateway.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/apollo.driver.d.ts","./node_modules/@nestjs/apollo/dist/drivers/index.d.ts","./node_modules/@nestjs/apollo/dist/errors/authentication.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/forbidden.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/user-input.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/validation.error.d.ts","./node_modules/@nestjs/apollo/dist/errors/index.d.ts","./node_modules/@nestjs/apollo/dist/utils/get-apollo-server.d.ts","./node_modules/@nestjs/apollo/dist/utils/index.d.ts","./node_modules/@nestjs/apollo/dist/services/plugins-explorer.service.d.ts","./node_modules/@nestjs/apollo/dist/services/index.d.ts","./node_modules/@nestjs/apollo/dist/index.d.ts","./node_modules/graphql-subscriptions/dist/pubsub-async-iterable-iterator.d.ts","./node_modules/graphql-subscriptions/dist/pubsub-engine.d.ts","./node_modules/graphql-subscriptions/dist/pubsub.d.ts","./node_modules/graphql-subscriptions/dist/with-filter.d.ts","./node_modules/graphql-subscriptions/dist/index.d.ts","./src/common/validators/password.validator.ts","./src/users/dto/create-user.dto.ts","./src/users/dto/update-user.dto.ts","./node_modules/bcryptjs/umd/types.d.ts","./node_modules/bcryptjs/umd/index.d.ts","./src/common/utils/user.utils.ts","./src/common/constants/app.constants.ts","./src/common/dto/pagination.dto.ts","./src/common/utils/pagination.util.ts","./src/common/utils/sanitization.utils.ts","./src/users/dto/get-users.dto.ts","./src/caching/caching.constants.ts","./src/caching/caching.service.ts","./src/users/users.service.ts","./src/auth/decorators/current-user.decorator.ts","./src/common/export/export.service.ts","./src/users/users.controller.ts","./src/users/users.module.ts","./src/courses/dto/create-course.dto.ts","./src/courses/dto/update-course.dto.ts","./src/courses/dto/course-search.dto.ts","./src/caching/strategies/cache-strategies.service.ts","./src/caching/invalidation/invalidation.service.ts","./src/courses/courses.service.ts","./src/courses/dto/create-module.dto.ts","./src/courses/modules/modules.service.ts","./src/courses/dto/create-lesson.dto.ts","./src/courses/lessons/lessons.service.ts","./src/courses/enrollments/enrollments.service.ts","./src/courses/courses.controller.ts","./src/courses/courses.module.ts","./src/assessment/enums/assessment-status.enum.ts","./src/assessment/enums/question-type.enum.ts","./src/assessment/entities/question.entity.ts","./src/assessment/entities/assessment.entity.ts","./src/assessment/entities/answer.entity.ts","./src/assessment/entities/assessment-attempt.entity.ts","./src/assessment/feedback/feedback-generation.service.ts","./src/assessment/scoring/score-calculation.service.ts","./src/assessment/assessments.service.ts","./src/assessment/assessment.controller.ts","./src/assessment/questions/question-bank.service.ts","./src/assessment/assessment.module.ts","./node_modules/@types/ms/index.d.ts","./node_modules/@types/jsonwebtoken/index.d.ts","./node_modules/@nestjs/jwt/dist/interfaces/jwt-module-options.interface.d.ts","./node_modules/@nestjs/jwt/dist/interfaces/index.d.ts","./node_modules/@nestjs/jwt/dist/jwt.errors.d.ts","./node_modules/@nestjs/jwt/dist/jwt.module.d.ts","./node_modules/@nestjs/jwt/dist/jwt.service.d.ts","./node_modules/@nestjs/jwt/dist/index.d.ts","./node_modules/@nestjs/jwt/index.d.ts","./src/auth/dto/auth.dto.ts","./src/notifications/entities/notification.entity.ts","./src/notifications/entities/notification-preferences.entity.ts","./src/notifications/dto/notification.dto.ts","./src/auth/guards/ws-jwt-auth.guard.ts","./src/notifications/notifications.gateway.ts","./src/notifications/notification-templates.service.ts","./src/notifications/preferences/preferences.service.ts","./node_modules/@types/nodemailer/lib/dkim/index.d.ts","./node_modules/@types/nodemailer/lib/mailer/mail-message.d.ts","./node_modules/@types/nodemailer/lib/xoauth2/index.d.ts","./node_modules/@types/nodemailer/lib/mailer/index.d.ts","./node_modules/@types/nodemailer/lib/mime-node/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-connection/index.d.ts","./node_modules/@types/nodemailer/lib/shared/index.d.ts","./node_modules/@types/nodemailer/lib/json-transport/index.d.ts","./node_modules/@types/nodemailer/lib/sendmail-transport/index.d.ts","./node_modules/@types/nodemailer/lib/ses-transport/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-pool/index.d.ts","./node_modules/@types/nodemailer/lib/smtp-transport/index.d.ts","./node_modules/@types/nodemailer/lib/stream-transport/index.d.ts","./node_modules/@types/nodemailer/index.d.ts","./node_modules/handlebars/types/index.d.ts","./src/notifications/email/email.service.ts","./src/notifications/notifications.service.ts","./src/audit-log/enums/audit-action.enum.ts","./src/audit-log/audit-log.entity.ts","./src/audit-log/audit-log.service.ts","./src/auth/auth.service.ts","./src/auth/auth.controller.ts","./node_modules/@types/passport-strategy/index.d.ts","./node_modules/@types/passport-jwt/index.d.ts","./src/auth/strategies/jwt.strategy.ts","./src/notifications/notifications.controller.ts","./src/notifications/email/email.processor.ts","./src/notifications/notifications.module.ts","./src/audit-log/audit-log.controller.ts","./src/audit-log/interceptors/audit-log.interceptor.ts","./src/audit-log/tasks/audit-retention.task.ts","./src/audit-log/audit-log.module.ts","./src/auth/auth.module.ts","./src/graphql/types/course.type.ts","./src/graphql/types/user.type.ts","./src/graphql/types/assessment.type.ts","./src/graphql/inputs/user.input.ts","./src/graphql/inputs/course.input.ts","./src/graphql/resolvers/query.resolver.ts","./src/graphql/inputs/assessment.input.ts","./src/graphql/resolvers/mutation.resolver.ts","./src/graphql/resolvers/subscription.resolver.ts","./src/graphql/resolvers/user.resolver.ts","./src/graphql/resolvers/course.resolver.ts","./src/graphql/resolvers/assessment.resolver.ts","./node_modules/dataloader/index.d.ts","./src/graphql/services/dataloader.service.ts","./src/graphql/services/query-complexity.service.ts","./src/graphql/graphql.module.ts","./src/migrations/entities/migration.entity.ts","./src/migrations/conflicts/conflict-resolution.service.ts","./src/migrations/validation/schema-validation.service.ts","./src/migrations/samples/001-create-users-table.migration.ts","./src/migrations/samples/002-create-courses-table.migration.ts","./src/migrations/samples/003-create-course-modules-table.migration.ts","./src/migrations/samples/004-create-lessons-table.migration.ts","./src/migrations/samples/005-create-enrollments-table.migration.ts","./src/migrations/samples/006-create-migrations-tracking-table.migration.ts","./src/migrations/migration.registry.ts","./src/migrations/rollback/rollback.service.ts","./src/migrations/environments/environment-sync.service.ts","./src/migrations/migration.service.ts","./src/migrations/migration.controller.ts","./src/migrations/migration-runner.service.ts","./src/migrations/migration.module.ts","./src/ab-testing/entities/variant-metric.entity.ts","./src/ab-testing/entities/experiment-variant.entity.ts","./src/ab-testing/entities/experiment-metric.entity.ts","./src/ab-testing/entities/experiment.entity.ts","./src/ab-testing/ab-testing.service.ts","./src/ab-testing/experiments/experiment.service.ts","./src/ab-testing/analysis/statistical-analysis.service.ts","./src/ab-testing/automation/automated-decision.service.ts","./src/ab-testing/reporting/ab-testing-reports.service.ts","./src/ab-testing/ab-testing.controller.ts","./src/ab-testing/ab-testing.module.ts","./src/observability/interfaces/observability.interfaces.ts","./src/observability/logging/structured-logger.service.ts","./src/observability/logging/log-aggregation.service.ts","./node_modules/@opentelemetry/api/build/src/baggage/internal/symbol.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/types.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/utils.d.ts","./node_modules/@opentelemetry/api/build/src/common/Exception.d.ts","./node_modules/@opentelemetry/api/build/src/common/Time.d.ts","./node_modules/@opentelemetry/api/build/src/common/Attributes.d.ts","./node_modules/@opentelemetry/api/build/src/context/types.d.ts","./node_modules/@opentelemetry/api/build/src/context/context.d.ts","./node_modules/@opentelemetry/api/build/src/api/context.d.ts","./node_modules/@opentelemetry/api/build/src/diag/types.d.ts","./node_modules/@opentelemetry/api/build/src/diag/consoleLogger.d.ts","./node_modules/@opentelemetry/api/build/src/api/diag.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/Metric.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/Meter.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/NoopMeter.d.ts","./node_modules/@opentelemetry/api/build/src/metrics/MeterProvider.d.ts","./node_modules/@opentelemetry/api/build/src/api/metrics.d.ts","./node_modules/@opentelemetry/api/build/src/propagation/TextMapPropagator.d.ts","./node_modules/@opentelemetry/api/build/src/baggage/context-helpers.d.ts","./node_modules/@opentelemetry/api/build/src/api/propagation.d.ts","./node_modules/@opentelemetry/api/build/src/trace/attributes.d.ts","./node_modules/@opentelemetry/api/build/src/trace/trace_state.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span_context.d.ts","./node_modules/@opentelemetry/api/build/src/trace/link.d.ts","./node_modules/@opentelemetry/api/build/src/trace/status.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span.d.ts","./node_modules/@opentelemetry/api/build/src/trace/span_kind.d.ts","./node_modules/@opentelemetry/api/build/src/trace/SpanOptions.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer_options.d.ts","./node_modules/@opentelemetry/api/build/src/trace/ProxyTracer.d.ts","./node_modules/@opentelemetry/api/build/src/trace/tracer_provider.d.ts","./node_modules/@opentelemetry/api/build/src/trace/ProxyTracerProvider.d.ts","./node_modules/@opentelemetry/api/build/src/trace/SamplingResult.d.ts","./node_modules/@opentelemetry/api/build/src/trace/Sampler.d.ts","./node_modules/@opentelemetry/api/build/src/trace/trace_flags.d.ts","./node_modules/@opentelemetry/api/build/src/trace/internal/utils.d.ts","./node_modules/@opentelemetry/api/build/src/trace/spancontext-utils.d.ts","./node_modules/@opentelemetry/api/build/src/trace/invalid-span-constants.d.ts","./node_modules/@opentelemetry/api/build/src/trace/context-utils.d.ts","./node_modules/@opentelemetry/api/build/src/api/trace.d.ts","./node_modules/@opentelemetry/api/build/src/context-api.d.ts","./node_modules/@opentelemetry/api/build/src/diag-api.d.ts","./node_modules/@opentelemetry/api/build/src/metrics-api.d.ts","./node_modules/@opentelemetry/api/build/src/propagation-api.d.ts","./node_modules/@opentelemetry/api/build/src/trace-api.d.ts","./node_modules/@opentelemetry/api/build/src/index.d.ts","./src/observability/tracing/distributed-tracing.service.ts","./src/observability/metrics/metrics-analysis.service.ts","./src/observability/anomaly/anomaly-detection.service.ts","./src/observability/observability.service.ts","./src/observability/observability.controller.ts","./src/observability/observability.module.ts","./src/rate-limiting/services/distrubutes.service.ts","./src/rate-limiting/services/quota.service.ts","./src/rate-limiting/services/adaptive-rate-limiting.service.ts","./src/rate-limiting/services/throttling.service.ts","./src/rate-limiting/dto/create-rate-limiting.dto.ts","./src/rate-limiting/dto/update-rate-limiting.dto.ts","./src/rate-limiting/rate-limiting.service.ts","./src/rate-limiting/services/rate-limiting.module.ts","./src/caching/warming/cache-warming.service.ts","./src/caching/analytics/cache-analytics.service.ts","./src/caching/cache-management.controller.ts","./src/caching/caching.module.ts","./src/feature-flags/interfaces/index.ts","./src/feature-flags/analytics/flag-analytics.service.ts","./src/feature-flags/rollout/rollout.service.ts","./src/feature-flags/experimentation/experimentation.service.ts","./src/feature-flags/targeting/targeting.service.ts","./src/feature-flags/evaluation/flag-evaluation.service.ts","./src/feature-flags/feature-flags.module.ts","./node_modules/@elastic/transport/lib/symbols.d.ts","./node_modules/@elastic/transport/lib/connection/BaseConnection.d.ts","./node_modules/hpagent/index.d.ts","./node_modules/@elastic/transport/lib/connection/HttpConnection.d.ts","./node_modules/undici/types/header.d.ts","./node_modules/undici/types/readable.d.ts","./node_modules/undici/types/file.d.ts","./node_modules/undici/types/fetch.d.ts","./node_modules/undici/types/formdata.d.ts","./node_modules/undici/types/connector.d.ts","./node_modules/undici/types/client.d.ts","./node_modules/undici/types/errors.d.ts","./node_modules/undici/types/dispatcher.d.ts","./node_modules/undici/types/global-dispatcher.d.ts","./node_modules/undici/types/global-origin.d.ts","./node_modules/undici/types/pool-stats.d.ts","./node_modules/undici/types/pool.d.ts","./node_modules/undici/types/handlers.d.ts","./node_modules/undici/types/balanced-pool.d.ts","./node_modules/undici/types/agent.d.ts","./node_modules/undici/types/mock-interceptor.d.ts","./node_modules/undici/types/mock-agent.d.ts","./node_modules/undici/types/mock-client.d.ts","./node_modules/undici/types/mock-pool.d.ts","./node_modules/undici/types/mock-errors.d.ts","./node_modules/undici/types/proxy-agent.d.ts","./node_modules/undici/types/env-http-proxy-agent.d.ts","./node_modules/undici/types/retry-handler.d.ts","./node_modules/undici/types/retry-agent.d.ts","./node_modules/undici/types/api.d.ts","./node_modules/undici/types/interceptors.d.ts","./node_modules/undici/types/util.d.ts","./node_modules/undici/types/cookies.d.ts","./node_modules/undici/types/patch.d.ts","./node_modules/undici/types/websocket.d.ts","./node_modules/undici/types/eventsource.d.ts","./node_modules/undici/types/filereader.d.ts","./node_modules/undici/types/diagnostics-channel.d.ts","./node_modules/undici/types/content-type.d.ts","./node_modules/undici/types/cache.d.ts","./node_modules/undici/types/index.d.ts","./node_modules/undici/index.d.ts","./node_modules/@elastic/transport/lib/connection/UndiciConnection.d.ts","./node_modules/@elastic/transport/lib/connection/index.d.ts","./node_modules/@elastic/transport/lib/Serializer.d.ts","./node_modules/@elastic/transport/lib/pool/BaseConnectionPool.d.ts","./node_modules/@elastic/transport/lib/pool/WeightedConnectionPool.d.ts","./node_modules/@elastic/transport/lib/pool/ClusterConnectionPool.d.ts","./node_modules/@elastic/transport/lib/pool/CloudConnectionPool.d.ts","./node_modules/@elastic/transport/lib/pool/index.d.ts","./node_modules/@elastic/transport/lib/Transport.d.ts","./node_modules/@elastic/transport/lib/types.d.ts","./node_modules/@elastic/transport/lib/errors.d.ts","./node_modules/@elastic/transport/lib/Diagnostic.d.ts","./node_modules/@elastic/transport/index.d.ts","./node_modules/@elastic/elasticsearch/lib/sniffingTransport.d.ts","./node_modules/flatbuffers/js/constants.d.ts","./node_modules/flatbuffers/js/encoding.d.ts","./node_modules/flatbuffers/js/byte-buffer.d.ts","./node_modules/flatbuffers/js/builder.d.ts","./node_modules/flatbuffers/js/types.d.ts","./node_modules/flatbuffers/js/utils.d.ts","./node_modules/flatbuffers/js/flatbuffers.d.ts","./node_modules/apache-arrow/fb/body-compression-method.d.ts","./node_modules/apache-arrow/fb/compression-type.d.ts","./node_modules/apache-arrow/fb/body-compression.d.ts","./node_modules/apache-arrow/fb/buffer.d.ts","./node_modules/apache-arrow/fb/field-node.d.ts","./node_modules/apache-arrow/fb/record-batch.d.ts","./node_modules/apache-arrow/fb/dictionary-batch.d.ts","./node_modules/apache-arrow/fb/endianness.d.ts","./node_modules/apache-arrow/fb/dictionary-kind.d.ts","./node_modules/apache-arrow/fb/int.d.ts","./node_modules/apache-arrow/fb/dictionary-encoding.d.ts","./node_modules/apache-arrow/fb/key-value.d.ts","./node_modules/apache-arrow/fb/binary.d.ts","./node_modules/apache-arrow/fb/bool.d.ts","./node_modules/apache-arrow/fb/date-unit.d.ts","./node_modules/apache-arrow/fb/date.d.ts","./node_modules/apache-arrow/fb/decimal.d.ts","./node_modules/apache-arrow/fb/time-unit.d.ts","./node_modules/apache-arrow/fb/duration.d.ts","./node_modules/apache-arrow/fb/fixed-size-binary.d.ts","./node_modules/apache-arrow/fb/fixed-size-list.d.ts","./node_modules/apache-arrow/fb/precision.d.ts","./node_modules/apache-arrow/fb/floating-point.d.ts","./node_modules/apache-arrow/fb/interval-unit.d.ts","./node_modules/apache-arrow/fb/interval.d.ts","./node_modules/apache-arrow/fb/large-binary.d.ts","./node_modules/apache-arrow/fb/large-list.d.ts","./node_modules/apache-arrow/fb/large-utf8.d.ts","./node_modules/apache-arrow/fb/list.d.ts","./node_modules/apache-arrow/fb/map.d.ts","./node_modules/apache-arrow/fb/null.d.ts","./node_modules/apache-arrow/fb/run-end-encoded.d.ts","./node_modules/apache-arrow/fb/struct-.d.ts","./node_modules/apache-arrow/fb/time.d.ts","./node_modules/apache-arrow/fb/timestamp.d.ts","./node_modules/apache-arrow/fb/union-mode.d.ts","./node_modules/apache-arrow/fb/union.d.ts","./node_modules/apache-arrow/fb/utf8.d.ts","./node_modules/apache-arrow/fb/type.d.ts","./node_modules/apache-arrow/fb/field.d.ts","./node_modules/apache-arrow/fb/schema.d.ts","./node_modules/apache-arrow/fb/sparse-matrix-compressed-axis.d.ts","./node_modules/apache-arrow/fb/sparse-matrix-index-csx.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index-coo.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index-csf.d.ts","./node_modules/apache-arrow/fb/sparse-tensor-index.d.ts","./node_modules/apache-arrow/fb/tensor-dim.d.ts","./node_modules/apache-arrow/fb/sparse-tensor.d.ts","./node_modules/apache-arrow/fb/tensor.d.ts","./node_modules/apache-arrow/fb/message-header.d.ts","./node_modules/apache-arrow/fb/metadata-version.d.ts","./node_modules/apache-arrow/enum.d.ts","./node_modules/apache-arrow/schema.d.ts","./node_modules/apache-arrow/row/map.d.ts","./node_modules/apache-arrow/row/struct.d.ts","./node_modules/apache-arrow/builder/buffer.d.ts","./node_modules/apache-arrow/io/node/builder.d.ts","./node_modules/apache-arrow/io/whatwg/builder.d.ts","./node_modules/apache-arrow/builder.d.ts","./node_modules/apache-arrow/builder/bool.d.ts","./node_modules/apache-arrow/builder/null.d.ts","./node_modules/apache-arrow/builder/date.d.ts","./node_modules/apache-arrow/builder/decimal.d.ts","./node_modules/apache-arrow/builder/int.d.ts","./node_modules/apache-arrow/builder/dictionary.d.ts","./node_modules/apache-arrow/builder/fixedsizebinary.d.ts","./node_modules/apache-arrow/builder/float.d.ts","./node_modules/apache-arrow/builder/time.d.ts","./node_modules/apache-arrow/builder/timestamp.d.ts","./node_modules/apache-arrow/builder/interval.d.ts","./node_modules/apache-arrow/builder/duration.d.ts","./node_modules/apache-arrow/builder/utf8.d.ts","./node_modules/apache-arrow/builder/largeutf8.d.ts","./node_modules/apache-arrow/builder/binary.d.ts","./node_modules/apache-arrow/builder/largebinary.d.ts","./node_modules/apache-arrow/builder/list.d.ts","./node_modules/apache-arrow/builder/fixedsizelist.d.ts","./node_modules/apache-arrow/builder/map.d.ts","./node_modules/apache-arrow/builder/struct.d.ts","./node_modules/apache-arrow/builder/union.d.ts","./node_modules/apache-arrow/interfaces.d.ts","./node_modules/apache-arrow/type.d.ts","./node_modules/apache-arrow/vector.d.ts","./node_modules/apache-arrow/data.d.ts","./node_modules/apache-arrow/recordbatch.d.ts","./node_modules/apache-arrow/table.d.ts","./node_modules/apache-arrow/visitor.d.ts","./node_modules/apache-arrow/factories.d.ts","./node_modules/apache-arrow/io/interfaces.d.ts","./node_modules/apache-arrow/util/buffer.d.ts","./node_modules/apache-arrow/io/stream.d.ts","./node_modules/apache-arrow/fb/block.d.ts","./node_modules/apache-arrow/ipc/metadata/file.d.ts","./node_modules/apache-arrow/ipc/metadata/json.d.ts","./node_modules/apache-arrow/ipc/metadata/message.d.ts","./node_modules/apache-arrow/io/file.d.ts","./node_modules/apache-arrow/ipc/message.d.ts","./node_modules/apache-arrow/ipc/reader.d.ts","./node_modules/apache-arrow/ipc/writer.d.ts","./node_modules/apache-arrow/ipc/serialization.d.ts","./node_modules/apache-arrow/ipc/compression/registry.d.ts","./node_modules/apache-arrow/util/bn.d.ts","./node_modules/apache-arrow/util/int.d.ts","./node_modules/apache-arrow/util/bit.d.ts","./node_modules/apache-arrow/util/interval.d.ts","./node_modules/apache-arrow/visitor/typecomparator.d.ts","./node_modules/apache-arrow/Arrow.d.ts","./node_modules/apache-arrow/Arrow.dom.d.ts","./node_modules/apache-arrow/Arrow.node.d.ts","./node_modules/@elastic/elasticsearch/lib/api/types.d.ts","./node_modules/@elastic/elasticsearch/lib/helpers.d.ts","./node_modules/@elastic/elasticsearch/lib/api/typesWithBodyKey.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/async_search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/autoscaling.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/bulk.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/capabilities.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/cat.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ccr.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/clear_scroll.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/close_point_in_time.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/cluster.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/connector.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/count.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/create.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/dangling_indices.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_by_query.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_by_query_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/delete_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/enrich.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/eql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/esql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/exists.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/exists_source.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/explain.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/features.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/field_caps.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/fleet.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script_context.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_script_languages.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/get_source.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/graph.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/health_report.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ilm.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/index.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/indices.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/inference.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/info.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ingest.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/knn_search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/license.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/logstash.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/mget.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/migration.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ml.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/monitoring.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/msearch.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/msearch_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/mtermvectors.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/nodes.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/open_point_in_time.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ping.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/profiling.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/put_script.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/query_rules.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/rank_eval.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/reindex.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/reindex_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/render_search_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/rollup.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/scripts_painless_execute.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/scroll.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_application.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_mvt.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_shards.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/search_template.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/searchable_snapshots.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/security.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/shutdown.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/simulate.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/slm.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/snapshot.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/sql.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/ssl.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/streams.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/synonyms.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/tasks.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/terms_enum.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/termvectors.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/text_structure.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/transform.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update_by_query.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/update_by_query_rethrottle.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/watcher.d.ts","./node_modules/@elastic/elasticsearch/lib/api/api/xpack.d.ts","./node_modules/@elastic/elasticsearch/lib/api/index.d.ts","./node_modules/@elastic/elasticsearch/lib/client.d.ts","./node_modules/@elastic/elasticsearch/index.d.ts","./node_modules/@nestjs/elasticsearch/dist/interfaces/elasticsearch-module-options.interface.d.ts","./node_modules/@nestjs/elasticsearch/dist/elasticsearch.module.d.ts","./node_modules/@nestjs/elasticsearch/dist/elasticsearch.service.d.ts","./node_modules/@nestjs/elasticsearch/dist/interfaces/index.d.ts","./node_modules/@nestjs/elasticsearch/dist/index.d.ts","./node_modules/@nestjs/elasticsearch/index.d.ts","./src/search/autocomplete/autocomplete.service.ts","./src/search/filters/search-filters.service.ts","./src/search/search.service.ts","./src/search/search.controller.ts","./src/search/indexing/indexing.service.ts","./src/config/elasticsearch.config.ts","./src/search/search.module.ts","./src/email-marketing/entities/email-template.entity.ts","./src/email-marketing/entities/ab-test-variant.entity.ts","./src/email-marketing/enums/ab-test-status.enum.ts","./src/email-marketing/entities/ab-test.entity.ts","./src/email-marketing/enums/recipient-status.enum.ts","./src/email-marketing/entities/campaign-recipient.entity.ts","./src/email-marketing/enums/campaign-status.enum.ts","./src/email-marketing/entities/campaign.entity.ts","./src/email-marketing/enums/segment-rule-operator.enum.ts","./src/email-marketing/enums/segment-rule-field.enum.ts","./src/email-marketing/entities/segment-rule.entity.ts","./src/email-marketing/entities/segment.entity.ts","./src/email-marketing/dto/create-segment.dto.ts","./src/email-marketing/dto/update-segment.dto.ts","./src/email-marketing/segmentation/segmentation.service.ts","./src/email-marketing/dto/create-template.dto.ts","./src/email-marketing/dto/update-template.dto.ts","./src/email-marketing/templates/template-management.service.ts","./src/email-marketing/enums/email-event-type.enum.ts","./src/email-marketing/entities/email-event.entity.ts","./src/email-marketing/dto/create-ab-test.dto.ts","./src/email-marketing/ab-testing/ab-testing.service.ts","./src/email-marketing/analytics/email-analytics.service.ts","./src/email-marketing/dto/create-campaign.dto.ts","./src/email-marketing/dto/update-campaign.dto.ts","./src/email-marketing/dto/schedule-campaign.dto.ts","./src/email-marketing/email-marketing.service.ts","./src/email-marketing/enums/trigger-type.enum.ts","./src/email-marketing/entities/automation-trigger.entity.ts","./src/email-marketing/enums/action-type.enum.ts","./src/email-marketing/entities/automation-action.entity.ts","./src/email-marketing/enums/workflow-status.enum.ts","./src/email-marketing/entities/automation-workflow.entity.ts","./src/email-marketing/dto/create-automation.dto.ts","./src/email-marketing/dto/update-automation.dto.ts","./src/email-marketing/automation/automation.service.ts","./src/email-marketing/sender/email-sender.service.ts","./src/email-marketing/email-marketing.controller.ts","./src/email-marketing/templates/template.controller.ts","./src/email-marketing/automation/automation.controller.ts","./src/email-marketing/dto/add-segment-members.dto.ts","./src/email-marketing/segmentation/segment.controller.ts","./src/email-marketing/analytics/email-analytics.controller.ts","./src/email-marketing/ab-testing/ab-testing.controller.ts","./src/email-marketing/tracking/tracking.controller.ts","./src/email-marketing/entities/email-subscription.entity.ts","./src/email-marketing/processors/email-queue.processor.ts","./src/email-marketing/email-marketing.module.ts","./src/gamification/entities/point-transaction.entity.ts","./src/gamification/entities/user-progress.entity.ts","./src/gamification/entities/badge.entity.ts","./src/gamification/entities/user-badge.entity.ts","./src/gamification/entities/challenge.entity.ts","./src/gamification/entities/user-challenge.entity.ts","./src/gamification/points/points.service.ts","./src/gamification/badges/badges.service.ts","./src/gamification/leaderboards/leaderboards.service.ts","./src/gamification/challenges/challenges.service.ts","./src/gamification/gamification.service.ts","./src/gamification/gamification.controller.ts","./src/gamification/gamification.module.ts","./src/learning-paths/services/skill-assessment.service.ts","./src/learning-paths/services/path-generation.service.ts","./src/learning-paths/services/milestone-tracking.service.ts","./src/learning-paths/learning-paths.service.ts","./src/learning-paths/learning-paths.controller.ts","./src/learning-paths/learning-paths.module.ts","./node_modules/@huggingface/tasks/dist/commonjs/pipelines.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/audio-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/automatic-speech-recognition/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/chat-completion/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/document-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/feature-extraction/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/fill-mask/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-text/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-segmentation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-text-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/image-text-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/object-detection/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/depth-estimation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/sentence-similarity/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/summarization/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/table-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-image/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-video/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-to-speech/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/token-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/translation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/text-generation/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/video-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/visual-question-answering/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-image-classification/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/zero-shot-object-detection/inference.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tasks/index.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/widget-example.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/tokenizer-data.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-data.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-libraries-downloads.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/model-libraries.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/library-to-tasks.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/default-widget-inputs.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/gguf.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/common.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/types.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/inputs.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/snippets/index.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/hardware.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/local-apps.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/dataset-libraries.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/inference-providers.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/eval.d.ts","./node_modules/@huggingface/tasks/dist/commonjs/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/types.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/custom/request.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/custom/streamingRequest.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/utils.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/audioClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/audioToAudio.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/automaticSpeechRecognition.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/audio/textToSpeech.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/utils.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageSegmentation.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageToImage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageToText.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageToVideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageTextToImage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/imageTextToVideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/objectDetection.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/textToImage.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/textToVideo.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/cv/zeroShotImageClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/chatCompletion.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/chatCompletionStream.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/featureExtraction.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/fillMask.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/questionAnswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/sentenceSimilarity.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/summarization.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/tableQuestionAnswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textGeneration.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/textGenerationStream.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/tokenClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/translation.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/nlp/zeroShotClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/multimodal/documentQuestionAnswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/multimodal/visualQuestionAnswering.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/tabular/tabularClassification.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/tabular/tabularRegression.d.ts","./node_modules/@huggingface/inference/dist/commonjs/tasks/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/InferenceClient.d.ts","./node_modules/@huggingface/inference/dist/commonjs/vendor/type-fest/basic.d.ts","./node_modules/@huggingface/inference/dist/commonjs/errors.d.ts","./node_modules/@huggingface/inference/dist/commonjs/snippets/getInferenceSnippets.d.ts","./node_modules/@huggingface/inference/dist/commonjs/snippets/index.d.ts","./node_modules/@huggingface/inference/dist/commonjs/providers/providerHelper.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/getProviderHelper.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/makeRequestOptions.d.ts","./node_modules/@huggingface/inference/dist/commonjs/lib/logger.d.ts","./node_modules/@huggingface/inference/dist/commonjs/index.d.ts","./src/moderation/auto/auto-moderation.service.ts","./src/moderation/manual/review-item.entity.ts","./src/moderation/manual/manual-review.service.ts","./src/moderation/safety/content-safety.service.ts","./src/moderation/analytics/moderation-event.entity.ts","./src/moderation/analytics/moderation-analytics.service.ts","./src/moderation/moderation.service.ts","./src/moderation/moderation.module.ts","./node_modules/@nestjs/axios/dist/interfaces/http-module.interface.d.ts","./node_modules/@nestjs/axios/dist/interfaces/index.d.ts","./node_modules/@nestjs/axios/dist/http.module.d.ts","./node_modules/@nestjs/axios/dist/http.service.d.ts","./node_modules/@nestjs/axios/dist/index.d.ts","./node_modules/@nestjs/axios/index.d.ts","./src/orchestration/discovery/service-discovery.service.ts","./src/common/utils/correlation.utils.ts","./src/orchestration/service-mesh/service-mesh.service.ts","./src/orchestration/workflow/workflow-engine.service.ts","./src/orchestration/locks/distributed-lock.service.ts","./src/orchestration/health/health-checker.service.ts","./src/orchestration/orchestration.module.ts","./src/security/security.service.ts","./src/security/encryption/encryption.service.ts","./src/security/threats/threat-detection.service.ts","./src/security/compliance/compliance.service.ts","./src/security/audit/audit-logging.service.ts","./src/security/security.module.ts","./src/tenancy/entities/tenant.entity.ts","./src/tenancy/entities/tenant-config.entity.ts","./src/tenancy/entities/tenant-billing.entity.ts","./src/tenancy/entities/tenant-customization.entity.ts","./src/tenancy/dto/tenant.dto.ts","./src/tenancy/billing/tenant-billing.service.ts","./src/tenancy/customization/customization.service.ts","./src/tenancy/tenancy.service.ts","./src/tenancy/admin/tenant-admin.service.ts","./src/tenancy/tenancy.controller.ts","./src/tenancy/isolation/isolation.service.ts","./src/tenancy/decorators/requires-tenant.decorator.ts","./src/tenancy/guards/tenant.guard.ts","./src/tenancy/tenancy.module.ts","./src/cdn/optimization/asset-optimization.service.ts","./src/cdn/providers/cloudflare.service.ts","./src/cdn/caching/edge-caching.service.ts","./src/cdn/geo/geo-location.service.ts","./src/cdn/cdn.service.ts","./src/cdn/dto/upload-content.dto.ts","./src/cdn/cdn.controller.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/auth/httpAuthSchemeProvider.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/enums.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/models_0.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/AssociateAliasCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/AssociateDistributionTenantWebACLCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/AssociateDistributionWebACLCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CopyDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateAnycastIpListCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateCachePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateCloudFrontOriginAccessIdentityCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateConnectionGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateContinuousDeploymentPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateDistributionWithTagsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateFieldLevelEncryptionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateFieldLevelEncryptionProfileCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateInvalidationCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateInvalidationForDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateKeyGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateKeyValueStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateMonitoringSubscriptionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateOriginAccessControlCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateOriginRequestPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreatePublicKeyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateRealtimeLogConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateResponseHeadersPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateStreamingDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateStreamingDistributionWithTagsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateTrustStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/CreateVpcOriginCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteAnycastIpListCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteCachePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteCloudFrontOriginAccessIdentityCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteConnectionGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteContinuousDeploymentPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteFieldLevelEncryptionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteFieldLevelEncryptionProfileCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteKeyGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteKeyValueStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteMonitoringSubscriptionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteOriginAccessControlCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteOriginRequestPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeletePublicKeyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteRealtimeLogConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteResourcePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteResponseHeadersPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteStreamingDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteTrustStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DeleteVpcOriginCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DescribeConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DescribeFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DescribeKeyValueStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DisassociateDistributionTenantWebACLCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/DisassociateDistributionWebACLCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetAnycastIpListCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetCachePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetCachePolicyConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetCloudFrontOriginAccessIdentityCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetCloudFrontOriginAccessIdentityConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetConnectionGroupByRoutingEndpointCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetConnectionGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetContinuousDeploymentPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetContinuousDeploymentPolicyConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetDistributionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetDistributionTenantByDomainCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetFieldLevelEncryptionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetFieldLevelEncryptionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetFieldLevelEncryptionProfileCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetFieldLevelEncryptionProfileConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetInvalidationCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetInvalidationForDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetKeyGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetKeyGroupConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/models_1.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetManagedCertificateDetailsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetMonitoringSubscriptionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetOriginAccessControlCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetOriginAccessControlConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetOriginRequestPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetOriginRequestPolicyConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetPublicKeyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetPublicKeyConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetRealtimeLogConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetResourcePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetResponseHeadersPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetResponseHeadersPolicyConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetStreamingDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetStreamingDistributionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetTrustStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/GetVpcOriginCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListAnycastIpListsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListCachePoliciesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListCloudFrontOriginAccessIdentitiesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListConflictingAliasesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListConnectionFunctionsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListConnectionGroupsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListContinuousDeploymentPoliciesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByAnycastIpListIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByCachePolicyIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByConnectionModeCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByKeyGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByOriginRequestPolicyIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByOwnedResourceCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByRealtimeLogConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByResponseHeadersPolicyIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByTrustStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByVpcOriginIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsByWebACLIdCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionTenantsByCustomizationCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDistributionTenantsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListDomainConflictsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListFieldLevelEncryptionConfigsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListFieldLevelEncryptionProfilesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListFunctionsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListInvalidationsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListInvalidationsForDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListKeyGroupsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListKeyValueStoresCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListOriginAccessControlsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListOriginRequestPoliciesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListPublicKeysCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListRealtimeLogConfigsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListResponseHeadersPoliciesCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListStreamingDistributionsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListTagsForResourceCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListTrustStoresCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/ListVpcOriginsCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/PublishConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/PublishFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/PutResourcePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/TagResourceCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/TestConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/TestFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UntagResourceCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateAnycastIpListCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateCachePolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateCloudFrontOriginAccessIdentityCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateConnectionFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateConnectionGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateContinuousDeploymentPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateDistributionTenantCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateDistributionWithStagingConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateDomainAssociationCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateFieldLevelEncryptionConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateFieldLevelEncryptionProfileCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateFunctionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateKeyGroupCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateKeyValueStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateOriginAccessControlCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateOriginRequestPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdatePublicKeyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateRealtimeLogConfigCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateResponseHeadersPolicyCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateStreamingDistributionCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateTrustStoreCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/UpdateVpcOriginCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/VerifyDnsConfigurationCommand.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/endpoint/EndpointParameters.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/auth/httpAuthExtensionConfiguration.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/extensionConfiguration.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/runtimeExtensions.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/CloudFrontClient.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/CloudFront.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/commands/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/schemas/schemas_0.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/Interfaces.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListCloudFrontOriginAccessIdentitiesPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListConnectionFunctionsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListConnectionGroupsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionsByConnectionFunctionPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionsByConnectionModePaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionsByTrustStorePaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionTenantsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDistributionTenantsByCustomizationPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListDomainConflictsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListInvalidationsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListInvalidationsForDistributionTenantPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListKeyValueStoresPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListOriginAccessControlsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListPublicKeysPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListStreamingDistributionsPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/ListTrustStoresPaginator.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/pagination/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitForDistributionDeployed.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitForInvalidationCompleted.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitForInvalidationForDistributionTenantCompleted.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/waitForStreamingDistributionDeployed.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/waiters/index.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/CloudFrontServiceException.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/models/errors.d.ts","./node_modules/@aws-sdk/client-cloudfront/dist-types/index.d.ts","./src/cdn/providers/aws-cloudfront.service.ts","./src/cdn/cdn.module.ts","./src/payments/entities/payment.entity.ts","./src/payments/entities/subscription.entity.ts","./src/payments/dto/create-payment.dto.ts","./src/payments/entities/refund.entity.ts","./src/payments/entities/invoice.entity.ts","./src/payments/dto/refund.dto.ts","./src/payments/dto/create-subscription.dto.ts","./node_modules/stripe/types/lib.d.ts","./node_modules/stripe/types/crypto/crypto.d.ts","./node_modules/stripe/types/net/net.d.ts","./node_modules/stripe/types/shared.d.ts","./node_modules/stripe/types/Errors.d.ts","./node_modules/stripe/types/OAuth.d.ts","./node_modules/stripe/types/Webhooks.d.ts","./node_modules/stripe/types/EventTypes.d.ts","./node_modules/stripe/types/UpcomingInvoices.d.ts","./node_modules/stripe/types/ThinEvent.d.ts","./node_modules/stripe/types/AccountLinksResource.d.ts","./node_modules/stripe/types/AccountSessionsResource.d.ts","./node_modules/stripe/types/AccountsResource.d.ts","./node_modules/stripe/types/ApplePayDomainsResource.d.ts","./node_modules/stripe/types/ApplicationFeesResource.d.ts","./node_modules/stripe/types/Apps/SecretsResource.d.ts","./node_modules/stripe/types/BalanceResource.d.ts","./node_modules/stripe/types/BalanceTransactionsResource.d.ts","./node_modules/stripe/types/Billing/AlertsResource.d.ts","./node_modules/stripe/types/Billing/CreditBalanceSummaryResource.d.ts","./node_modules/stripe/types/Billing/CreditBalanceTransactionsResource.d.ts","./node_modules/stripe/types/Billing/CreditGrantsResource.d.ts","./node_modules/stripe/types/Billing/MeterEventAdjustmentsResource.d.ts","./node_modules/stripe/types/Billing/MeterEventsResource.d.ts","./node_modules/stripe/types/Billing/MetersResource.d.ts","./node_modules/stripe/types/BillingPortal/ConfigurationsResource.d.ts","./node_modules/stripe/types/BillingPortal/SessionsResource.d.ts","./node_modules/stripe/types/ChargesResource.d.ts","./node_modules/stripe/types/Checkout/SessionsResource.d.ts","./node_modules/stripe/types/Climate/OrdersResource.d.ts","./node_modules/stripe/types/Climate/ProductsResource.d.ts","./node_modules/stripe/types/Climate/SuppliersResource.d.ts","./node_modules/stripe/types/ConfirmationTokensResource.d.ts","./node_modules/stripe/types/CountrySpecsResource.d.ts","./node_modules/stripe/types/CouponsResource.d.ts","./node_modules/stripe/types/CreditNotesResource.d.ts","./node_modules/stripe/types/CustomerSessionsResource.d.ts","./node_modules/stripe/types/CustomersResource.d.ts","./node_modules/stripe/types/DisputesResource.d.ts","./node_modules/stripe/types/Entitlements/ActiveEntitlementsResource.d.ts","./node_modules/stripe/types/Entitlements/FeaturesResource.d.ts","./node_modules/stripe/types/EphemeralKeysResource.d.ts","./node_modules/stripe/types/EventsResource.d.ts","./node_modules/stripe/types/ExchangeRatesResource.d.ts","./node_modules/stripe/types/FileLinksResource.d.ts","./node_modules/stripe/types/FilesResource.d.ts","./node_modules/stripe/types/FinancialConnections/AccountsResource.d.ts","./node_modules/stripe/types/FinancialConnections/SessionsResource.d.ts","./node_modules/stripe/types/FinancialConnections/TransactionsResource.d.ts","./node_modules/stripe/types/Forwarding/RequestsResource.d.ts","./node_modules/stripe/types/Identity/VerificationReportsResource.d.ts","./node_modules/stripe/types/Identity/VerificationSessionsResource.d.ts","./node_modules/stripe/types/InvoiceItemsResource.d.ts","./node_modules/stripe/types/InvoicePaymentsResource.d.ts","./node_modules/stripe/types/InvoiceRenderingTemplatesResource.d.ts","./node_modules/stripe/types/InvoicesResource.d.ts","./node_modules/stripe/types/Issuing/AuthorizationsResource.d.ts","./node_modules/stripe/types/Issuing/CardholdersResource.d.ts","./node_modules/stripe/types/Issuing/CardsResource.d.ts","./node_modules/stripe/types/Issuing/DisputesResource.d.ts","./node_modules/stripe/types/Issuing/PersonalizationDesignsResource.d.ts","./node_modules/stripe/types/Issuing/PhysicalBundlesResource.d.ts","./node_modules/stripe/types/Issuing/TokensResource.d.ts","./node_modules/stripe/types/Issuing/TransactionsResource.d.ts","./node_modules/stripe/types/MandatesResource.d.ts","./node_modules/stripe/types/PaymentIntentsResource.d.ts","./node_modules/stripe/types/PaymentLinksResource.d.ts","./node_modules/stripe/types/PaymentMethodConfigurationsResource.d.ts","./node_modules/stripe/types/PaymentMethodDomainsResource.d.ts","./node_modules/stripe/types/PaymentMethodsResource.d.ts","./node_modules/stripe/types/PayoutsResource.d.ts","./node_modules/stripe/types/PlansResource.d.ts","./node_modules/stripe/types/PricesResource.d.ts","./node_modules/stripe/types/ProductsResource.d.ts","./node_modules/stripe/types/PromotionCodesResource.d.ts","./node_modules/stripe/types/QuotesResource.d.ts","./node_modules/stripe/types/Radar/EarlyFraudWarningsResource.d.ts","./node_modules/stripe/types/Radar/ValueListItemsResource.d.ts","./node_modules/stripe/types/Radar/ValueListsResource.d.ts","./node_modules/stripe/types/RefundsResource.d.ts","./node_modules/stripe/types/Reporting/ReportRunsResource.d.ts","./node_modules/stripe/types/Reporting/ReportTypesResource.d.ts","./node_modules/stripe/types/ReviewsResource.d.ts","./node_modules/stripe/types/SetupAttemptsResource.d.ts","./node_modules/stripe/types/SetupIntentsResource.d.ts","./node_modules/stripe/types/ShippingRatesResource.d.ts","./node_modules/stripe/types/Sigma/ScheduledQueryRunsResource.d.ts","./node_modules/stripe/types/SourcesResource.d.ts","./node_modules/stripe/types/SubscriptionItemsResource.d.ts","./node_modules/stripe/types/SubscriptionSchedulesResource.d.ts","./node_modules/stripe/types/SubscriptionsResource.d.ts","./node_modules/stripe/types/Tax/CalculationsResource.d.ts","./node_modules/stripe/types/Tax/RegistrationsResource.d.ts","./node_modules/stripe/types/Tax/SettingsResource.d.ts","./node_modules/stripe/types/Tax/TransactionsResource.d.ts","./node_modules/stripe/types/TaxCodesResource.d.ts","./node_modules/stripe/types/TaxIdsResource.d.ts","./node_modules/stripe/types/TaxRatesResource.d.ts","./node_modules/stripe/types/Terminal/ConfigurationsResource.d.ts","./node_modules/stripe/types/Terminal/ConnectionTokensResource.d.ts","./node_modules/stripe/types/Terminal/LocationsResource.d.ts","./node_modules/stripe/types/Terminal/ReadersResource.d.ts","./node_modules/stripe/types/TestHelpers/ConfirmationTokensResource.d.ts","./node_modules/stripe/types/TestHelpers/CustomersResource.d.ts","./node_modules/stripe/types/TestHelpers/Issuing/AuthorizationsResource.d.ts","./node_modules/stripe/types/TestHelpers/Issuing/CardsResource.d.ts","./node_modules/stripe/types/TestHelpers/Issuing/PersonalizationDesignsResource.d.ts","./node_modules/stripe/types/TestHelpers/Issuing/TransactionsResource.d.ts","./node_modules/stripe/types/TestHelpers/RefundsResource.d.ts","./node_modules/stripe/types/TestHelpers/Terminal/ReadersResource.d.ts","./node_modules/stripe/types/TestHelpers/TestClocksResource.d.ts","./node_modules/stripe/types/TestHelpers/Treasury/InboundTransfersResource.d.ts","./node_modules/stripe/types/TestHelpers/Treasury/OutboundPaymentsResource.d.ts","./node_modules/stripe/types/TestHelpers/Treasury/OutboundTransfersResource.d.ts","./node_modules/stripe/types/TestHelpers/Treasury/ReceivedCreditsResource.d.ts","./node_modules/stripe/types/TestHelpers/Treasury/ReceivedDebitsResource.d.ts","./node_modules/stripe/types/TokensResource.d.ts","./node_modules/stripe/types/TopupsResource.d.ts","./node_modules/stripe/types/TransfersResource.d.ts","./node_modules/stripe/types/Treasury/CreditReversalsResource.d.ts","./node_modules/stripe/types/Treasury/DebitReversalsResource.d.ts","./node_modules/stripe/types/Treasury/FinancialAccountsResource.d.ts","./node_modules/stripe/types/Treasury/InboundTransfersResource.d.ts","./node_modules/stripe/types/Treasury/OutboundPaymentsResource.d.ts","./node_modules/stripe/types/Treasury/OutboundTransfersResource.d.ts","./node_modules/stripe/types/Treasury/ReceivedCreditsResource.d.ts","./node_modules/stripe/types/Treasury/ReceivedDebitsResource.d.ts","./node_modules/stripe/types/Treasury/TransactionEntriesResource.d.ts","./node_modules/stripe/types/Treasury/TransactionsResource.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventAdjustmentsResource.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventSessionResource.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventStreamResource.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventsResource.d.ts","./node_modules/stripe/types/V2/Core/EventDestinationsResource.d.ts","./node_modules/stripe/types/V2/EventTypes.d.ts","./node_modules/stripe/types/V2/Core/EventsResource.d.ts","./node_modules/stripe/types/WebhookEndpointsResource.d.ts","./node_modules/stripe/types/AccountLinks.d.ts","./node_modules/stripe/types/AccountSessions.d.ts","./node_modules/stripe/types/Accounts.d.ts","./node_modules/stripe/types/ApplePayDomains.d.ts","./node_modules/stripe/types/ApplicationFees.d.ts","./node_modules/stripe/types/Applications.d.ts","./node_modules/stripe/types/Apps/Secrets.d.ts","./node_modules/stripe/types/Balance.d.ts","./node_modules/stripe/types/BalanceTransactionSources.d.ts","./node_modules/stripe/types/BalanceTransactions.d.ts","./node_modules/stripe/types/BankAccounts.d.ts","./node_modules/stripe/types/Billing/AlertTriggereds.d.ts","./node_modules/stripe/types/Billing/Alerts.d.ts","./node_modules/stripe/types/Billing/CreditBalanceSummary.d.ts","./node_modules/stripe/types/Billing/CreditBalanceTransactions.d.ts","./node_modules/stripe/types/Billing/CreditGrants.d.ts","./node_modules/stripe/types/Billing/MeterEventAdjustments.d.ts","./node_modules/stripe/types/Billing/MeterEventSummaries.d.ts","./node_modules/stripe/types/Billing/MeterEvents.d.ts","./node_modules/stripe/types/Billing/Meters.d.ts","./node_modules/stripe/types/BillingPortal/Configurations.d.ts","./node_modules/stripe/types/BillingPortal/Sessions.d.ts","./node_modules/stripe/types/Capabilities.d.ts","./node_modules/stripe/types/Cards.d.ts","./node_modules/stripe/types/CashBalances.d.ts","./node_modules/stripe/types/Charges.d.ts","./node_modules/stripe/types/Checkout/Sessions.d.ts","./node_modules/stripe/types/Climate/Orders.d.ts","./node_modules/stripe/types/Climate/Products.d.ts","./node_modules/stripe/types/Climate/Suppliers.d.ts","./node_modules/stripe/types/ConfirmationTokens.d.ts","./node_modules/stripe/types/ConnectCollectionTransfers.d.ts","./node_modules/stripe/types/CountrySpecs.d.ts","./node_modules/stripe/types/Coupons.d.ts","./node_modules/stripe/types/CreditNoteLineItems.d.ts","./node_modules/stripe/types/CreditNotes.d.ts","./node_modules/stripe/types/CustomerBalanceTransactions.d.ts","./node_modules/stripe/types/CustomerCashBalanceTransactions.d.ts","./node_modules/stripe/types/CustomerSessions.d.ts","./node_modules/stripe/types/CustomerSources.d.ts","./node_modules/stripe/types/Customers.d.ts","./node_modules/stripe/types/Discounts.d.ts","./node_modules/stripe/types/Disputes.d.ts","./node_modules/stripe/types/Entitlements/ActiveEntitlementSummaries.d.ts","./node_modules/stripe/types/Entitlements/ActiveEntitlements.d.ts","./node_modules/stripe/types/Entitlements/Features.d.ts","./node_modules/stripe/types/EphemeralKeys.d.ts","./node_modules/stripe/types/Events.d.ts","./node_modules/stripe/types/ExchangeRates.d.ts","./node_modules/stripe/types/ExternalAccounts.d.ts","./node_modules/stripe/types/FeeRefunds.d.ts","./node_modules/stripe/types/FileLinks.d.ts","./node_modules/stripe/types/Files.d.ts","./node_modules/stripe/types/FinancialConnections/AccountOwners.d.ts","./node_modules/stripe/types/FinancialConnections/AccountOwnerships.d.ts","./node_modules/stripe/types/FinancialConnections/Accounts.d.ts","./node_modules/stripe/types/FinancialConnections/Sessions.d.ts","./node_modules/stripe/types/FinancialConnections/Transactions.d.ts","./node_modules/stripe/types/Forwarding/Requests.d.ts","./node_modules/stripe/types/FundingInstructions.d.ts","./node_modules/stripe/types/Identity/VerificationReports.d.ts","./node_modules/stripe/types/Identity/VerificationSessions.d.ts","./node_modules/stripe/types/InvoiceItems.d.ts","./node_modules/stripe/types/InvoiceLineItems.d.ts","./node_modules/stripe/types/InvoicePayments.d.ts","./node_modules/stripe/types/InvoiceRenderingTemplates.d.ts","./node_modules/stripe/types/Invoices.d.ts","./node_modules/stripe/types/Issuing/Authorizations.d.ts","./node_modules/stripe/types/Issuing/Cardholders.d.ts","./node_modules/stripe/types/Issuing/Cards.d.ts","./node_modules/stripe/types/Issuing/Disputes.d.ts","./node_modules/stripe/types/Issuing/PersonalizationDesigns.d.ts","./node_modules/stripe/types/Issuing/PhysicalBundles.d.ts","./node_modules/stripe/types/Issuing/Tokens.d.ts","./node_modules/stripe/types/Issuing/Transactions.d.ts","./node_modules/stripe/types/LineItems.d.ts","./node_modules/stripe/types/LoginLinks.d.ts","./node_modules/stripe/types/Mandates.d.ts","./node_modules/stripe/types/PaymentIntents.d.ts","./node_modules/stripe/types/PaymentLinks.d.ts","./node_modules/stripe/types/PaymentMethodConfigurations.d.ts","./node_modules/stripe/types/PaymentMethodDomains.d.ts","./node_modules/stripe/types/PaymentMethods.d.ts","./node_modules/stripe/types/Payouts.d.ts","./node_modules/stripe/types/Persons.d.ts","./node_modules/stripe/types/Plans.d.ts","./node_modules/stripe/types/Prices.d.ts","./node_modules/stripe/types/ProductFeatures.d.ts","./node_modules/stripe/types/Products.d.ts","./node_modules/stripe/types/PromotionCodes.d.ts","./node_modules/stripe/types/Quotes.d.ts","./node_modules/stripe/types/Radar/EarlyFraudWarnings.d.ts","./node_modules/stripe/types/Radar/ValueListItems.d.ts","./node_modules/stripe/types/Radar/ValueLists.d.ts","./node_modules/stripe/types/Refunds.d.ts","./node_modules/stripe/types/Reporting/ReportRuns.d.ts","./node_modules/stripe/types/Reporting/ReportTypes.d.ts","./node_modules/stripe/types/ReserveTransactions.d.ts","./node_modules/stripe/types/Reviews.d.ts","./node_modules/stripe/types/SetupAttempts.d.ts","./node_modules/stripe/types/SetupIntents.d.ts","./node_modules/stripe/types/ShippingRates.d.ts","./node_modules/stripe/types/Sigma/ScheduledQueryRuns.d.ts","./node_modules/stripe/types/SourceMandateNotifications.d.ts","./node_modules/stripe/types/SourceTransactions.d.ts","./node_modules/stripe/types/Sources.d.ts","./node_modules/stripe/types/SubscriptionItems.d.ts","./node_modules/stripe/types/SubscriptionSchedules.d.ts","./node_modules/stripe/types/Subscriptions.d.ts","./node_modules/stripe/types/Tax/CalculationLineItems.d.ts","./node_modules/stripe/types/Tax/Calculations.d.ts","./node_modules/stripe/types/Tax/Registrations.d.ts","./node_modules/stripe/types/Tax/Settings.d.ts","./node_modules/stripe/types/Tax/TransactionLineItems.d.ts","./node_modules/stripe/types/Tax/Transactions.d.ts","./node_modules/stripe/types/TaxCodes.d.ts","./node_modules/stripe/types/TaxDeductedAtSources.d.ts","./node_modules/stripe/types/TaxIds.d.ts","./node_modules/stripe/types/TaxRates.d.ts","./node_modules/stripe/types/Terminal/Configurations.d.ts","./node_modules/stripe/types/Terminal/ConnectionTokens.d.ts","./node_modules/stripe/types/Terminal/Locations.d.ts","./node_modules/stripe/types/Terminal/Readers.d.ts","./node_modules/stripe/types/TestHelpers/TestClocks.d.ts","./node_modules/stripe/types/Tokens.d.ts","./node_modules/stripe/types/Topups.d.ts","./node_modules/stripe/types/TransferReversals.d.ts","./node_modules/stripe/types/Transfers.d.ts","./node_modules/stripe/types/Treasury/CreditReversals.d.ts","./node_modules/stripe/types/Treasury/DebitReversals.d.ts","./node_modules/stripe/types/Treasury/FinancialAccountFeatures.d.ts","./node_modules/stripe/types/Treasury/FinancialAccounts.d.ts","./node_modules/stripe/types/Treasury/InboundTransfers.d.ts","./node_modules/stripe/types/Treasury/OutboundPayments.d.ts","./node_modules/stripe/types/Treasury/OutboundTransfers.d.ts","./node_modules/stripe/types/Treasury/ReceivedCredits.d.ts","./node_modules/stripe/types/Treasury/ReceivedDebits.d.ts","./node_modules/stripe/types/Treasury/TransactionEntries.d.ts","./node_modules/stripe/types/Treasury/Transactions.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventAdjustments.d.ts","./node_modules/stripe/types/V2/Billing/MeterEventSessions.d.ts","./node_modules/stripe/types/V2/Billing/MeterEvents.d.ts","./node_modules/stripe/types/V2/EventDestinations.d.ts","./node_modules/stripe/types/V2/Events.d.ts","./node_modules/stripe/types/WebhookEndpoints.d.ts","./node_modules/stripe/types/index.d.ts","./src/payments/providers/stripe.service.ts","./src/payments/providers/provider-factory.service.ts","./src/payments/interfaces/payment-provider.interface.ts","./src/payments/payments.service.ts","./src/payments/payments.controller.ts","./src/payments/webhooks/entities/webhook-retry.entity.ts","./src/payments/webhooks/webhook-queue.service.ts","./src/payments/webhooks/webhook.service.ts","./src/payments/webhooks/webhook.controller.ts","./src/payments/webhooks/webhook-management.controller.ts","./src/payments/webhooks/webhook-retry.processor.ts","./src/payments/subscriptions/subscriptions.service.ts","./src/payments/subscriptions/subscription-job.processor.ts","./src/common/database/transaction-helper.service.ts","./src/payments/payments.module.ts","./src/localization/entities/translation.entity.ts","./src/localization/language-detection.service.ts","./src/common/types/request-with-locale.ts","./src/localization/language.middleware.ts","./src/localization/dto/bundle-query.dto.ts","./src/localization/dto/create-translation.dto.ts","./src/localization/dto/export-query.dto.ts","./src/localization/dto/import-translations.dto.ts","./src/localization/dto/list-translations-query.dto.ts","./src/localization/dto/update-translation.dto.ts","./src/localization/localization.constants.ts","./src/localization/localization.service.ts","./src/localization/localization.controller.ts","./src/localization/localization.module.ts","./src/common/csrf/csrf.service.ts","./src/common/middleware/csrf.middleware.ts","./src/common/csrf/csrf.controller.ts","./src/common/csrf/csrf.module.ts","./src/common/timeout/timeout.controller.ts","./src/common/examples/timeout-example.controller.ts","./src/common/timeout/timeout.module.ts","./src/app.module.ts","./node_modules/@types/express-session/index.d.ts","./node_modules/@redis/client/dist/lib/commands/generic-transformers.d.ts","./node_modules/@redis/client/dist/lib/client/parser.d.ts","./node_modules/@redis/client/dist/lib/errors.d.ts","./node_modules/@redis/client/dist/lib/lua-script.d.ts","./node_modules/@redis/client/dist/lib/RESP/decoder.d.ts","./node_modules/@redis/client/dist/lib/RESP/verbatim-string.d.ts","./node_modules/@redis/client/dist/lib/RESP/types.d.ts","./node_modules/@redis/client/dist/lib/utils/digest.d.ts","./node_modules/@redis/client/dist/lib/commands/CLIENT_KILL.d.ts","./node_modules/@redis/client/dist/lib/commands/CLUSTER_FAILOVER.d.ts","./node_modules/@redis/client/dist/lib/commands/CLUSTER_SETSLOT.d.ts","./node_modules/@redis/client/dist/lib/commands/COMMAND_LIST.d.ts","./node_modules/@redis/client/dist/lib/commands/FLUSHALL.d.ts","./node_modules/@redis/client/dist/lib/commands/SET.d.ts","./node_modules/@redis/client/dist/lib/commands/ACL_LOG.d.ts","./node_modules/@redis/client/dist/lib/commands/AUTH.d.ts","./node_modules/@redis/client/dist/lib/commands/BGSAVE.d.ts","./node_modules/@redis/client/dist/lib/commands/BITCOUNT.d.ts","./node_modules/@redis/client/dist/lib/commands/BITFIELD.d.ts","./node_modules/@redis/client/dist/lib/commands/BITFIELD_RO.d.ts","./node_modules/@redis/client/dist/lib/commands/BITOP.d.ts","./node_modules/@redis/client/dist/lib/commands/LMPOP.d.ts","./node_modules/@redis/client/dist/lib/commands/ZMPOP.d.ts","./node_modules/@redis/client/dist/lib/commands/CLIENT_INFO.d.ts","./node_modules/@redis/client/dist/lib/commands/CLIENT_LIST.d.ts","./node_modules/@redis/client/dist/lib/commands/CLIENT_TRACKING.d.ts","./node_modules/@redis/client/dist/lib/commands/CLUSTER_RESET.d.ts","./node_modules/@redis/client/dist/lib/commands/COPY.d.ts","./node_modules/@redis/client/dist/lib/commands/EVAL.d.ts","./node_modules/@redis/client/dist/lib/commands/FUNCTION_LIST.d.ts","./node_modules/@redis/client/dist/lib/commands/FUNCTION_LIST_WITHCODE.d.ts","./node_modules/@redis/client/dist/lib/commands/FUNCTION_LOAD.d.ts","./node_modules/@redis/client/dist/lib/commands/FUNCTION_RESTORE.d.ts","./node_modules/@redis/client/dist/lib/commands/GEOSEARCH.d.ts","./node_modules/@redis/client/dist/lib/commands/GEOADD.d.ts","./node_modules/@redis/client/dist/lib/commands/GEOSEARCH_WITH.d.ts","./node_modules/@redis/client/dist/lib/commands/GEORADIUS_STORE.d.ts","./node_modules/@redis/client/dist/lib/commands/GEORADIUSBYMEMBER_STORE.d.ts","./node_modules/@redis/client/dist/lib/commands/GEOSEARCHSTORE.d.ts","./node_modules/@redis/client/dist/lib/commands/GETEX.d.ts","./node_modules/@redis/client/dist/lib/commands/HELLO.d.ts","./node_modules/@redis/client/dist/lib/commands/HEXPIRE.d.ts","./node_modules/@redis/client/dist/lib/commands/HGETEX.d.ts","./node_modules/@redis/client/dist/lib/commands/HRANDFIELD_COUNT_WITHVALUES.d.ts","./node_modules/@redis/client/dist/lib/commands/SCAN.d.ts","./node_modules/@redis/client/dist/lib/commands/HSET.d.ts","./node_modules/@redis/client/dist/lib/commands/HSETEX.d.ts","./node_modules/@redis/client/dist/lib/commands/HOTKEYS_GET.d.ts","./node_modules/@redis/client/dist/lib/commands/HOTKEYS_START.d.ts","./node_modules/@redis/client/dist/lib/commands/LATENCY_GRAPH.d.ts","./node_modules/@redis/client/dist/lib/commands/LATENCY_HISTORY.d.ts","./node_modules/@redis/client/dist/lib/commands/LCS_IDX.d.ts","./node_modules/@redis/client/dist/lib/commands/LCS_IDX_WITHMATCHLEN.d.ts","./node_modules/@redis/client/dist/lib/commands/LPOS.d.ts","./node_modules/@redis/client/dist/lib/commands/MEMORY_STATS.d.ts","./node_modules/@redis/client/dist/lib/commands/MEMORY_USAGE.d.ts","./node_modules/@redis/client/dist/lib/commands/MIGRATE.d.ts","./node_modules/@redis/client/dist/lib/commands/MODULE_LIST.d.ts","./node_modules/@redis/client/dist/lib/commands/MSET.d.ts","./node_modules/@redis/client/dist/lib/commands/RESTORE.d.ts","./node_modules/@redis/client/dist/lib/commands/SINTERCARD.d.ts","./node_modules/@redis/client/dist/lib/commands/SORT.d.ts","./node_modules/@redis/client/dist/lib/commands/common-stream.types.d.ts","./node_modules/@redis/client/dist/lib/commands/XADD.d.ts","./node_modules/@redis/client/dist/lib/commands/XAUTOCLAIM.d.ts","./node_modules/@redis/client/dist/lib/commands/XCLAIM.d.ts","./node_modules/@redis/client/dist/lib/commands/XCFGSET.d.ts","./node_modules/@redis/client/dist/lib/commands/XGROUP_CREATE.d.ts","./node_modules/@redis/client/dist/lib/commands/XGROUP_SETID.d.ts","./node_modules/@redis/client/dist/lib/commands/XINFO_CONSUMERS.d.ts","./node_modules/@redis/client/dist/lib/commands/XINFO_GROUPS.d.ts","./node_modules/@redis/client/dist/lib/commands/XINFO_STREAM.d.ts","./node_modules/@redis/client/dist/lib/commands/XPENDING_RANGE.d.ts","./node_modules/@redis/client/dist/lib/commands/XRANGE.d.ts","./node_modules/@redis/client/dist/lib/commands/XREAD.d.ts","./node_modules/@redis/client/dist/lib/commands/XREADGROUP.d.ts","./node_modules/@redis/client/dist/lib/commands/XSETID.d.ts","./node_modules/@redis/client/dist/lib/commands/XTRIM.d.ts","./node_modules/@redis/client/dist/lib/commands/ZADD_INCR.d.ts","./node_modules/@redis/client/dist/lib/commands/ZADD.d.ts","./node_modules/@redis/client/dist/lib/commands/ZINTER.d.ts","./node_modules/@redis/client/dist/lib/commands/ZINTERCARD.d.ts","./node_modules/@redis/client/dist/lib/commands/ZRANGE.d.ts","./node_modules/@redis/client/dist/lib/commands/ZRANGEBYLEX.d.ts","./node_modules/@redis/client/dist/lib/commands/ZRANGEBYSCORE.d.ts","./node_modules/@redis/client/dist/lib/commands/ZRANGESTORE.d.ts","./node_modules/@redis/client/dist/lib/commands/ZUNION.d.ts","./node_modules/@redis/client/dist/lib/commands/ZUNIONSTORE.d.ts","./node_modules/@redis/client/dist/lib/commands/VADD.d.ts","./node_modules/@redis/client/dist/lib/commands/VINFO.d.ts","./node_modules/@redis/client/dist/lib/commands/VSIM.d.ts","./node_modules/@redis/client/dist/lib/commands/index.d.ts","./node_modules/@redis/client/dist/lib/client/socket.d.ts","./node_modules/@redis/client/dist/lib/authx/identity-provider.d.ts","./node_modules/@redis/client/dist/lib/authx/token.d.ts","./node_modules/@redis/client/dist/lib/authx/disposable.d.ts","./node_modules/@redis/client/dist/lib/authx/token-manager.d.ts","./node_modules/@redis/client/dist/lib/authx/credentials-provider.d.ts","./node_modules/@redis/client/dist/lib/authx/index.d.ts","./node_modules/@redis/client/dist/lib/client/pub-sub.d.ts","./node_modules/@redis/client/dist/lib/client/commands-queue.d.ts","./node_modules/@redis/client/dist/lib/multi-command.d.ts","./node_modules/@redis/client/dist/lib/client/multi-command.d.ts","./node_modules/@redis/client/dist/lib/client/legacy-mode.d.ts","./node_modules/@redis/client/dist/lib/client/cache.d.ts","./node_modules/@redis/client/dist/lib/cluster/cluster-slots.d.ts","./node_modules/@redis/client/dist/lib/cluster/multi-command.d.ts","./node_modules/@redis/client/dist/lib/cluster/index.d.ts","./node_modules/@redis/client/dist/lib/client/identity.d.ts","./node_modules/@redis/client/dist/lib/client/pool.d.ts","./node_modules/@redis/client/dist/lib/client/enterprise-maintenance-manager.d.ts","./node_modules/@redis/client/dist/lib/client/index.d.ts","./node_modules/@redis/client/dist/lib/sentinel/types.d.ts","./node_modules/@redis/client/dist/lib/sentinel/multi-commands.d.ts","./node_modules/@redis/client/dist/lib/sentinel/index.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/types.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/metrics.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/client-registry.d.ts","./node_modules/@redis/client/dist/lib/opentelemetry/index.d.ts","./node_modules/@redis/client/dist/lib/client/tracing.d.ts","./node_modules/@redis/client/dist/index.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/INFO.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/INSERT.d.ts","./node_modules/@redis/bloom/dist/lib/commands/bloom/RESERVE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/INCRBY.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/INFO.d.ts","./node_modules/@redis/bloom/dist/lib/commands/count-min-sketch/MERGE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/INFO.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/INSERT.d.ts","./node_modules/@redis/bloom/dist/lib/commands/cuckoo/RESERVE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/CREATE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/INFO.d.ts","./node_modules/@redis/bloom/dist/lib/commands/t-digest/MERGE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/INCRBY.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/INFO.d.ts","./node_modules/@redis/bloom/dist/lib/commands/top-k/RESERVE.d.ts","./node_modules/@redis/bloom/dist/lib/commands/index.d.ts","./node_modules/@redis/bloom/dist/lib/index.d.ts","./node_modules/@redis/json/dist/lib/commands/ARRINDEX.d.ts","./node_modules/@redis/json/dist/lib/commands/ARRLEN.d.ts","./node_modules/@redis/json/dist/lib/commands/ARRPOP.d.ts","./node_modules/@redis/json/dist/lib/commands/CLEAR.d.ts","./node_modules/@redis/json/dist/lib/commands/DEBUG_MEMORY.d.ts","./node_modules/@redis/json/dist/lib/commands/DEL.d.ts","./node_modules/@redis/json/dist/lib/commands/FORGET.d.ts","./node_modules/@redis/json/dist/lib/commands/GET.d.ts","./node_modules/@redis/json/dist/lib/commands/MSET.d.ts","./node_modules/@redis/json/dist/lib/commands/OBJKEYS.d.ts","./node_modules/@redis/json/dist/lib/commands/OBJLEN.d.ts","./node_modules/@redis/json/dist/lib/commands/SET.d.ts","./node_modules/@redis/json/dist/lib/commands/STRAPPEND.d.ts","./node_modules/@redis/json/dist/lib/commands/STRLEN.d.ts","./node_modules/@redis/json/dist/lib/commands/TYPE.d.ts","./node_modules/@redis/json/dist/lib/commands/index.d.ts","./node_modules/@redis/json/dist/lib/index.d.ts","./node_modules/@redis/search/dist/lib/commands/CREATE.d.ts","./node_modules/@redis/search/dist/lib/commands/SEARCH.d.ts","./node_modules/@redis/search/dist/lib/commands/AGGREGATE.d.ts","./node_modules/@redis/search/dist/lib/commands/AGGREGATE_WITHCURSOR.d.ts","./node_modules/@redis/search/dist/lib/commands/CURSOR_READ.d.ts","./node_modules/@redis/search/dist/lib/commands/DROPINDEX.d.ts","./node_modules/@redis/search/dist/lib/commands/EXPLAIN.d.ts","./node_modules/@redis/search/dist/lib/commands/EXPLAINCLI.d.ts","./node_modules/@redis/search/dist/lib/commands/HYBRID.d.ts","./node_modules/@redis/search/dist/lib/commands/INFO.d.ts","./node_modules/@redis/search/dist/lib/commands/PROFILE_SEARCH.d.ts","./node_modules/@redis/search/dist/lib/commands/SEARCH_NOCONTENT.d.ts","./node_modules/@redis/search/dist/lib/commands/SPELLCHECK.d.ts","./node_modules/@redis/search/dist/lib/commands/SUGADD.d.ts","./node_modules/@redis/search/dist/lib/commands/SUGGET.d.ts","./node_modules/@redis/search/dist/lib/commands/SYNUPDATE.d.ts","./node_modules/@redis/search/dist/lib/commands/index.d.ts","./node_modules/@redis/search/dist/lib/index.d.ts","./node_modules/@redis/time-series/dist/lib/commands/ADD.d.ts","./node_modules/@redis/time-series/dist/lib/commands/helpers.d.ts","./node_modules/@redis/time-series/dist/lib/commands/CREATE.d.ts","./node_modules/@redis/time-series/dist/lib/commands/ALTER.d.ts","./node_modules/@redis/time-series/dist/lib/commands/CREATERULE.d.ts","./node_modules/@redis/time-series/dist/lib/commands/INCRBY.d.ts","./node_modules/@redis/time-series/dist/lib/commands/GET.d.ts","./node_modules/@redis/time-series/dist/lib/commands/INFO.d.ts","./node_modules/@redis/time-series/dist/lib/commands/INFO_DEBUG.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MADD.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MGET.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MGET_WITHLABELS.d.ts","./node_modules/@redis/time-series/dist/lib/commands/RANGE.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE_GROUPBY.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE_SELECTED_LABELS.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE_SELECTED_LABELS_GROUPBY.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE_WITHLABELS_GROUPBY.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE_WITHLABELS.d.ts","./node_modules/@redis/time-series/dist/lib/commands/MRANGE.d.ts","./node_modules/@redis/time-series/dist/lib/commands/index.d.ts","./node_modules/@redis/time-series/dist/lib/index.d.ts","./node_modules/redis/dist/index.d.ts","./node_modules/connect-redis/dist/connect-redis.d.cts","./node_modules/@types/multer/index.d.ts","./src/interfaces/api-error.interface.ts","./src/common/interceptors/global-exception.filter.ts","./src/common/interceptors/response-transform.interceptor.ts","./src/common/interceptors/logging.interceptor.ts","./src/config/cache.config.ts","./node_modules/helmet/index.d.cts","./src/config/cors.config.ts","./src/main.ts","./src/assessment/dto/create-assessment.dto.ts","./src/assessment/dto/update-assessment.dto.ts","./src/audit-log/decorators/audit.decorator.ts","./src/audit-log/tests/audit-log.test.ts","./src/auth/jwt.strategy.ts","./src/caching/decorators/cache.decorator.ts","./src/caching/interceptors/cache.interceptor.ts","./src/common/common.module.ts","./src/common/database/examples/booking-transaction.example.ts","./src/common/database/examples/payment-transaction.example.ts","./src/common/database/examples/voting-transaction.example.ts","./src/common/decorators/translate.decorator.ts","./src/common/examples/transaction-management.example.ts","./src/common/interceptors/api-error.interface.ts","./src/common/interceptors/cache.interceptor.ts","./src/common/lazy-loading/lazy-module-loader.service.ts","./src/common/lazy-loading/lazy-loading.module.ts","./src/common/lazy-loading/index.ts","./src/common/naming/naming.service.ts","./src/common/naming/naming.module.ts","./src/common/validators/is-strong-password.validator.ts","./src/config/swagger.config.ts","./src/courses/guards/ws-jwt-auth.guard.ts","./src/email-marketing/dto/index.ts","./src/email-marketing/entities/index.ts","./src/email-marketing/enums/index.ts","./src/gateways/messaging.gateway.ts","./src/gateways/notifications.gateway.ts","./src/graphql/inputs/index.ts","./src/graphql/middleware/dataloader.middleware.ts","./src/graphql/resolvers/index.ts","./src/graphql/types/index.ts","./src/messaging/tracing/tracing.service.ts","./src/messaging/messaging.service.ts","./src/messaging/event-bus/event-bus.service.ts","./src/messaging/discovery/service-discovery.service.ts","./src/messaging/circuit-breaker/circuit-breaker.service.ts","./src/messaging/messaging.module.ts","./src/migrations/samples/sample-user-table.migration.ts","./node_modules/@nestjs/mapped-types/dist/mapped-type.interface.d.ts","./node_modules/@nestjs/mapped-types/dist/types/remove-fields-with-type.type.d.ts","./node_modules/@nestjs/mapped-types/dist/intersection-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/omit-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/partial-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/pick-type.helper.d.ts","./node_modules/@nestjs/mapped-types/dist/type-helpers.utils.d.ts","./node_modules/@nestjs/mapped-types/dist/index.d.ts","./node_modules/@nestjs/mapped-types/index.d.ts","./src/payments/dto/update-payment.dto.ts","./src/payments/providers/payment-provider.interface.ts","./src/payments/webhooks/migration-helper.ts","./src/payments/webhooks/stripe-webhook.guard.ts","./src/payments/webhooks/dto/webhook-retry.dto.ts","./src/rate-limiting/services/limit-guard/guard.ts","./src/rate-limiting/rate-limiting.controller.ts","./src/rate-limiting/entities/rate-limiting.entity.ts","./src/tenancy/decorators/current-tenant.decorator.ts","./node_modules/@jest/expect-utils/build/index.d.ts","./node_modules/jest-matcher-utils/node_modules/chalk/index.d.ts","./node_modules/@sinclair/typebox/typebox.d.ts","./node_modules/@jest/schemas/build/index.d.ts","./node_modules/pretty-format/build/index.d.ts","./node_modules/jest-diff/build/index.d.ts","./node_modules/jest-matcher-utils/build/index.d.ts","./node_modules/expect/build/index.d.ts","./node_modules/@types/jest/index.d.ts"],"fileIdsList":[[546,593,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2922,2923,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2923,3007,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3006,3087,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2924,2925,2926,2931,2996,3001,3002,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2924,2995,2996,3000,3001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2925,2927,2929,2930,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2928,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2927,2929,2930,2931,2997,2999,3000,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2927,2930,2931,2997,2999,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2922,2924,2925,2927,2931,2998,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2928,3001,3003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2924,3001,3003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2925,3008,3009,3014,3079,3084,3085,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,3008,3078,3079,3083,3084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2925,3010,3012,3013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3011,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3010,3012,3013,3014,3080,3082,3083,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,3010,3013,3014,3080,3082,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,2920,2925,3007,3008,3010,3014,3081,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3011,3084,3086,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2918,3008,3084,3086,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2921,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2918,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2916,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2920,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2914,2915,2917,2919,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3005,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2222,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1858,1922,1932,2014,2029,2040,2059,2095,3817,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,3988,3991,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,3819,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1952,2040,2095,3819,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,3901,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3820,3821,3822,3823,3824,3825,3826,3827,3828,3829,3830,3831,3832,3833,3834,3835,3836,3837,3838,3839,3840,3841,3842,3843,3844,3845,3846,3847,3848,3849,3850,3851,3852,3853,3854,3855,3856,3857,3858,3859,3860,3861,3862,3863,3864,3865,3866,3867,3868,3869,3870,3871,3872,3873,3874,3875,3876,3877,3878,3879,3880,3881,3882,3883,3884,3885,3886,3887,3888,3889,3890,3891,3892,3893,3894,3895,3896,3897,3898,3899,3900,3902,3903,3904,3905,3906,3907,3908,3909,3910,3911,3912,3913,3914,3915,3916,3917,3918,3919,3920,3921,3922,3923,3924,3925,3926,3927,3928,3929,3930,3931,3932,3933,3934,3935,3936,3937,3938,3939,3940,3941,3942,3943,3944,3945,3946,3947,3948,3949,3950,3951,3952,3953,3954,3955,3956,3957,3958,3959,3960,3961,3962,3963,3964,3965,3966,3967,3968,3969,3970,3971,3972,3973,3974,3975,3976,3977,3978,3979,3980,3981,3982,3983,3984,3985,3986,3987,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,3989,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,3819,3901,3988,3990,3991,3992,3993,3994,3995,4014,4019,4020,4021,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,4020,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3818,3819,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3920,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3922,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3923,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3938,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3939,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3927,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3928,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3934,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3937,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3940,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3945,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3944,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3947,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3948,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3950,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3953,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,3955,3996,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3996,3997,3998,3999,4000,4001,4002,4003,4004,4005,4006,4007,4008,4009,4010,4011,4012,4013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3990,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4015,4016,4017,4018,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3888,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3897,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3898,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2222,3914,3992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1858,1922,1932,2014,2029,2040,2059,2095,2421,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,2477,2480,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2421,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2423,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2424,2425,2426,2427,2428,2429,2430,2431,2432,2433,2434,2435,2436,2437,2438,2439,2440,2441,2442,2443,2444,2445,2446,2447,2448,2449,2450,2451,2452,2453,2454,2455,2456,2457,2458,2459,2460,2461,2462,2463,2464,2465,2466,2467,2468,2469,2470,2471,2472,2473,2474,2475,2476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,2478,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2422,2423,2477,2479,2480,2481,2482,2483,2484,2494,2495,2496,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,2495,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2422,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2435,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2454,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2455,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2456,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2457,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2458,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2459,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2460,2485,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2485,2486,2487,2488,2489,2490,2491,2492,2493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2479,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1830,1857,1858,1902,1922,1932,2011,2014,2029,2031,2040,2059,2095,2104,2105,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,2218,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2105,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2103,2104,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2107,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2107,2207,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,2095,2207,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2108,2109,2110,2111,2112,2113,2114,2115,2116,2117,2118,2119,2120,2121,2122,2123,2124,2125,2126,2127,2128,2129,2130,2131,2132,2133,2134,2135,2136,2137,2138,2139,2140,2141,2142,2143,2144,2145,2146,2147,2148,2149,2150,2151,2152,2153,2154,2155,2156,2157,2158,2159,2160,2161,2162,2163,2164,2165,2166,2167,2168,2169,2170,2171,2172,2173,2174,2175,2176,2177,2178,2179,2180,2181,2182,2183,2184,2185,2186,2187,2188,2189,2190,2191,2192,2193,2194,2195,2196,2197,2198,2199,2200,2201,2202,2203,2204,2205,2206,2208,2209,2210,2211,2212,2213,2214,2215,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,1932,2216,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2104,2106,2107,2207,2217,2218,2219,2223,2224,2225,2231,2236,2237,2238,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2095,2106,2237,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2106,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2106,2107,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2173,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2174,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2177,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2179,2226,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2226,2227,2228,2229,2230,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2232,2233,2234,2235,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2167,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2168,2219,2222,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2097,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2096,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1847,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2096,2097,2098,2099,2100,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1847,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1919,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2101,2102,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1987,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1988,1989,1991,1992,1993,1994,1995,1996,1997,1998,1999,2000,2001,2002,2003,2004,2005,2006,2007,2008,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1970,1993,1994,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,1991,1992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1990,1993,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1970,2005,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,2002,2004,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2003,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,2001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2000,2002,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1990,2001,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1847,1848,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1848,1851,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1852,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1852,1853,1854,1855,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1848,1849,1850,1851,1853,1856,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1848,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1859,1860,1861,1862,1934,1935,1936,1937,2010,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2009,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1860,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1904,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,1903,1905,1906,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1903,1906,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1903,1904,1905,1906,1907,1920,1921,1922,1933,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1902,1903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1934,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1935,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2012,2013,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1902,1922,2012,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1876,1877,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1870,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1872,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1870,1871,1873,1874,1875,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1863,1864,1865,1866,1867,1868,1869,1872,1876,1877,1878,1879,1880,1881,1882,1883,1884,1885,1886,1887,1888,1889,1890,1891,1892,1893,1894,1895,1896,1897,1898,1899,1900,1901,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1876,1877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,3379,3496,3497,3498,3587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,3496,3498,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3499,3500,3501,3502,3503,3504,3505,3506,3507,3508,3509,3510,3511,3512,3513,3514,3515,3516,3517,3518,3519,3520,3521,3522,3523,3524,3525,3526,3527,3528,3529,3530,3531,3532,3533,3534,3535,3536,3537,3538,3539,3540,3541,3542,3543,3544,3545,3546,3547,3548,3549,3550,3551,3552,3553,3554,3555,3556,3557,3558,3559,3560,3561,3562,3563,3564,3565,3566,3567,3568,3569,3570,3571,3572,3573,3574,3575,3576,3577,3578,3579,3580,3581,3582,3583,3584,3585,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3325,3374,3375,3378,3379,3497,3586,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,3378,3495,3496,3587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3378,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3367,3368,3373,3374,3375,3376,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,3367,3373,3375,3376,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3324,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,3298,3324,3367,3368,3373,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,631,634,635,3324,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,3325,3326,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3325,3365,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3325,3327,3366,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,3374,3375,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3324,3367,3375,3377,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3367,3369,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3369,3370,3371,3372,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,634,3367,3374,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2986,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3069,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2986,2987,2988,2989,2990,2991,2992,2993,2994,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3069,3070,3071,3072,3073,3074,3075,3076,3077,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2992,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3075,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2736,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3015,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2932,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3015,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2932,2933,2934,2935,2936,2937,2938,2939,2940,2941,2942,2943,2944,2945,2946,2947,2948,2949,2950,2951,2952,2953,2954,2955,2956,2957,2958,2959,2960,2961,2962,2963,2964,2965,2966,2967,2968,2969,2970,2971,2972,2973,2974,2975,2976,2977,2978,2979,2980,2981,2982,2983,2984,2985,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3015,3016,3017,3018,3019,3020,3021,3022,3023,3024,3025,3026,3027,3028,3029,3030,3031,3032,3033,3034,3035,3036,3037,3038,3039,3040,3041,3042,3043,3044,3045,3046,3047,3048,3049,3050,3051,3052,3053,3054,3055,3056,3057,3058,3059,3060,3061,3062,3063,3064,3065,3066,3067,3068,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2972,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3055,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2933,2936,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,3016,3019,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3760,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3758,3759,3761,3763,3765,3766,3767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3764,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3765,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3725,3726,3730,3731,3732,3733,3734,3735,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3762,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3720,3723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,3720,3728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3721,3722,3724,3725,3726,3727,3729,3730,3731,3732,3733,3734,3735,3736,3737,3738,3739,3740,3741,3742,3743,3744,3745,3746,3747,3748,3749,3750,3751,3752,3753,3754,3755,3756,3757,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3719,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3701,3702,3703,3704,3706,3707,3708,3709,3713,3714,3715,3716,3717,3718,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3702,3703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3704,3705,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3710,3711,3712,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3701,3711,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3669,3670,3671,3672,3673,3674,3675,3676,3677,3678,3679,3680,3681,3682,3683,3684,3685,3686,3687,3688,3689,3690,3691,3692,3693,3694,3695,3696,3697,3698,3699,3700,3706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4617],[546,593,2912,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3004,3092,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,2730,2911,3092,3093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,3092,3093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3094,3095,3096,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3098,3099,3100,3101,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2913,3092,3097,3102,3104,3106,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3004,3088,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3089,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2911,3089,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3089,3090,3091,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3105,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[454,460,546,593,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3004,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3103,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3778,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,546,593,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3778,3779,3780,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1615,1616,1618,1620,1622,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1649,1651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1646,1647,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1653,1654,1655,1656,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1658,1659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1623,1648,1651,1652,1657,1660,1663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1647,1649,1650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1649,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1661,1662,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1678,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1048,1678,1679,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1681,1682,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1676,1680,1683,1685,1686,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1684,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1047,1048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1677,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1677,1678,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1687,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[53,304,305,306,307,308,309,310,311,312,313,314,315,316,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,290,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[263,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[253,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[321,322,323,324,325,326,327,328,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[258,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[317,320,329,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[318,319,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[294,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[258,259,260,261,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[331,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[276,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[331,332,333,334,335,336,337,338,339,340,341,342,343,344,345,346,347,348,349,350,351,352,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[359,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[354,355,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[356,358,546,593,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,262,303,330,353,358,360,367,390,395,397,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,248,249,471,476,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[248,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,247,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,369,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[250,371,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,251,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[255,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[268,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[270,271,272,273,274,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,263,278,282,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[276,277,283,284,285,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,55,56,57,58,248,249,250,251,252,253,254,255,256,257,263,268,269,275,282,286,287,288,290,298,299,300,301,302,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[264,265,266,267,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,264,265,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,262,263,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,266,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,294,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[289,291,292,293,294,295,296,297,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,256,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[290,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[54,256,289,293,295,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[265,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[291,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,290,291,292,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,260,280,298,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,279,281,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[252,254,263,269,278,283,299,300,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,252,254,257,299,300,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[261,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,303,361,365,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[365,366,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,361,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,361,362,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[362,363,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[362,363,364,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[382,383,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[382,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[383,384,385,386,387,388,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[381,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[373,383,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[383,384,385,386,387,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,382,383,386,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[368,374,375,376,377,378,379,380,389,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,303,374,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,373,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[257,373,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[250,256,257,369,370,371,372,373,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,369,370,391,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,369,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[393,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[330,391,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[391,392,394,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[280,357,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[289,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[262,303,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[396,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,644,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,534,539,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[533,539,546,593,644,645,646,649,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[539,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[540,546,593,642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[534,540,546,593,643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[535,536,537,538,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,647,648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[539,546,593,644,650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,650,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,282,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[440,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,460,461,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[442,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,454,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[464,465,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[58,303,455,460,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,441,467,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,398,468,471,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,460,462,473,475,479,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,477,478,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[468,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,455,460,462,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[481,483,484,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[57,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,454,455,460,480,482,485,488,493,494,507,508,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,440,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[467,470,509,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[494,506,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,441,462,463,466,469,501,506,510,513,517,518,519,521,523,529,531,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,448,456,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,452,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,451,452,453,454,459,460,462,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[454,455,458,460,496,505,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,447,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[495,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,455,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,448,455,459,500,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,447,459,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,453,454,458,498,502,503,504,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,448,455,456,457,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[256,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,442,455,458,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[459,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[444,445,446,455,459,460,499,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[451,500,511,512,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,442,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,442,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[443,444,445,446,449,451,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[448,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[450,451,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,443,444,445,446,449,450,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[486,487,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,460,462,474,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[497,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[287,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[268,303,514,515,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[516,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,462,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,455,462,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,303,398,448,455,456,457,459,460,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,280,303,398,441,455,462,500,518,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[281,282,398,440,520,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[490,491,492,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,489,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[522,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[525,527,528,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[524,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[526,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,454,459,525,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[472,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,442,455,459,460,462,497,498,500,501,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[530,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3590,3591,3592,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,3588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1671,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1670,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1670,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1665,1666,1672,1673,1674,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1667,1668,1669,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,398,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,2802,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2627,2805,2806,2807,2808,2809,2810,2811,2812,2813,2814,2815,2816,2817,2818,2819,2820,2821,2822,2823,2824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2628,2802,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2802,2818,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[532,546,593,2730,2804,2879,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2880,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2883,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2877,2878,2885,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2884,2886,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2882,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2732,2804,2858,2859,2877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2797,2804,2877,2878,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,532,546,593,2799,2880,2882,2888,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2797,2804,2825,2861,2862,2863,2874,2879,2881,2882,2883,2887,2888,2889,2890,2893,2897,2898,2901,2908,2909,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2731,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,398,546,593,2730,2732,2796,2797,2798,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,2731,2732,2733,2734,2735,2798,2799,2800,2801,2802,2803,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2891,2892,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2836,2840,2850,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2827,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2836,2839,2840,2849,2850,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2826,2837,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2830,2839,2840,2842,2843,2844,2849,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2852,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2830,2839,2840,2843,2846,2849,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2830,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2829,2830,2840,2843,2851,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2836,2844,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2804,2853,2854,2855,2856,2857,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2858,2859,2895,2896,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2827,2831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2827,2828,2829,2831,2832,2833,2834,2835,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2826,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2731,2804,2826,2827,2828,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2804,2826,2827,2828,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2730,2804,2826,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2894,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2838,2841,2845,2847,2848,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2836,2842,2846,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2804,2838,2841,2845,2847,2848,2849,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[459,460,546,593,2860,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,482,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,482,546,593,2862,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2867,2873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2861,2862,2863,2874,2875,2876,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[451,459,460,509,532,546,593,2804,2824,2860,2861,2880,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[460,532,546,593,2730,2799,2804,2861,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2899,2900,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2902,2904,2905,2906,2907,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2903,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2643,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2737,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2737,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2740,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2737,2738,2739,2740,2741,2742,2743,2744,2745,2746,2747,2748,2749,2750,2751,2752,2753,2754,2755,2756,2757,2758,2759,2760,2761,2762,2763,2764,2765,2766,2767,2768,2769,2770,2771,2772,2773,2774,2775,2776,2777,2778,2779,2780,2781,2782,2783,2784,2785,2786,2787,2788,2789,2790,2791,2792,2793,2794,2795,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2738,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2741,2742,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3159,3160,3161,3162,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3158,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3157,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3158,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3159,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3163,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4597,4599,4600,4601,4602,4603],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4597,4598],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4604],[398,546,593,1692,1694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1691,1694,1695,1696,1698,1699,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1692,1693,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1692,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1700,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,282,303,398,440,546,593,607,609,1741,1742,1743,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1744,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1745,1747,1758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1741,1742,1746,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,607,609,1601,1741,1742,1743,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1754,1756,1757,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1748,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1749,1750,1751,1752,1753,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,1748,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1755,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1755,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1580,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1581,1582,1583,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1562,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1563,1584,1585,1586,1587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1585,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1588,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,401,402,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,402,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[415,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,401,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[399,400,403,404,405,406,407,408,409,410,411,412,413,414,416,417,418,419,420,421,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,426,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,422,426,427,428,433,435,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,424,425,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[401,423,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,426,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[429,430,431,432,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[434,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[436,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1705,1706,1707,1708,1709,1710,1711,1712,1714,1715,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[303,546,593,1705,1706,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1704,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,1705,1706,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1704,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1705,1707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1704,1705,1713,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1039,1040,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1038,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,1037,1038,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1041,1043,1044,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1038,1042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1045,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2513,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,303,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2515,2516,2517,2527,2528,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2526,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2530,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2532,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,501,546,593,2534,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,2514,2526,2529,2531,2533,2535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[278,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2519,2520,2521,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2518,2522,2523,2524,2525,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3261,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3265,3267,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3254,3258,3269,3270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3283,3289,3291,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3252,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3263,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3253,3254,3255,3256,3257,3258,3259,3260,3261,3262,3263,3264,3265,3266,3267,3268,3269,3271,3272,3273,3274,3275,3276,3277,3278,3279,3280,3281,3282,3283,3284,3285,3286,3287,3288,3289,3290,3292,3293,3294,3295,3296,3297,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3268,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3264,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3257,3258,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3264,3265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3271,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3292,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3277,3279,3280,3281,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3281,3283,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3272,3275,3278,3285,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3272,3273,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3256,3257,3275,3278,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3257,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3274,3277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3273,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3274,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3272,3274,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3255,3256,3272,3274,3275,3276,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3274,3277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3258,3277,3279,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3280,3281,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4474,4475,4476,4477,4478,4479,4480,4481,4482,4483,4484,4485,4486,4487,4488],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4489],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4355,4356,4357,4358,4359,4360,4388,4444,4457,4460,4462,4464,4465,4467,4471,4472],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4355,4359],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4355,4356,4357,4358],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4448],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4446,4447,4448,4449,4450],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4446,4447,4448],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4464],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4357,4359,4452,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4453,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4460],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4397,4444,4445,4451,4452,4453,4454,4455,4456,4457,4461,4462,4463],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4355,4359,4444,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4444,4454],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4453,4454,4455,4457,4461,4464],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4453],[546,593,604,612,631,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,600,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4452,4457,4460,4464],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4445,4452,4453,4457,4458,4459,4461,4464],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4371],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4376],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4382,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4386],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4404,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4368],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4415],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4427],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4361,4362,4363,4364,4365,4366,4367,4368,4369,4370,4371,4372,4373,4374,4375,4376,4377,4378,4379,4380,4381,4382,4383,4384,4385,4386,4387,4388,4389,4390,4391,4392,4393,4394,4395,4396,4397,4398,4399,4400,4401,4402,4403,4404,4405,4406,4407,4408,4409,4410,4411,4412,4413,4414,4415,4416,4417,4418,4419,4420,4421,4422,4423,4424,4425,4426,4427,4428,4429,4430,4431,4432,4433,4434,4435,4436,4437,4438,4439,4440,4441,4442,4443,4473],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4461,4468],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4468,4469,4470],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4468],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4452,4453,4454,4457,4461,4464,4465,4466],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4444,4454,4465],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4444,4445,4453,4457,4464,4467],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4491,4492,4493,4494,4495,4496,4497,4498,4499,4500,4501,4502,4503,4504,4505],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4506],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4508,4509],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4510],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4511],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4509],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4510],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4473],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4509,4510],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4508],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4359,4473,4509],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4508,4509,4510,4511,4512,4513,4514,4515,4516,4517,4518,4519,4520,4521,4522,4523],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4508,4509,4510,4524],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4527],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4528],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4526,4527],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4527,4530],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4354,4359,4533],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4536],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538,4539,4540],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4527,4538,4539],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4354,4359,4526],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4526,4527,4528,4529,4530,4531,4532,4533,4534,4535,4536,4537,4538,4539,4540,4541,4542,4543,4544],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4530,4538,4539,4545],[546,593,2015,2016,2017,2018,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2019,2022,2028,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2020,2021,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2024,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2025,2026,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2025,2026,2027,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1984,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1984,1985,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1938,1939,1940,1985,1986,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,1939,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1971,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1932,1970,1974,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1972,1973,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1970,1974,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1952,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1953,1954,1973,1974,1975,1976,1977,1978,1979,1980,1981,1982,1983,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1973,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1981,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1964,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1955,1957,1958,1959,1960,1961,1962,1963,1964,1965,1966,1967,1968,1969,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1956,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1963,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1958,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2085,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2082,2083,2086,2087,2088,2089,2090,2091,2092,2093,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2030,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2032,2033,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2034,2035,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2032,2033,2036,2037,2038,2039,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2051,2052,2053,2054,2055,2056,2057,2058,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2049,2051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2050,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2055,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1831,1844,1845,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1843,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1831,1844,1845,1846,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1928,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1925,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1926,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,1923,1924,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1923,1924,1925,1927,1928,1929,1930,1931,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1832,1833,1834,1835,1837,1838,1839,1840,1841,1842,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1836,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1837,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1908,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1908,1909,1910,1911,1912,1913,1914,1915,1916,1917,1918,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1984,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2040,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2060,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2070,2071,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2072,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2060,2061,2062,2063,2064,2065,2066,2067,2068,2069,2073,2074,2075,2076,2077,2078,2079,2080,2081,2094,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1761,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1774,1775,1776,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1774,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,1764,1775,1776,1777,1778,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1767,1773,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1786,1788,1789,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1786,1787,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1791,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1768,1793,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,1794,1795,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1793,1794,1795,1796,1797,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1770,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1772,1774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1783,1784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,1803,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1799,1800,1803,1804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1773,1802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1780,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,1771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,1770,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1765,1807,1808,1809,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1762,1766,1767,1768,1769,1770,1771,1772,1773,1774,1779,1782,1783,1784,1785,1787,1790,1791,1792,1798,1801,1802,1805,1806,1810,1811,1812,1813,1814,1815,1816,1817,1818,1819,1820,1821,1822,1823,1824,1826,1827,1828,1829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1763,1767,1768,1769,1770,1773,1777,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1767,1785,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1768,1774,1813,1815,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1768,1774,1813,1814,1815,1816,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,1773,1787,1817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,1791,1808,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1773,1783,1784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,1781,1813,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1766,1767,1823,1824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,608,1767,1772,1785,1813,1822,1823,1824,1825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1767,1785,1801,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1772,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2041,2042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1830,1922,2041,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2042,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2041,2042,2043,2044,2045,2046,2047,2048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1830,1922,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1943,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1942,1944,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1941,1942,1945,1946,1947,1948,1949,1950,1951,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2220,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2220,2221,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,1598,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,641,1592,1593,1594,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,604,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1595,1597,1599,1600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4619,4622],[546,593,598,641,3156,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1572,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1566,1568,1569,1573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,1567,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1567,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,1568,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1565,1567,1568,1569,1570,1571,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1564,1570,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,599,604,612,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,595,604,612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[541,542,543,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,596,635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,597,598,605,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,623,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,599,601,604,612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,602,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,603,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,606,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,606,619,623,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,604,607,612,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,607,608,612,623,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,623,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[544,545,546,547,548,549,550,589,590,591,592,593,594,595,596,597,598,599,600,601,602,603,604,605,606,607,608,609,610,611,612,613,614,615,616,617,618,619,620,621,622,623,624,625,626,627,628,629,630,631,632,633,634,635,636,637,638,639,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,611,634,639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,604,612,623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,614,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,615,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,591,592,593,594,595,596,597,598,599,600,601,602,603,604,605,606,607,608,609,610,611,612,613,614,615,616,617,618,619,620,621,622,623,624,625,626,627,628,629,630,631,632,633,634,635,636,637,638,639,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,619,620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,619,621,635,637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,624,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,625,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,590,593,623,628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,629,630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,629,630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,612,623,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,632,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,618,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,598,635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,611,637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,588,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,588,593,604,606,615,623,626,634,637,639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3174,3176,3180,3181,3182,3183,3184,3185,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,623,634,641,3173,3174,3175,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3176,3177,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3176,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3178,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,623,631,641,3175,3177,3179,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3174,3176,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,3174,3175,3176,3177,3178,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,641,3174,3176,3177,3179,3186,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,623,641,3179,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3157,3195,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1601,1697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,605,623,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,1596,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2312,2313,2314,2315,2316,2317,2318,2319,2320,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,609,612,623,631,634,640,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,3436,3438,3439,3440,3441,3445,3446,3447,3448,3449,3450,3451,3452,3453,3454,3455,3456,3457,3458,3459,3460,3461,3462,3463,3464,3465,3466,3467,3468,3469,3470,3471,3472,3473,3474,3475,3476,3477,3481,3483,3484,3485,3486,3487,3488,3489,3490,3491,3492,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3494,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3442,3443,3444,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3450,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3442,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3468,3469,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3401,3404,3408,3410,3422,3436,3437,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3445,3467,3468,3469,3470,3472,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3387,3388,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3401,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3392,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3395,3396,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3404,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3397,3398,3425,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3408,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3392,3393,3427,3434,3435,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3389,3390,3391,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3394,3398,3426,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3396,3428,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3396,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3429,3430,3431,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3425,3432,3433,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3390,3425,3433,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3396,3399,3400,3402,3403,3405,3406,3407,3409,3411,3412,3413,3414,3415,3416,3417,3418,3419,3420,3421,3423,3424,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3422,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3445,3446,3447,3448,3449,3450,3451,3452,3453,3454,3455,3456,3457,3458,3459,3460,3461,3462,3463,3464,3465,3466,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3475,3476,3477,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,606,623,626,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3445,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3475,3476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3445,3468,3469,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3438,3439,3475,3476,3477,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3438,3439,3476,3478,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3386,3387,3388,3389,3390,3391,3392,3393,3426,3427,3438,3439,3468,3476,3480,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3438,3439,3468,3469,3470,3471,3475,3476,3477,3479,3481,3482,3483,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3388,3468,3472,3484,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,3388,3438,3439,3468,3469,3470,3471,3472,3475,3476,3477,3479,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3469,3470,3472,3481,3493,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3468,3469,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3468,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3468,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3467,3468,3469,3470,3471,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,3439,3440,3441,3467,3469,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3476,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,641,3386,3467,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3467,3468,3470,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3438,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3439,3468,3469,3470,3473,3481,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3116,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1048,1560,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1438,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1412,1413,1414,1415,1416,1417,1418,1419,1420,1421,1422,1423,1424,1425,1426,1427,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,1455,1456,1457,1458,1459,1460,1461,1462,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1453,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1463,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1050,1051,1084,1123,1311,1402,1406,1410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1051,1400,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1052,1053,1054,1055,1056,1057,1058,1059,1060,1061,1062,1063,1064,1065,1066,1067,1068,1069,1070,1071,1072,1073,1074,1075,1076,1077,1078,1079,1080,1081,1082,1083,1085,1086,1087,1088,1089,1090,1091,1092,1093,1094,1095,1096,1097,1098,1099,1100,1101,1102,1103,1104,1105,1106,1107,1108,1109,1110,1111,1112,1113,1114,1115,1116,1117,1118,1119,1120,1121,1122,1123,1124,1125,1126,1127,1128,1129,1130,1131,1132,1133,1134,1135,1136,1137,1138,1139,1140,1141,1142,1143,1144,1145,1146,1147,1148,1149,1150,1151,1152,1153,1154,1155,1156,1157,1158,1159,1160,1161,1162,1163,1164,1165,1166,1167,1168,1169,1170,1171,1172,1173,1174,1175,1176,1177,1178,1179,1180,1181,1182,1183,1184,1185,1186,1187,1188,1189,1190,1191,1192,1193,1194,1195,1196,1197,1198,1199,1200,1201,1202,1203,1204,1205,1206,1207,1208,1209,1210,1211,1212,1213,1214,1215,1216,1217,1218,1219,1220,1221,1222,1223,1224,1225,1226,1227,1228,1229,1230,1231,1232,1233,1234,1235,1236,1237,1238,1239,1240,1241,1242,1243,1244,1245,1246,1247,1248,1249,1250,1251,1252,1253,1254,1255,1256,1257,1258,1259,1260,1261,1262,1263,1264,1265,1266,1267,1268,1269,1270,1271,1272,1273,1274,1275,1276,1277,1278,1279,1280,1281,1282,1283,1284,1285,1286,1287,1288,1289,1290,1291,1292,1293,1294,1295,1296,1297,1298,1299,1300,1301,1302,1303,1304,1305,1306,1307,1308,1309,1310,1311,1312,1313,1314,1315,1316,1317,1318,1319,1320,1321,1322,1323,1324,1325,1326,1327,1328,1329,1330,1331,1332,1333,1334,1335,1336,1337,1338,1339,1340,1341,1342,1343,1344,1345,1346,1347,1348,1349,1350,1351,1352,1353,1354,1355,1356,1357,1358,1359,1360,1361,1362,1363,1364,1365,1366,1367,1368,1369,1370,1371,1372,1373,1374,1375,1376,1377,1378,1379,1380,1381,1382,1383,1384,1385,1386,1387,1388,1389,1390,1391,1392,1393,1394,1395,1396,1397,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1049,1051,1084,1165,1250,1398,1399,1400,1401,1403,1404,1405,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1398,1403,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,612,631,641,1051,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,641,1051,1400,1406,1410,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1176,1177,1178,1179,1180,1181,1182,1183,1184,1185,1186,1187,1188,1189,1190,1191,1192,1193,1194,1195,1196,1197,1198,1199,1200,1201,1202,1203,1204,1205,1206,1207,1208,1209,1210,1211,1212,1213,1214,1215,1216,1217,1218,1219,1220,1221,1222,1223,1224,1225,1226,1227,1228,1229,1230,1231,1232,1233,1234,1235,1236,1237,1238,1239,1240,1241,1242,1243,1244,1245,1246,1247,1248,1249,1250,1251,1252,1253,1254,1255,1256,1257,1258,1259,1260,1261,1262,1263,1264,1265,1266,1267,1268,1269,1270,1271,1272,1273,1274,1275,1276,1277,1278,1279,1280,1281,1282,1283,1284,1285,1286,1287,1288,1289,1290,1291,1292,1293,1294,1295,1296,1297,1298,1299,1300,1301,1302,1303,1304,1305,1306,1307,1308,1309,1310,1311,1312,1313,1314,1315,1316,1317,1318,1319,1320,1321,1322,1323,1324,1325,1326,1327,1328,1329,1330,1331,1332,1333,1334,1335,1336,1337,1338,1339,1340,1341,1342,1343,1344,1345,1346,1347,1348,1349,1350,1351,1352,1353,1354,1355,1356,1357,1358,1359,1360,1361,1362,1363,1364,1365,1366,1367,1368,1369,1370,1371,1372,1373,1374,1375,1376,1377,1378,1379,1380,1381,1382,1383,1384,1385,1386,1387,1388,1389,1390,1391,1392,1393,1394,1395,1396,1397,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1051,1400,1406,1407,1408,1409,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1403,1407,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1178,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1183,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1185,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1074,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1101,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1123,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1215,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1213,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1211,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1221,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1066,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1232,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1247,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1248,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1250,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1260,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1280,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1325,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1339,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1341,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1364,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1368,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1374,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1376,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1378,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1379,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1381,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1084,1395,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1049,1050,1406,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1402,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1471,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1465,1466,1467,1468,1469,1470,1471,1472,1473,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1406,1471,1472,1474,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1474,1475,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1477,1478,1479,1480,1481,1482,1483,1484,1485,1486,1487,1488,1489,1490,1491,1492,1493,1494,1495,1496,1497,1498,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1503,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1504,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1503,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1520,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1520,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1527,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1527,1528,1529,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1501,1502,1503,1504,1505,1506,1507,1508,1509,1510,1511,1512,1513,1514,1515,1516,1517,1518,1519,1520,1521,1522,1523,1524,1525,1526,1527,1528,1529,1530,1531,1532,1533,1534,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1503,1520,1535,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1537,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1546,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1549,1558,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1051,1537,1538,1539,1540,1541,1542,1543,1544,1545,1546,1547,1548,1549,1550,1551,1552,1553,1554,1555,1556,1557,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1411,1412,1413,1414,1415,1416,1417,1418,1419,1420,1421,1422,1423,1424,1425,1426,1427,1428,1429,1430,1431,1432,1433,1434,1435,1436,1437,1439,1440,1441,1442,1443,1444,1445,1446,1447,1448,1449,1450,1451,1452,1454,1455,1456,1457,1458,1459,1460,1461,1462,1464,1465,1466,1467,1468,1469,1470,1471,1472,1473,1476,1477,1478,1479,1480,1481,1482,1483,1484,1485,1486,1487,1488,1489,1490,1491,1492,1493,1494,1495,1496,1497,1498,1500,1501,1502,1503,1504,1505,1506,1507,1508,1509,1510,1511,1512,1513,1514,1515,1516,1517,1518,1519,1520,1521,1522,1523,1524,1525,1526,1527,1528,1529,1530,1531,1532,1533,1534,1536,1537,1538,1539,1540,1541,1542,1543,1544,1545,1546,1547,1548,1549,1550,1551,1552,1553,1554,1555,1556,1557,1559,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,1047,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2612,2613,2614,2615,2616,2617,2618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2602,2610,2611,2619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2603,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2594,2595,2596,2597,2598,2599,2600,2603,2604,2605,2606,2607,2608,2609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2602,2604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2605,2610,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,2284,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2285,2286,2287,2288,2290,2291,2292,2293,2294,2295,2296,2297,2298,2299,2300,2301,2302,2303,2304,2305,2306,2307,2308,2309,2310,2311,2322,2323,2324,2325,2326,2327,2328,2329,2330,2331,2332,2333,2334,2335,2336,2337,2338,2339,2340,2341,2342,2343,2344,2345,2346,2347,2348,2349,2350,2351,2352,2353,2354,2355,2356,2357,2358,2359,2362,2363,2364,2365,2366,2367,2368,2369,2370,2371,2372,2373,2374,2375,2376,2377,2378,2379,2380,2381,2382,2383,2384,2385,2386,2387,2388,2389,2390,2391,2392,2393,2394,2395,2396,2397,2398,2399,2400,2401,2402,2403,2404,2405,2406,2407,2408,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2321,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2361,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2279,2280,2281,2282,2283,2284,2289,2409,2410,2411,2412,2416,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2281,2414,2415,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2283,2413,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2284,2289,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2279,2280,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4352,4547],[546,593,1573,1576,1578,1579,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1573,1578,1579,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1573,1574,1578,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,594,1573,1575,1576,1577,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2538,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2538,2539,2540,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2541,2544,2545,2548,2552,2553,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,623,2543,2544,2545,2546,2547,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,2541,2544,2548,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,607,2541,2542,2543,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,2549,2550,2551,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2541,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2541,2543,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2544,2548,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4615,4621],[546,593,3382,3384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3381,3384,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3380,3381,3382,3383,3384,3385,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3382,3383,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3109,3110,3111,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3109,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3108,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,3109,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2730,2864,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2864,2865,2866,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2637,2638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2704,2705,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2633,2634,2637,2639,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2633,2644,2658,2659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2642,2643,2644,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2639,2641,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2642,2643,2644,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2629,2645,2650,2657,2660,2661,2703,2706,2729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2635,2636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2635,2636,2637,2638,2640,2651,2652,2653,2654,2655,2656,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2636,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2631,2636,2637,2639,2652,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2635,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2644,2658,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2633,2634,2637,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2640,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2633,2641,2642,2643,2646,2647,2648,2649,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2639,2641,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2641,2642,2643,2655,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2642,2653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,2707,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2639,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2640,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2662,2707,2708,2709,2710,2711,2712,2713,2714,2715,2716,2717,2718,2719,2720,2721,2722,2723,2724,2725,2726,2727,2728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2631,2637,2641,2642,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2632,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,2643,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2632,2637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,2641,2642,2643,2655,2662,2910,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2663,2664,2665,2666,2667,2668,2669,2670,2671,2672,2673,2674,2675,2676,2677,2678,2679,2680,2681,2682,2683,2684,2685,2686,2687,2688,2689,2690,2691,2692,2693,2694,2695,2696,2697,2698,2699,2700,2701,2702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2655,2663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2663,2665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2630,2637,2639,2642,2662,2663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,609,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,1624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1640,1641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1625,1629,1639,1643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1625,1626,1628,1629,1636,1639,1640,1642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,641,1629,1636,1637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,641,1624,1625,1626,1628,1629,1637,1638,1643,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1632,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,631,641,1624,1630,1632,1633,1638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,641,1624,1630,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,1625,1626,1627,1630,1634,1635,1636,1637,1638,1639,1643,1644,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1629,1631,1634,1635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1624,1625,1627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4619],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4616,4620],[546,593,2360,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4618],[546,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4353,4359,4473,4474,4475,4476,4477,4478,4479,4480,4481,4482,4483,4484,4485,4486,4487,4488,4490,4491,4492,4493,4494,4495,4496,4497,4498,4499,4500,4501,4502,4503,4504,4505,4506,4507,4508,4509,4510,4511,4512,4513,4514,4515,4516,4517,4518,4519,4520,4521,4522,4523,4525,4526,4528,4529,4531,4532,4533,4534,4535,4536,4537,4538,4539,4540,4541,4542,4543,4544,4545,4546],[59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,75,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,115,116,117,118,119,120,121,122,123,124,125,126,128,129,130,131,132,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,178,179,180,182,191,193,194,195,196,197,198,200,201,203,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,238,239,240,241,242,243,244,245,246,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,60,61,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,181,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[61,62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,62,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,63,220,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,220,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,228,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,63,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[72,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[95,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[116,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,104,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,111,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,104,122,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,63,122,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,163,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,182,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[188,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,181,188,189,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[181,182,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[188,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,63,188,189,190,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[204,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[199,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[202,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[60,62,182,183,184,185,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[104,182,183,184,185,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[182,184,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[62,183,184,186,187,191,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[59,62,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[63,206,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,105,106,107,108,109,110,112,113,114,115,116,117,118,119,120,121,123,124,125,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[192,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2559,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2559,2560,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2555,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2557,2561,2562,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2554,2556,2557,2564,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,608,609,2554,2556,2557,2561,2562,2563,2564,2565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2557,2558,2561,2563,2564,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,2554,2556,2557,2558,2561,2562,2563,2565,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,4032,4033,4034,4035,4036,4037,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313],[546,593,607,641,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,4032,4033,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4032,4033,4034,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2637,2644,2868,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2869,2870,2871,2872,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,607,641,2543,2730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,719,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,825,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,825,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,715,716,717,718,720,721,829,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,673,716,720,723,727,759,775,776,778,780,786,787,788,789,825,826,827,828,831,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,791,793,795,796,806,808,809,810,811,812,813,814,816,818,819,820,821,824,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,715,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,696,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,970,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,991,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,664,665,695,937,938,939,940,941,942,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,946,947,948,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,955,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,953,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,983,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,971,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,664,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,699,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,694,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,700,701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,656,658,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,711,712,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,654,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,654,655,662,668,670,672,686,687,688,691,692,722,723,725,726,831,837,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,689,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,815,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,790,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,672,690,723,725,731,733,747,760,764,768,775,829,835,837,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,612,631,731,732,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,792,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,807,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,794,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,817,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,723,822,823,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,797,798,799,800,801,802,803,804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,722,723,805,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,733,735,739,740,741,742,743,770,772,773,774,776,778,779,780,784,785,787,829,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,733,736,740,744,745,769,770,772,773,774,786,829,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,829,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,714,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,696,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,695,697,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,693,698,702,703,704,705,706,707,708,709,710,713,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,653,654,655,659,699,700,701,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,831,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,686,718,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,789,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,727,873,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,873,874,875,878,879,880,881,882,883,884,885,886,887,888,889,890,891,892,893,894,895,896,897,898,899,900,901,902,903,904,905,906,907,908,909,910,911,912,913,914,915,916,917,918,919,920,921,922,923,924,925,926,927,928,929,930,931,932,933,934,935,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,728,735,736,737,738,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,663,668,729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,739,740,777,831,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,733,734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,744,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,730,734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,659,668,775,776,778,786,787,825,826,829,855,868,869,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[52,546,593,656,658,659,662,663,665,668,669,670,671,672,673,693,694,698,699,701,702,703,714,715,716,717,718,721,723,724,725,727,728,729,730,733,734,735,736,737,738,739,740,741,742,743,746,747,749,750,751,752,753,754,755,756,757,758,759,761,764,765,768,770,771,772,773,774,775,776,777,778,781,782,786,787,788,789,825,829,831,834,835,836,837,838,848,849,851,852,853,854,855,869,870,871,872,936,943,944,945,949,950,951,952,954,956,957,958,959,960,961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980,981,982,984,985,986,987,988,989,990,992,993,994,995,996,997,998,999,1000,1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,1011,1012,1013,1014,1015,1016,1017,1018,1019,1020,1021,1022,1024,1025,1026,1027,1028,1029,1030,1031,1032,1034,1036,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,717,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,838,1017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,717,838,1017,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,665,666,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,680,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,657,660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,858,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,661,667,676,677,681,683,713,762,766,830,832,856,857,858,859,860,861,862,863,864,865,866,867,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,656,657,660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,703,704,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,762,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,662,663,667,668,675,685,829,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,676,678,679,682,684,686,829,831,833,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,680,681,685,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,674,675,678,679,682,684,685,686,703,704,711,712,713,763,767,829,830,833,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,766,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,654,655,673,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,685,686,831,832,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,855,856,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,677,831,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,652,653,654,655,657,661,668,674,685,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,653,673,683,685,686,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,686,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,788,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,789,829,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,673,1030,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,672,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,831,876,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,686,877,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,605,623,718,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,781,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,774,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,726,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,770,772,783,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,829,831,840,847,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,686,727,770,782,829,831,838,839,840,846,847,848,849,850,851,852,853,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,703,726,829,831,839,840,841,842,843,844,845,846,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,829,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,675,703,731,735,736,737,738,740,781,782,831,838,844,845,847,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,703,773,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,703,770,773,781,782,829,838,854,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,782,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,675,831,847,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,771,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,831,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,670,672,690,723,725,731,747,760,764,768,771,780,786,829,835,837,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,658,668,778,786,787,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,733,735,739,740,741,742,743,770,772,773,774,784,785,787,855,1023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,733,739,740,744,745,775,787,838,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,659,668,733,735,739,740,741,742,743,770,772,773,774,784,785,786,838,855,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,777,787,855,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,726,783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,686,690,691,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,690,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,724,746,761,765,834,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,725,727,747,764,768,831,835,836,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,761,763,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,669,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,765,767,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,674,724,727,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,833,834,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,684,746,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,671,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,748,759,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,749,750,751,752,753,754,755,756,757,758,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,753,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,668,675,686,786,829,831,838,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,670,672,686,689,715,725,730,734,747,764,768,775,782,826,831,835,837,848,849,850,851,852,853,855,877,1023,1024,1025,1033,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,786,831,1035,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,564,593,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,593,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,557,560,593,631,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,557,560,593,612,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,553,556,559,593,604,623,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,567,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,558,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,581,582,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,556,560,593,626,634,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,581,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,554,555,593,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,554,555,556,557,558,559,560,561,562,564,565,566,567,568,569,570,571,572,573,574,575,576,577,578,579,580,582,583,584,585,586,587,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,575,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,567,568,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,558,560,568,569,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,559,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,555,560,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,560,564,568,569,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,564,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,558,560,563,593,634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,552,557,560,567,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,555,560,581,593,639,641,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3364,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,634,3336,3340,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,623,634,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,631,634,3333,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,612,634,3333,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,604,623,634,3328,3329,3332,3335,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3343,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3334,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3357,3358,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,626,634,641,3332,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3357,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,641,3330,3331,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3330,3331,3332,3333,3334,3335,3336,3337,3338,3340,3341,3342,3343,3344,3345,3346,3347,3348,3349,3350,3351,3352,3353,3354,3355,3356,3358,3359,3360,3361,3362,3363,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,601,3336,3351,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3343,3344,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3334,3336,3344,3345,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3335,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3331,3336,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3340,3344,3345,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3340,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,634,3334,3336,3339,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3328,3333,3336,3343,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3336,3351,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,639,641,3331,3336,3357,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2249,2250,2251,2252,2253,2254,2255,2257,2258,2259,2260,2261,2262,2263,2264,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2251,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2251,2256,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2574,2575,2576,3242,3243,3244,3245,3246,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3238,3239,3240,3241,3242,3243,3244,3245,3246,3247,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3238,3239,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,3244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3238,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3239,3240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3239,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3238,3239,3240,3241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3239,3241,3244,3245,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,438,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,438,439,532,546,593,651,1046,1561,1591,1608,1609,1610,1614,1664,1675,1688,1690,1716,1719,1721,1724,1725,1726,1727,1728,1730,1732,1740,2272,2507,2578,2585,2626,3155,3200,3205,3221,3237,3248,3304,3312,3316,3323,3601,3649,3662,3668,3776,3789,3795,3809,4024,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4329,4343,4347,4350],[398,546,593,1703,3152,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3146,3147,3148,3149,3150,3151,3152,3153,3154,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3144,3146,3147,3148,3149,3150,3151,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4558],[546,593,1037,3146,3149,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3144,3147,3148,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3145,3147,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3145,3146,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,3190,3191,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1589,3191,3192,3201,3202,3203,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,2624,3190,3191,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3190,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1601,3190,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,651,1037,3191,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,1716,3127,3165,3193,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1611,1701,1724,3130,3164,3193,3194,3197,3200,3204,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,651,1611,1723,2574,3117,3118,3126,3164,3165,3189,3190,3192,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2574,3113,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1701,1702,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,2574,2575,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,2566,3157,3164,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,1701,2574,3157,3164,3196,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,1701,3157,3164,3196,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,2498,2500,2501,2502,2503,2504,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1589,1608,1664,1733,2272,2276,2278,2420,2498,2499,2501,2502,2505,2506,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,1589,1591,1602,1603,1646,1664,1733,2273,2274,2275,2276,2418,2419,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,635,651,1602,2241,2420,2497,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2273,2274,2275,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2277,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2273,2274,2275,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2276,2277,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,605,1037,1046,2241,2276,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2275,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1590,1591,1602,2273,2276,2277,2278,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,614,635,651,1037,1046,1646,1664,1733,2239,2241,2273,2276,2419,2420,2497,2499,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,594,605,635,651,1037,1046,1602,1646,1664,1733,2241,2277,2278,2419,2420,2497,2500,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3125,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,2589,2590,3125,3134,3135,3313,3314,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,1675,3124,3125,3134,3135,3313,3314,3315,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,3124,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,532,546,593,3124,3125,3314,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4563],[398,546,593,1675,3124,3125,3134,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3124,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3124,3125,3134,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3811,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1759,1760,2240,2244,2245,2246,2247,3814,3815,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1688,1759,1760,2245,2246,2247,3810,3811,3812,3813,3814,3816,4023,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1048,1688,1760,2240,3810,3811,3812,3813,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2243,3814,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,2239,4022,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1717,2240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2508,2509,2510,2511,2512,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,2512,2569,2577,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2508,2509,2510,2511,2512,2536,2537,2566,2567,2568,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4328],[398,437,546,593,1601,1703,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344,4345,4346],[398,546,593,651,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1611,1613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,532,546,593,1611,1612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4332],[546,593,2417,2620,3119,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1730,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4328],[398,546,593,1037,1046,1646,1664,2572,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,2589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,1716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1716,2536,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1048,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1601,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4549,4550],[180,247,398,546,593,1601,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[180,247,398,546,593,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1728,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4573,4574],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4573],[398,532,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4344],[398,532,546,593,1601,1731,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4576],[398,437,546,593,1703,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1729,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4348,4349],[546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,592,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,3119,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2566,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3113,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1561,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,651,3594,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1689,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1732,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,3131,3132,3133,3136,3137,3138,3139,3140,3141,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,2570,2571,2572,2573,2574,3136,3138,3140,3141,3142,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1675,2570,2571,2573,3121,3122,3124,3125,3132,3133,3135,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3131,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2572,2573,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2570,2573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2571,2572,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2573,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2571,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2566,3164,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2570,2571,3139,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2570,2571,2573,3137,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2576,2579,2580,2581,2582,2583,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2579,2580,2581,2582,2583,2584,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3605,3622,3623,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3603,3604,3605,3620,3621,3622,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3609,3620,3621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3634,3635,3636,3637,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1675,1733,1734,3629,3630,3631,3632,3633,3634,3635,3636,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,3629,3631,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,2620,3610,3611,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3614,3615,3617,3618,3622,3625,3626,3627,3635,3636,3642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3635,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3614,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3617,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3609,3625,3626,3627,3628,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1664,1733,3602,3603,3605,3607,3609,3612,3613,3616,3619,3621,3623,3624,3628,3630,3632,3634,3637,3638,3639,3640,3641,3643,3644,3645,3646,3647,3648,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1733,3607,3608,3609,3616,3619,3623,3624,3625,3626,3627,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3605,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3603,3604,3609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3631,3634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3629,3634,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3630,3632,3633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3606,3609,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3602,3605,3607,3608,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3602,3603,3605,3607,3609,3612,3613,3621,3630,3632,3634,3647,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3610,3611,3613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,1037,3612,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3604,3606,3608,3610,3611,3620,3629,3631,3633,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1646,1664,1733,3606,3607,3608,3609,3616,3623,3638,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3613,3614,3615,3616,3642,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3610,3611,3612,3613,3614,3615,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3186,3619,3620,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3187,3602,3617,3618,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,3602,3617,3618,3619,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,3620,3624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,3318,3319,3320,3321,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3317,3319,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3318,3319,3320,3321,3322,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3652,3653,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3654,3655,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,3652,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2574,3654,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3656,3657,3658,3659,3660,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3650,3651,3652,3653,3654,3655,3656,3657,3658,3659,3660,3661,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3656,3657,3659,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,2574,3650,3651,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2536,2537,2566,2567,2568,3164,3169,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,614,2911,3107,3112,3130,3143,3155,3205,3211,3213,3214,3215,3216,3217,3219,3220,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3209,3210,3212,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,2574,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,3219,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3126,3206,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3211,3213,3214,3215,3216,3217,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2911,3112,3126,3136,3152,3206,3207,3208,3209,3210,3212,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,2911,3121,3126,3136,3152,3206,3207,3208,3209,3210,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2911,3112,3206,3207,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3136,3206,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2573,2574,3126,3136,3147,3152,3218,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2911,3207,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3206,3207,3208,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2574,2911,3206,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1037,1601,1645,1703,1716,1718,1719,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1718,1720,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,614,1037,1645,1717,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3666,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3663,3664,3665,3666,3667,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3663,3664,3665,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4331,4332],[398,437,546,593,1601,1703,2574,2575,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4331,4332,4334,4335,4336,4337,4338,4339,4341],[398,546,593,1046,1703,2576,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4330,4331,4333,4341,4342],[398,546,593,651,1037,1046,1048,1688,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4330,4331,4335,4337,4338,4339,4340],[398,437,532,546,593,595,613,1645,1719,1722,1731,1732,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4351,4352,4548,4551,4552,4553,4554,4555,4556],[398,437,546,593,1703,1716,1759,2240,2266,2267,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,1664,1733,1760,2241,2242,2245,2246,2247,2248,2266,2268,2269,2271,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1760,2240,2241,2242,2245,2246,2247,2248,2265,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,607,609,1037,1046,1760,2241,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2243,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,1760,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,607,609,613,614,1037,1046,1646,1664,1733,1760,2240,2241,2270,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,623,651,1760,2239,2240,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,614,2240,2243,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1717,2240,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1645,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2244,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1675,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,1664,1675,1725,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590,4591,4592,4593,4594],[398,546,593,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4590],[398,546,593,3298,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1601,3223,3232,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3222,3223,3224,3232,3233,3234,3235,3236,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,3225,3226,3227,3228,3229,3230,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3222,3223,3224,3231,3232,3233,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3222,3231,3234,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3773,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3768,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3770,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3769,3770,3771,3772,3773,3774,3775,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3769,3771,3772,3774,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1590,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1591,1601,1603,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1591,1602,1603,1604,1605,1606,1607,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1602,1605,1606,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,613,1591,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1602,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3166,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,3188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,605,614,651,1646,1664,1733,3186,3187,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3166,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,3127,3167,3168,3189,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2536,2537,2566,2567,2568,3166,3169,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1046,1664,1733,3164,3166,3167,3170,3171,3172,3188,3189,3198,3199,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,1675,1734,2624,3166,3167,3168,3170,3171,3172,3188,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3167,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3249,3300,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2265,3249,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,3251,3299,3300,3301,3302,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3250,3251,3299,3300,3301,3302,3303,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3250,3251,3299,3300,3301,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3249,3298,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1717,3783,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1645,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3782,3783,3785,3786,3787,3788,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[247,398,546,593,1717,3782,3783,3784,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4025,4026,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,4028,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,4027,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4605],[546,593,1037,2574,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,1716,2574,2575,2576,4025,4026,4027,4029,4030,4031,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4317,4318],[398,546,593,1046,1611,1664,1733,2574,3130,4025,4026,4028,4029,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4315,4316,4318,4319,4320,4321,4322,4323,4324,4325,4326,4327,4328],[398,546,593,1037,1046,1611,1612,2574,3118,4025,4026,4027,4028,4029,4030,4031,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317],[398,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4315],[398,546,593,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320],[398,546,593,1601,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320,4321],[398,546,593,1037,1046,1646,1664,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4320],[398,546,593,1037,1046,1646,1664,1733,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317,4318,4320],[398,437,546,593,1601,1716,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4322],[398,546,593,4025,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4316,4317,4318,4320,4321],[437,546,593,2417,2586,2620,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2586,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1646,1664,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2586,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,2623,2624,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,1716,2587,2588,2589,2590,2591,2592,2593,2621,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,1664,1733,2588,2591,2592,2593,2622,2623,2625,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1733,2586,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1580,1589,1646,1664,1733,2587,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,3309,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1703,3309,3310,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314,4611],[398,546,593,3306,3308,3309,3310,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,613,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3305,3306,3307,3308,3311,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3305,3306,3307,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3594,3597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1716,3597,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,3594,3595,3596,3597,3598,3599,3600,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,3124,3125,3594,3595,3596,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,3790,3791,3792,3793,3794,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1589,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1645,1722,1723,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,598,651,1645,1722,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1048,1675,1688,1734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1675,1733,1734,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1646,1664,1675,1733,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1664,1733,1735,1736,1737,1738,1739,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1675,1734,1735,1736,1737,1738,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3122,3796,3797,3798,3799,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,3798,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3799,3800,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,532,546,593,3806,3807,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1703,2574,2575,2576,3796,3800,3801,3802,3803,3804,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,3796,3797,3798,3799,3801,3802,3803,3804,3805,3806,3808,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1037,1046,3796,3797,3798,3799,3800,3801,3802,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,2417,3120,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[437,546,593,2417,3114,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[546,593,1037,2572,2573,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,437,546,593,1601,1703,1716,2574,2575,2576,3114,3115,3126,3127,3128,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,1046,1664,1703,1733,2572,2574,2576,3126,3128,3129,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314],[398,546,593,651,1037,1046,1675,2574,3114,3115,3117,3118,3120,3121,3122,3123,3124,3125,4032,4033,4034,4035,4036,4037,4038,4039,4040,4041,4042,4043,4044,4045,4046,4047,4048,4049,4050,4051,4052,4053,4054,4055,4056,4057,4058,4059,4060,4061,4062,4063,4064,4065,4066,4067,4068,4069,4070,4071,4072,4073,4074,4075,4076,4077,4078,4079,4080,4081,4082,4083,4084,4085,4086,4087,4088,4089,4090,4091,4092,4093,4094,4095,4096,4097,4098,4099,4100,4101,4102,4103,4104,4105,4106,4107,4108,4109,4110,4111,4112,4113,4114,4115,4116,4117,4118,4119,4120,4121,4122,4123,4124,4125,4126,4127,4128,4129,4130,4131,4132,4133,4134,4135,4136,4137,4138,4139,4140,4141,4142,4143,4144,4145,4146,4147,4148,4149,4150,4151,4152,4153,4154,4155,4156,4157,4158,4159,4160,4161,4162,4163,4164,4165,4166,4167,4168,4169,4170,4171,4172,4173,4174,4175,4176,4177,4178,4179,4180,4181,4182,4183,4184,4185,4186,4187,4188,4189,4190,4191,4192,4193,4194,4195,4196,4197,4198,4199,4200,4201,4202,4203,4204,4205,4206,4207,4208,4209,4210,4211,4212,4213,4214,4215,4216,4217,4218,4219,4220,4221,4222,4223,4224,4225,4226,4227,4228,4229,4230,4231,4232,4233,4234,4235,4236,4237,4238,4239,4240,4241,4242,4243,4244,4245,4246,4247,4248,4249,4250,4251,4252,4253,4254,4255,4256,4257,4258,4259,4260,4261,4262,4263,4264,4265,4266,4267,4268,4269,4270,4271,4272,4273,4274,4275,4276,4277,4278,4279,4280,4281,4282,4283,4284,4285,4286,4287,4288,4289,4290,4291,4292,4293,4294,4295,4296,4297,4298,4299,4300,4301,4302,4303,4304,4305,4306,4307,4308,4309,4310,4311,4312,4313,4314]],"fileInfos":[{"version":"bcd24271a113971ba9eb71ff8cb01bc6b0f872a85c23fdbe5d93065b375933cd","affectsGlobalScope":true,"impliedFormat":1},{"version":"3f88bedbeb09c6f5a6645cb24c7c55f1aa22d19ae96c8e6959cbd8b85a707bc6","impliedFormat":1},{"version":"7fe93b39b810eadd916be8db880dd7f0f7012a5cc6ffb62de8f62a2117fa6f1f","impliedFormat":1},{"version":"bb0074cc08b84a2374af33d8bf044b80851ccc9e719a5e202eacf40db2c31600","impliedFormat":1},{"version":"1a7daebe4f45fb03d9ec53d60008fbf9ac45a697fdc89e4ce218bc94b94f94d6","impliedFormat":1},{"version":"f94b133a3cb14a288803be545ac2683e0d0ff6661bcd37e31aaaec54fc382aed","impliedFormat":1},{"version":"f59d0650799f8782fd74cf73c19223730c6d1b9198671b1c5b3a38e1188b5953","impliedFormat":1},{"version":"8a15b4607d9a499e2dbeed9ec0d3c0d7372c850b2d5f1fb259e8f6d41d468a84","impliedFormat":1},{"version":"d6b1eba8496bdd0eed6fc8a685768fe01b2da4a0388b5fe7df558290bffcf32f","affectsGlobalScope":true,"impliedFormat":1},{"version":"eadcffda2aa84802c73938e589b9e58248d74c59cb7fcbca6474e3435ac15504","affectsGlobalScope":true,"impliedFormat":1},{"version":"105ba8ff7ba746404fe1a2e189d1d3d2e0eb29a08c18dded791af02f29fb4711","affectsGlobalScope":true,"impliedFormat":1},{"version":"00343ca5b2e3d48fa5df1db6e32ea2a59afab09590274a6cccb1dbae82e60c7c","affectsGlobalScope":true,"impliedFormat":1},{"version":"ebd9f816d4002697cb2864bea1f0b70a103124e18a8cd9645eeccc09bdf80ab4","affectsGlobalScope":true,"impliedFormat":1},{"version":"2c1afac30a01772cd2a9a298a7ce7706b5892e447bb46bdbeef720f7b5da77ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"7b0225f483e4fa685625ebe43dd584bb7973bbd84e66a6ba7bbe175ee1048b4f","affectsGlobalScope":true,"impliedFormat":1},{"version":"c0a4b8ac6ce74679c1da2b3795296f5896e31c38e888469a8e0f99dc3305de60","affectsGlobalScope":true,"impliedFormat":1},{"version":"3084a7b5f569088e0146533a00830e206565de65cae2239509168b11434cd84f","affectsGlobalScope":true,"impliedFormat":1},{"version":"c5079c53f0f141a0698faa903e76cb41cd664e3efb01cc17a5c46ec2eb0bef42","affectsGlobalScope":true,"impliedFormat":1},{"version":"32cafbc484dea6b0ab62cf8473182bbcb23020d70845b406f80b7526f38ae862","affectsGlobalScope":true,"impliedFormat":1},{"version":"fca4cdcb6d6c5ef18a869003d02c9f0fd95df8cfaf6eb431cd3376bc034cad36","affectsGlobalScope":true,"impliedFormat":1},{"version":"b93ec88115de9a9dc1b602291b85baf825c85666bf25985cc5f698073892b467","affectsGlobalScope":true,"impliedFormat":1},{"version":"f5c06dcc3fe849fcb297c247865a161f995cc29de7aa823afdd75aaaddc1419b","affectsGlobalScope":true,"impliedFormat":1},{"version":"b77e16112127a4b169ef0b8c3a4d730edf459c5f25fe52d5e436a6919206c4d7","affectsGlobalScope":true,"impliedFormat":1},{"version":"fbffd9337146eff822c7c00acbb78b01ea7ea23987f6c961eba689349e744f8c","affectsGlobalScope":true,"impliedFormat":1},{"version":"a995c0e49b721312f74fdfb89e4ba29bd9824c770bbb4021d74d2bf560e4c6bd","affectsGlobalScope":true,"impliedFormat":1},{"version":"c7b3542146734342e440a84b213384bfa188835537ddbda50d30766f0593aff9","affectsGlobalScope":true,"impliedFormat":1},{"version":"ce6180fa19b1cccd07ee7f7dbb9a367ac19c0ed160573e4686425060b6df7f57","affectsGlobalScope":true,"impliedFormat":1},{"version":"3f02e2476bccb9dbe21280d6090f0df17d2f66b74711489415a8aa4df73c9675","affectsGlobalScope":true,"impliedFormat":1},{"version":"45e3ab34c1c013c8ab2dc1ba4c80c780744b13b5676800ae2e3be27ae862c40c","affectsGlobalScope":true,"impliedFormat":1},{"version":"805c86f6cca8d7702a62a844856dbaa2a3fd2abef0536e65d48732441dde5b5b","affectsGlobalScope":true,"impliedFormat":1},{"version":"e42e397f1a5a77994f0185fd1466520691456c772d06bf843e5084ceb879a0ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"f4c2b41f90c95b1c532ecc874bd3c111865793b23aebcc1c3cbbabcd5d76ffb0","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab26191cfad5b66afa11b8bf935ef1cd88fabfcb28d30b2dfa6fad877d050332","affectsGlobalScope":true,"impliedFormat":1},{"version":"2088bc26531e38fb05eedac2951480db5309f6be3fa4a08d2221abb0f5b4200d","affectsGlobalScope":true,"impliedFormat":1},{"version":"cb9d366c425fea79716a8fb3af0d78e6b22ebbab3bd64d25063b42dc9f531c1e","affectsGlobalScope":true,"impliedFormat":1},{"version":"500934a8089c26d57ebdb688fc9757389bb6207a3c8f0674d68efa900d2abb34","affectsGlobalScope":true,"impliedFormat":1},{"version":"689da16f46e647cef0d64b0def88910e818a5877ca5379ede156ca3afb780ac3","affectsGlobalScope":true,"impliedFormat":1},{"version":"bc21cc8b6fee4f4c2440d08035b7ea3c06b3511314c8bab6bef7a92de58a2593","affectsGlobalScope":true,"impliedFormat":1},{"version":"7ca53d13d2957003abb47922a71866ba7cb2068f8d154877c596d63c359fed25","affectsGlobalScope":true,"impliedFormat":1},{"version":"54725f8c4df3d900cb4dac84b64689ce29548da0b4e9b7c2de61d41c79293611","affectsGlobalScope":true,"impliedFormat":1},{"version":"e5594bc3076ac29e6c1ebda77939bc4c8833de72f654b6e376862c0473199323","affectsGlobalScope":true,"impliedFormat":1},{"version":"2f3eb332c2d73e729f3364fcc0c2b375e72a121e8157d25a82d67a138c83a95c","affectsGlobalScope":true,"impliedFormat":1},{"version":"6f4427f9642ce8d500970e4e69d1397f64072ab73b97e476b4002a646ac743b1","affectsGlobalScope":true,"impliedFormat":1},{"version":"48915f327cd1dea4d7bd358d9dc7732f58f9e1626a29cc0c05c8c692419d9bb7","affectsGlobalScope":true,"impliedFormat":1},{"version":"b7bf9377723203b5a6a4b920164df22d56a43f593269ba6ae1fdc97774b68855","affectsGlobalScope":true,"impliedFormat":1},{"version":"db9709688f82c9e5f65a119c64d835f906efe5f559d08b11642d56eb85b79357","affectsGlobalScope":true,"impliedFormat":1},{"version":"4b25b8c874acd1a4cf8444c3617e037d444d19080ac9f634b405583fd10ce1f7","affectsGlobalScope":true,"impliedFormat":1},{"version":"37be57d7c90cf1f8112ee2636a068d8fd181289f82b744160ec56a7dc158a9f5","affectsGlobalScope":true,"impliedFormat":1},{"version":"a917a49ac94cd26b754ab84e113369a75d1a47a710661d7cd25e961cc797065f","affectsGlobalScope":true,"impliedFormat":1},{"version":"1ce14b81c5cc821994aa8ec1d42b220dd41b27fcc06373bce3958af7421b77d4","affectsGlobalScope":true,"impliedFormat":1},{"version":"b3a048b3e9302ef9a34ef4ebb9aecfb28b66abb3bce577206a79fee559c230da","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d6d51a5118d000ed3bfe6e1dd1335bebfff3fef23cd2af2f84a24d30f90cc90","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d8dedbec739bc79642c1e96e9bfc0b83b25b104a0486aebf016fc7b85b39f48","impliedFormat":1},{"version":"e89535c3ec439608bcd0f68af555d0e5ddf121c54abe69343549718bd7506b9c","impliedFormat":1},{"version":"622a984b60c294ffb2f9152cf1d4d12e91d2b733d820eec949cf54d63a3c1025","impliedFormat":1},{"version":"81aae92abdeaccd9c1723cef39232c90c1aed9d9cf199e6e2a523b7d8e058a11","impliedFormat":1},{"version":"a63a6c6806a1e519688ef7bd8ca57be912fc0764485119dbd923021eb4e79665","impliedFormat":1},{"version":"75b57b109d774acca1e151df21cf5cb54c7a1df33a273f0457b9aee4ebd36fb9","impliedFormat":1},{"version":"073ca26c96184db9941b5ec0ddea6981c9b816156d9095747809e524fdd90e35","impliedFormat":1},{"version":"e41d17a2ec23306d953cda34e573ed62954ca6ea9b8c8b74e013d07a6886ce47","impliedFormat":1},{"version":"241bd4add06f06f0699dcd58f3b334718d85e3045d9e9d4fa556f11f4d1569c1","impliedFormat":1},{"version":"2ae3787e1498b20aad1b9c2ee9ea517ec30e89b70d242d8e3e52d1e091039695","impliedFormat":1},{"version":"c7c72c4cffb1bc83617eefed71ed68cc89df73cab9e19507ccdecb3e72b4967e","affectsGlobalScope":true,"impliedFormat":1},{"version":"b8bff8a60af0173430b18d9c3e5c443eaa3c515617210c0c7b3d2e1743c19ecb","impliedFormat":1},{"version":"38b38db08e7121828294dec10957a7a9ff263e33e2a904b346516d4a4acca482","impliedFormat":1},{"version":"a76ebdf2579e68e4cfe618269c47e5a12a4e045c2805ed7f7ab37af8daa6b091","impliedFormat":1},{"version":"8a2aaea564939c22be05d665cc955996721bad6d43148f8fa21ae8f64afecd37","impliedFormat":1},{"version":"e59d36b7b6e8ba2dd36d032a5f5c279d2460968c8b4e691ca384f118fb09b52a","impliedFormat":1},{"version":"e96885c0684c9042ec72a9a43ef977f6b4b4a2728f4b9e737edcbaa0c74e5bf6","impliedFormat":1},{"version":"95950a187596e206d32d5d9c7b932901088c65ed8f9040e614aa8e321e0225ef","impliedFormat":1},{"version":"89e061244da3fc21b7330f4bd32f47c1813dd4d7f1dc3d0883d88943f035b993","impliedFormat":1},{"version":"e46558c2e04d06207b080138678020448e7fc201f3d69c2601b0d1456105f29a","impliedFormat":1},{"version":"71549375db52b1163411dba383b5f4618bdf35dc57fa327a1c7d135cf9bf67d1","impliedFormat":1},{"version":"7e6b2d61d6215a4e82ea75bc31a80ebb8ad0c2b37a60c10c70dd671e8d9d6d5d","impliedFormat":1},{"version":"78bea05df2896083cca28ed75784dde46d4b194984e8fc559123b56873580a23","impliedFormat":1},{"version":"5dd04ced37b7ea09f29d277db11f160df7fd73ba8b9dba86cb25552e0653a637","impliedFormat":1},{"version":"f74b81712e06605677ae1f061600201c425430151f95b5ef4d04387ad7617e6a","impliedFormat":1},{"version":"9a72847fcf4ac937e352d40810f7b7aec7422d9178451148296cf1aa19467620","impliedFormat":1},{"version":"3ae18f60e0b96fa1e025059b7d25b3247ba4dcb5f4372f6d6e67ce2adac74eac","impliedFormat":1},{"version":"2b9260f44a2e071450ae82c110f5dc8f330c9e5c3e85567ed97248330f2bf639","impliedFormat":1},{"version":"4f196e13684186bda6f5115fc4677a87cf84a0c9c4fc17b8f51e0984f3697b6d","impliedFormat":1},{"version":"61419f2c5822b28c1ea483258437c1faab87d00c6f84481aa22afb3380d8e9a4","impliedFormat":1},{"version":"64479aee03812264e421c0bf5104a953ca7b02740ba80090aead1330d0effe91","impliedFormat":1},{"version":"0521108c9f8ddb17654a0a54dae6ba9667c99eddccfd6af5748113e022d1c37a","impliedFormat":1},{"version":"c5570e504be103e255d80c60b56c367bf45d502ca52ee35c55dec882f6563b5c","impliedFormat":1},{"version":"ee764e6e9a7f2b987cc1a2c0a9afd7a8f4d5ebc4fdb66ad557a7f14a8c2bd320","impliedFormat":1},{"version":"0520b5093712c10c6ef23b5fea2f833bf5481771977112500045e5ea7e8e2b69","impliedFormat":1},{"version":"5c3cf26654cf762ac4d7fd7b83f09acfe08eef88d2d6983b9a5a423cb4004ca3","impliedFormat":1},{"version":"e60fa19cf7911c1623b891155d7eb6b7e844e9afdf5738e3b46f3b687730a2bd","impliedFormat":1},{"version":"b1fd72ff2bb0ba91bb588f3e5329f8fc884eb859794f1c4657a2bfa122ae54d0","impliedFormat":1},{"version":"6cf42a4f3cfec648545925d43afaa8bb364ac10a839ffed88249da109361b275","impliedFormat":1},{"version":"d7058e75920120b142a9d57be25562a3cd9a936269fd52908505f530105f2ec4","impliedFormat":1},{"version":"6df52b70d7f7702202f672541a5f4a424d478ee5be51a9d37b8ccbe1dbf3c0f2","impliedFormat":1},{"version":"0ca7f997e9a4d8985e842b7c882e521b6f63233c4086e9fe79dd7a9dc4742b5e","impliedFormat":1},{"version":"91046b5c6b55d3b194c81fd4df52f687736fad3095e9d103ead92bb64dc160ee","impliedFormat":1},{"version":"db5704fdad56c74dfc5941283c1182ed471bd17598209d3ac4a49faa72e43cfc","impliedFormat":1},{"version":"758e8e89559b02b81bc0f8fd395b17ad5aff75490c862cbe369bb1a3d1577c40","impliedFormat":1},{"version":"2ee64342c077b1868f1834c063f575063051edd6e2964257d34aad032d6b657c","impliedFormat":1},{"version":"6f6b4b3d670b6a5f0e24ea001c1b3d36453c539195e875687950a178f1730fa7","impliedFormat":1},{"version":"a472a1d3f25ce13a1d44911cd3983956ac040ce2018e155435ea34afb25f864c","impliedFormat":1},{"version":"b48b83a86dd9cfe36f8776b3ff52fcd45b0e043c0538dc4a4b149ba45fe367b9","impliedFormat":1},{"version":"792de5c062444bd2ee0413fb766e57e03cce7cdaebbfc52fc0c7c8e95069c96b","impliedFormat":1},{"version":"a79e3e81094c7a04a885bad9b049c519aace53300fb8a0fe4f26727cb5a746ce","impliedFormat":1},{"version":"93181bac0d90db185bb730c95214f6118ae997fe836a98a49664147fbcaf1988","impliedFormat":1},{"version":"8a4e89564d8ea66ad87ee3762e07540f9f0656a62043c910d819b4746fc429c5","impliedFormat":1},{"version":"b9011d99942889a0f95e120d06b698c628b0b6fdc3e6b7ecb459b97ed7d5bcc6","impliedFormat":1},{"version":"4d639cbbcc2f8f9ce6d55d5d503830d6c2556251df332dc5255d75af53c8a0e7","impliedFormat":1},{"version":"cdb48277f600ab5f429ecf1c5ea046683bc6b9f73f3deab9a100adac4b34969c","impliedFormat":1},{"version":"75be84956a29040a1afbe864c0a7a369dfdb739380072484eff153905ef867ee","impliedFormat":1},{"version":"b06b4adc2ae03331a92abd1b19af8eb91ec2bf8541747ee355887a167d53145e","impliedFormat":1},{"version":"c54166a85bd60f86d1ebb90ce0117c0ecb850b8a33b366691629fdf26f1bbbd8","impliedFormat":1},{"version":"0d417c15c5c635384d5f1819cc253a540fe786cc3fda32f6a2ae266671506a21","impliedFormat":1},{"version":"80f23f1d60fbed356f726b3b26f9d348dddbb34027926d10d59fad961e70a730","impliedFormat":1},{"version":"cb59317243a11379a101eb2f27b9df1022674c3df1df0727360a0a3f963f523b","impliedFormat":1},{"version":"cc20bb2227dd5de0aab0c8d697d1572f8000550e62c7bf5c92f212f657dd88c5","impliedFormat":1},{"version":"06b8a7d46195b6b3980e523ef59746702fd210b71681a83a5cf73799623621f9","impliedFormat":1},{"version":"860e4405959f646c101b8005a191298b2381af8f33716dc5f42097e4620608f8","impliedFormat":1},{"version":"f7e32adf714b8f25d3c1783473abec3f2e82d5724538d8dcf6f51baaaff1ca7a","impliedFormat":1},{"version":"d0da80c845999a16c24d0783033fb5366ada98df17867c98ad433ede05cd87fd","impliedFormat":1},{"version":"bfbf80f9cd4558af2d7b2006065340aaaced15947d590045253ded50aabb9bc5","impliedFormat":1},{"version":"fd9a991b51870325e46ebb0e6e18722d313f60cd8e596e645ec5ac15b96dbf4e","impliedFormat":1},{"version":"c3bd2b94e4298f81743d92945b80e9b56c1cdfb2bef43c149b7106a2491b1fc9","impliedFormat":1},{"version":"a246cce57f558f9ebaffd55c1e5673da44ea603b4da3b2b47eb88915d30a9181","impliedFormat":1},{"version":"d993eacc103c5a065227153c9aae8acea3a4322fe1a169ee7c70b77015bf0bb2","impliedFormat":1},{"version":"fc2b03d0c042aa1627406e753a26a1eaad01b3c496510a78016822ef8d456bb6","impliedFormat":1},{"version":"063c7ebbe756f0155a8b453f410ca6b76ffa1bbc1048735bcaf9c7c81a1ce35f","impliedFormat":1},{"version":"314e402cd481370d08f63051ae8b8c8e6370db5ee3b8820eeeaaf8d722a6dac6","impliedFormat":1},{"version":"9669075ac38ce36b638b290ba468233980d9f38bdc62f0519213b2fd3e2552ec","impliedFormat":1},{"version":"4d123de012c24e2f373925100be73d50517ac490f9ed3578ac82d0168bfbd303","impliedFormat":1},{"version":"656c9af789629aa36b39092bee3757034009620439d9a39912f587538033ce28","impliedFormat":1},{"version":"3ac3f4bdb8c0905d4c3035d6f7fb20118c21e8a17bee46d3735195b0c2a9f39f","impliedFormat":1},{"version":"1f453e6798ed29c86f703e9b41662640d4f2e61337007f27ac1c616f20093f69","impliedFormat":1},{"version":"af43b7871ff21c62bf1a54ec5c488e31a8d3408d5b51ff2e9f8581b6c55f2fc7","impliedFormat":1},{"version":"70550511d25cbb0b6a64dcac7fffc3c1397fd4cbeb6b23ccc7f9b794ab8a6954","impliedFormat":1},{"version":"af0fbf08386603a62f2a78c42d998c90353b1f1d22e05a384545f7accf881e0a","impliedFormat":1},{"version":"cefc20054d20b85b534206dbcedd509bb74f87f3d8bc45c58c7be3a76caa45e1","impliedFormat":1},{"version":"ad6eee4877d0f7e5244d34bc5026fd6e9cf8e66c5c79416b73f9f6ebf132f924","impliedFormat":1},{"version":"4888fd2bcfee9a0ce89d0df860d233e0cee8ee9c479b6bd5a5d5f9aae98342fe","impliedFormat":1},{"version":"f4749c102ced952aa6f40f0b579865429c4869f6d83df91000e98005476bee87","impliedFormat":1},{"version":"56654d2c5923598384e71cb808fac2818ca3f07dd23bb018988a39d5e64f268b","impliedFormat":1},{"version":"8b6719d3b9e65863da5390cb26994602c10a315aa16e7d70778a63fee6c4c079","impliedFormat":1},{"version":"05f56cd4b929977d18df8f3d08a4c929a2592ef5af083e79974b20a063f30940","impliedFormat":1},{"version":"547d3c406a21b30e2b78629ecc0b2ddaf652d9e0bdb2d59ceebce5612906df33","impliedFormat":1},{"version":"b3a4f9385279443c3a5568ec914a9492b59a723386161fd5ef0619d9f8982f97","impliedFormat":1},{"version":"3fe66aba4fbe0c3ba196a4f9ed2a776fe99dc4d1567a558fb11693e9fcc4e6ed","impliedFormat":1},{"version":"140eef237c7db06fc5adcb5df434ee21e81ee3a6fd57e1a75b8b3750aa2df2d8","impliedFormat":1},{"version":"0944ec553e4744efae790c68807a461720cff9f3977d4911ac0d918a17c9dd99","impliedFormat":1},{"version":"cb46b38d5e791acaa243bf342b8b5f8491639847463ac965b93896d4fb0af0d9","impliedFormat":1},{"version":"7c7d9e116fe51100ff766703e6b5e4424f51ad8977fe474ddd8d0959aa6de257","impliedFormat":1},{"version":"af70a2567e586be0083df3938b6a6792e6821363d8ef559ad8d721a33a5bcdaf","impliedFormat":1},{"version":"006cff3a8bcb92d77953f49a94cd7d5272fef4ab488b9052ef82b6a1260d870b","impliedFormat":1},{"version":"7d44bfdc8ee5e9af70738ff652c622ae3ad81815e63ab49bdc593d34cb3a68e5","impliedFormat":1},{"version":"339814517abd4dbc7b5f013dfd3b5e37ef0ea914a8bbe65413ecffd668792bc6","impliedFormat":1},{"version":"34d5bc0a6958967ec237c99f980155b5145b76e6eb927c9ffc57d8680326b5d8","impliedFormat":1},{"version":"9eae79b70c9d8288032cbe1b21d0941f6bd4f315e14786b2c1d10bccc634e897","impliedFormat":1},{"version":"18ce015ed308ea469b13b17f99ce53bbb97975855b2a09b86c052eefa4aa013a","impliedFormat":1},{"version":"5a931bc4106194e474be141e0bc1046629510dc95b9a0e4b02a3783847222965","impliedFormat":1},{"version":"5e5f371bf23d5ced2212a5ff56675aefbd0c9b3f4d4fdda1b6123ac6e28f058c","impliedFormat":1},{"version":"907c17ad5a05eecb29b42b36cc8fec6437be27cc4986bb3a218e4f74f606911c","impliedFormat":1},{"version":"ce60a562cd2a92f37a88f2ddd99a3abfbc5848d7baf38c48fb8d3243701fcb75","impliedFormat":1},{"version":"a726ad2d0a98bfffbe8bc1cd2d90b6d831638c0adc750ce73103a471eb9a891c","impliedFormat":1},{"version":"f44c0c8ce58d3dacac016607a1a90e5342d830ea84c48d2e571408087ae55894","impliedFormat":1},{"version":"75a315a098e630e734d9bc932d9841b64b30f7a349a20cf4717bf93044eff113","impliedFormat":1},{"version":"9131d95e32b3d4611d4046a613e022637348f6cebfe68230d4e81b691e4761a1","impliedFormat":1},{"version":"b03aa292cfdcd4edc3af00a7dbd71136dd067ec70a7536b655b82f4dd444e857","impliedFormat":1},{"version":"b6e2b0448ced813b8c207810d96551a26e7d7bb73255eea4b9701698f78846d6","impliedFormat":1},{"version":"8ae10cd85c1bd94d2f2d17c4cbd25c068a4b2471c70c2d96434239f97040747a","impliedFormat":1},{"version":"9ed5b799c50467b0c9f81ddf544b6bcda3e34d92076d6cab183c84511e45c39f","impliedFormat":1},{"version":"b4fa87cc1833839e51c49f20de71230e259c15b2c9c3e89e4814acc1d1ef10de","impliedFormat":1},{"version":"e90ac9e4ac0326faa1bc39f37af38ace0f9d4a655cd6d147713c653139cf4928","impliedFormat":1},{"version":"ea27110249d12e072956473a86fd1965df8e1be985f3b686b4e277afefdde584","impliedFormat":1},{"version":"8776a368617ce51129b74db7d55c3373dadcce5d0701e61d106e99998922a239","impliedFormat":1},{"version":"5666075052877fe2fdddd5b16de03168076cf0f03fbca5c1d4a3b8f43cba570c","impliedFormat":1},{"version":"9108ab5af05418f599ab48186193b1b07034c79a4a212a7f73535903ba4ca249","impliedFormat":1},{"version":"bb4e2cdcadf9c9e6ee2820af23cee6582d47c9c9c13b0dca1baaffe01fbbcb5f","impliedFormat":1},{"version":"6e30d0b5a1441d831d19fe02300ab3d83726abd5141cbcc0e2993fa0efd33db4","impliedFormat":1},{"version":"423f28126b2fc8d8d6fa558035309000a1297ed24473c595b7dec52e5c7ebae5","impliedFormat":1},{"version":"fb30734f82083d4790775dae393cd004924ebcbfde49849d9430bf0f0229dd16","impliedFormat":1},{"version":"2c92b04a7a4a1cd9501e1be338bf435738964130fb2ad5bd6c339ee41224ac4c","impliedFormat":1},{"version":"c5c5f0157b41833180419dacfbd2bcce78fb1a51c136bd4bcba5249864d8b9b5","impliedFormat":1},{"version":"02ae43d5bae42efcd5a00d3923e764895ce056bca005a9f4e623aa6b4797c8af","impliedFormat":1},{"version":"db6e01f17012a9d7b610ae764f94a1af850f5d98c9c826ad61747dca0fb800bd","impliedFormat":1},{"version":"8a44b424edee7bb17dc35a558cc15f92555f14a0441205613e0e50452ab3a602","impliedFormat":1},{"version":"24a00d0f98b799e6f628373249ece352b328089c3383b5606214357e9107e7d5","impliedFormat":1},{"version":"33637e3bc64edd2075d4071c55d60b32bdb0d243652977c66c964021b6fc8066","impliedFormat":1},{"version":"0f0ad9f14dedfdca37260931fac1edf0f6b951c629e84027255512f06a6ebc4c","impliedFormat":1},{"version":"16ad86c48bf950f5a480dc812b64225ca4a071827d3d18ffc5ec1ae176399e36","impliedFormat":1},{"version":"8cbf55a11ff59fd2b8e39a4aa08e25c5ddce46e3af0ed71fb51610607a13c505","impliedFormat":1},{"version":"d5bc4544938741f5daf8f3a339bfbf0d880da9e89e79f44a6383aaf056fe0159","impliedFormat":1},{"version":"97f9169882d393e6f303f570168ca86b5fe9aab556e9a43672dae7e6bb8e6495","impliedFormat":1},{"version":"7c9adb3fcd7851497818120b7e151465406e711d6a596a71b807f3a17853cb58","impliedFormat":1},{"version":"6752d402f9282dd6f6317c8c048aaaac27295739a166eed27e00391b358fed9a","impliedFormat":1},{"version":"9fd7466b77020847dbc9d2165829796bf7ea00895b2520ff3752ffdcff53564b","impliedFormat":1},{"version":"fbfc12d54a4488c2eb166ed63bab0fb34413e97069af273210cf39da5280c8d6","impliedFormat":1},{"version":"85a84240002b7cf577cec637167f0383409d086e3c4443852ca248fc6e16711e","impliedFormat":1},{"version":"84794e3abd045880e0fadcf062b648faf982aa80cfc56d28d80120e298178626","impliedFormat":1},{"version":"053d8b827286a16a669a36ffc8ccc8acdf8cc154c096610aa12348b8c493c7b8","impliedFormat":1},{"version":"3cce4ce031710970fe12d4f7834375f5fd455aa129af4c11eb787935923ff551","impliedFormat":1},{"version":"8f62cbd3afbd6a07bb8c934294b6bfbe437021b89e53a4da7de2648ecfc7af25","impliedFormat":1},{"version":"62c3621d34fb2567c17a2c4b89914ebefbfbd1b1b875b070391a7d4f722e55dc","impliedFormat":1},{"version":"c05ac811542e0b59cb9c2e8f60e983461f0b0e39cea93e320fad447ff8e474f3","impliedFormat":1},{"version":"8e7a5b8f867b99cc8763c0b024068fb58e09f7da2c4810c12833e1ca6eb11c4f","impliedFormat":1},{"version":"132351cbd8437a463757d3510258d0fa98fd3ebef336f56d6f359cf3e177a3ce","impliedFormat":1},{"version":"df877050b04c29b9f8409aa10278d586825f511f0841d1ec41b6554f8362092b","impliedFormat":1},{"version":"33d1888c3c27d3180b7fd20bac84e97ecad94b49830d5dd306f9e770213027d1","impliedFormat":1},{"version":"ee942c58036a0de88505ffd7c129f86125b783888288c2389330168677d6347f","impliedFormat":1},{"version":"a3f317d500c30ea56d41501632cdcc376dae6d24770563a5e59c039e1c2a08ec","impliedFormat":1},{"version":"eb21ddc3a8136a12e69176531197def71dc28ffaf357b74d4bf83407bd845991","impliedFormat":1},{"version":"0c1651a159995dfa784c57b4ea9944f16bdf8d924ed2d8b3db5c25d25749a343","impliedFormat":1},{"version":"aaa13958e03409d72e179b5d7f6ec5c6cc666b7be14773ae7b6b5ee4921e52db","impliedFormat":1},{"version":"0a86e049843ad02977a94bb9cdfec287a6c5a0a4b6b5391a6648b1a122072c5a","impliedFormat":1},{"version":"40f06693e2e3e58526b713c937895c02e113552dc8ba81ecd49cdd9596567ddb","impliedFormat":1},{"version":"4ed5e1992aedb174fb8f5aa8796aa6d4dcb8bd819b4af1b162a222b680a37fa0","impliedFormat":1},{"version":"d7f4bd46a8b97232ea6f8c28012b8d2b995e55e729d11405f159d3e00c51420a","impliedFormat":1},{"version":"d604d413aff031f4bfbdae1560e54ebf503d374464d76d50a2c6ded4df525712","impliedFormat":1},{"version":"e4f4f9cf1e3ac9fd91ada072e4d428ecbf0aa6dc57138fb797b8a0ca3a1d521c","impliedFormat":1},{"version":"12bfd290936824373edda13f48a4094adee93239b9a73432db603127881a300d","impliedFormat":1},{"version":"340ceb3ea308f8e98264988a663640e567c553b8d6dc7d5e43a8f3b64f780374","impliedFormat":1},{"version":"c5a769564e530fba3ec696d0a5cff1709b9095a0bdf5b0826d940d2fc9786413","impliedFormat":1},{"version":"7124ef724c3fc833a17896f2d994c368230a8d4b235baed39aa8037db31de54f","impliedFormat":1},{"version":"5de1c0759a76e7710f76899dcae601386424eab11fb2efaf190f2b0f09c3d3d3","impliedFormat":1},{"version":"9c5ee8f7e581f045b6be979f062a61bf076d362bf89c7f966b993a23424e8b0d","impliedFormat":1},{"version":"1a11df987948a86aa1ec4867907c59bdf431f13ed2270444bf47f788a5c7f92d","impliedFormat":1},{"version":"8018dd2e95e7ce6e613ddd81672a54532614dc745520a2f9e3860ff7fb1be0ca","impliedFormat":1},{"version":"b756781cd40d465da57d1fc6a442c34ae61fe8c802d752aace24f6a43fedacee","impliedFormat":1},{"version":"0fe76167c87289ea094e01616dcbab795c11b56bad23e1ef8aba9aa37e93432a","impliedFormat":1},{"version":"3a45029dba46b1f091e8dc4d784e7be970e209cd7d4ff02bd15270a98a9ba24b","impliedFormat":1},{"version":"032c1581f921f8874cf42966f27fd04afcabbb7878fa708a8251cac5415a2a06","impliedFormat":1},{"version":"69c68ed9652842ce4b8e495d63d2cd425862104c9fb7661f72e7aa8a9ef836f8","impliedFormat":1},{"version":"0e704ee6e9fd8b6a5a7167886f4d8915f4bc22ed79f19cb7b32bd28458f50643","impliedFormat":1},{"version":"06f62a14599a68bcde148d1efd60c2e52e8fa540cc7dcfa4477af132bb3de271","impliedFormat":1},{"version":"904a96f84b1bcee9a7f0f258d17f8692e6652a0390566515fe6741a5c6db8c1c","impliedFormat":1},{"version":"11f19ce32d21222419cecab448fa335017ebebf4f9e5457c4fa9df42fa2dcca7","impliedFormat":1},{"version":"2e8ee2cbb5e9159764e2189cf5547aebd0e6b0d9a64d479397bb051cd1991744","impliedFormat":1},{"version":"1b0471d75f5adb7f545c1a97c02a0f825851b95fe6e069ac6ecaa461b8bb321d","impliedFormat":1},{"version":"1d157c31a02b1e5cca9bc495b3d8d39f4b42b409da79f863fb953fbe3c7d4884","impliedFormat":1},{"version":"07baaceaec03d88a4b78cb0651b25f1ae0322ac1aa0b555ae3749a79a41cba86","impliedFormat":1},{"version":"619a132f634b4ebe5b4b4179ea5870f62f2cb09916a25957bff17b408de8b56d","impliedFormat":1},{"version":"f60fa446a397eb1aead9c4e568faf2df8068b4d0306ebc075fb4be16ed26b741","impliedFormat":1},{"version":"f3cb784be4d9e91f966a0b5052a098d9b53b0af0d341f690585b0cc05c6ca412","impliedFormat":1},{"version":"350f63439f8fe2e06c97368ddc7fb6d6c676d54f59520966f7dbbe6a4586014e","impliedFormat":1},{"version":"eba613b9b357ac8c50a925fa31dc7e65ff3b95a07efbaa684b624f143d8d34ba","impliedFormat":1},{"version":"45b74185005ed45bec3f07cac6e4d68eaf02ead9ff5a66721679fb28020e5e7c","impliedFormat":1},{"version":"0f6199602df09bdb12b95b5434f5d7474b1490d2cd8cc036364ab3ba6fd24263","impliedFormat":1},{"version":"c8ca7fd9ec7a3ec82185bfc8213e4a7f63ae748fd6fced931741d23ef4ea3c0f","impliedFormat":1},{"version":"5c6a8a3c2a8d059f0592d4eab59b062210a1c871117968b10797dee36d991ef7","impliedFormat":1},{"version":"ad77fd25ece8e09247040826a777dc181f974d28257c9cd5acb4921b51967bd8","impliedFormat":1},{"version":"795a08ae4e193f345073b49f68826ab6a9b280400b440906e4ec5c237ae777e6","impliedFormat":1},{"version":"8153df63cf65122809db17128e5918f59d6bb43a371b5218f4430c4585f64085","impliedFormat":1},{"version":"a8150bc382dd12ce58e00764d2366e1d59a590288ee3123af8a4a2cb4ef7f9df","impliedFormat":1},{"version":"5adfaf2f9f33957264ad199a186456a4676b2724ed700fc313ff945d03372169","impliedFormat":1},{"version":"d5c41a741cd408c34cb91f84468f70e9bda3dfeabf33251a61039b3cdb8b22d8","impliedFormat":1},{"version":"c91d3f9753a311284e76cdcb348cbb50bca98733336ec726b54d77b7361b34de","impliedFormat":1},{"version":"cbaf4a4aa8a8c02aa681c5870d5c69127974de29b7e01df570edec391a417959","impliedFormat":1},{"version":"c7135e329a18b0e712378d5c7bc2faec6f5ab0e955ea0002250f9e232af8b3e4","impliedFormat":1},{"version":"340a45cd77b41d8a6deda248167fa23d3dc67ec798d411bd282f7b3d555b1695","impliedFormat":1},{"version":"fae330f86bc10db6841b310f32367aaa6f553036a3afc426e0389ddc5566cd74","impliedFormat":1},{"version":"cf25d45c02d5fd5d7adb16230a0e1d6715441eef5c0a79a21bfeaa9bbc058939","impliedFormat":1},{"version":"54c3822eaf6436f2eddc92dd6e410750465aba218adbf8ce5d488d773919ec01","impliedFormat":1},{"version":"99d99a765426accf8133737843fb024a154dc6545fc0ffbba968a7c0b848959d","impliedFormat":1},{"version":"c782c5fd5fa5491c827ecade05c3af3351201dd1c7e77e06711c8029b7a9ee4d","impliedFormat":1},{"version":"883d2104e448bb351c49dd9689a7e8117b480b614b2622732655cef03021bf6d","impliedFormat":1},{"version":"d9b00ee2eca9b149663fdba1c1956331841ae296ee03eaaff6c5becbc0ff1ea8","impliedFormat":1},{"version":"09a7e04beb0547c43270b327c067c85a4e2154372417390731dfe092c4350998","impliedFormat":1},{"version":"eee530aaa93e9ec362e3941ee8355e2d073c7b21d88c2af4713e3d701dab8fef","impliedFormat":1},{"version":"28d47319b97dbeee9130b78eae03b2061d46dedbf92b0d9de13ed7ab8399ccd0","impliedFormat":1},{"version":"8b8b92781a6bf150f9ee83f3d8ee278b6cdb98b8308c7ab3413684fc5d9078ef","impliedFormat":1},{"version":"7a0e4cd92545ad03910fd019ae9838718643bd4dde39881c745f236914901dfa","impliedFormat":1},{"version":"c99ebd20316217e349004ee1a0bc74d32d041fb6864093f10f31984c737b8cad","impliedFormat":1},{"version":"6f622e7f054f5ab86258362ac0a64a2d6a27f1e88732d6f5f052f422e08a70e7","impliedFormat":1},{"version":"d62d2ef93ceeb41cf9dfab25989a1e5f9ca5160741aac7f1453c69a6c14c69be","impliedFormat":1},{"version":"1491e80d72873fc586605283f2d9056ee59b166333a769e64378240df130d1c9","impliedFormat":1},{"version":"c32c073d389cfaa3b3e562423e16c2e6d26b8edebbb7d73ccffff4aa66f2171d","impliedFormat":1},{"version":"eca72bf229eecadb63e758613c62fab13815879053539a22477d83a48a21cd73","impliedFormat":1},{"version":"633db46fd1765736409a4767bfc670861468dde60dbb9a501fba4c1b72f8644d","impliedFormat":1},{"version":"689390db63cb282e6d0e5ce9b8f1ec2ec0912d0e2e6dac7235699a15ad17d339","impliedFormat":1},{"version":"f2ee748883723aa9325e5d7f30fce424f6a786706e1b91a5a55237c78ee89c4a","impliedFormat":1},{"version":"d928324d17146fce30b99a28d1d6b48648feac72bbd23641d3ce5ac34aefdfee","impliedFormat":1},{"version":"142f5190d730259339be1433931c0eb31ae7c7806f4e325f8a470bd9221b6533","impliedFormat":1},{"version":"c33a88f2578e8df2fdf36c6a0482bbee615eb3234c8f084ba31a9a96bd306b7f","impliedFormat":1},{"version":"22cca068109eb0e6b4f8acc3fe638d1e6ac277e2044246438763319792b546a1","impliedFormat":1},{"version":"8776e64e6165838ac152fa949456732755b0976d1867ae5534ce248f0ccd7f41","impliedFormat":1},{"version":"66cd33c4151ea27f6e17c6071652eadde9da1b3637dae65fd060212211c695ce","impliedFormat":1},{"version":"5c4c5b49bbb01828402bb04af1d71673b18852c11b7e95bfd5cf4c3d80d352c8","impliedFormat":1},{"version":"7030df3d920343df00324df59dc93a959a33e0f4940af3fefef8c07b7ee329bf","impliedFormat":1},{"version":"a96bc00e0c356e29e620eaec24a56d6dd7f4e304feefcc99066a1141c6fe05a7","impliedFormat":1},{"version":"d12cc0e5b09943c4cd0848f787eb9d07bf78b60798e4588c50582db9d4decc70","impliedFormat":1},{"version":"53b094f1afe442490555eeeb0384fc1ceb487560c83e31f9c64fb934c2dccd94","impliedFormat":1},{"version":"19c3760af3cbc9da99d5b7763b9e33aaf8d018bc2ed843287b7ff4343adf4634","impliedFormat":1},{"version":"9d1e38aeb76084848d2fcd39b458ec88246de028c0f3f448b304b15d764b23d2","impliedFormat":1},{"version":"d406da1eccf18cec56fd29730c24af69758fe3ff49c4f94335e797119cbc0554","impliedFormat":1},{"version":"4898c93890a136da9156c75acd1a80a941a961b3032a0cf14e1fa09a764448b7","impliedFormat":1},{"version":"f5d7a845e3e1c6c27351ea5f358073d0b0681537a2da6201fab254aa434121d3","impliedFormat":1},{"version":"9ddf8e9069327faa75d20135cab675779844f66590249769c3d35dd2a38c2ba9","impliedFormat":1},{"version":"d7c30f0abfe9e197e376b016086cf66b2ffb84015139963f37301ed0da9d3d0d","impliedFormat":1},{"version":"ff75bba0148f07775bcb54bf4823421ed4ebdb751b3bf79cc003bd22e49d7d73","impliedFormat":1},{"version":"d40d20ac633703a7333770bfd60360126fc3302d5392d237bbb76e8c529a4f95","impliedFormat":1},{"version":"35a9867207c488061fb4f6fe4715802fbc164b4400018d2fa0149ad02db9a61c","impliedFormat":1},{"version":"91bf47a209ad0eae090023c3ebc1165a491cf9758799368ffcbee8dbe7448f33","impliedFormat":1},{"version":"0abe2cd72812bbfc509975860277c7cd6f6e0be95d765a9da77fee98264a7e32","impliedFormat":1},{"version":"13286c0c8524606b17a8d68650970bab896fb505f348f71601abf0f2296e8913","impliedFormat":1},{"version":"fc2a131847515b3dff2f0e835633d9a00a9d03ed59e690e27eec85b7b0522f92","impliedFormat":1},{"version":"90433c678bc26751eb7a5d54a2bb0a14be6f5717f69abb5f7a04afc75dce15a4","impliedFormat":1},{"version":"cd0565ace87a2d7802bf4c20ea23a997c54e598b9eb89f9c75e69478c1f7a0b4","impliedFormat":1},{"version":"738020d2c8fc9df92d5dee4b682d35a776eaedfe2166d12bc8f186e1ea57cc52","impliedFormat":1},{"version":"86dd7c5657a0b0bc6bee8002edcfd544458d3d3c60974555746eb9b2583dc35e","impliedFormat":1},{"version":"d97b96b6ecd4ee03f9f1170722c825ef778430a6a0d7aab03b8929012bf773cd","impliedFormat":1},{"version":"f61963dc02ef27c48fb0e0016a413b1e00bcb8b97a3f5d4473cedc7b44c8dc77","impliedFormat":1},{"version":"272dbfe04cfa965d6fff63fdaba415c1b5a515b1881ae265148f8a84ddeb318f","impliedFormat":1},{"version":"2035fb009b5fafa9a4f4e3b3fdb06d9225b89f2cbbf17a5b62413bf72cea721a","impliedFormat":1},{"version":"eefafec7c059f07b885b79b327d381c9a560e82b439793de597441a4e68d774a","impliedFormat":1},{"version":"72636f59b635c378dc9ea5246b9b3517b1214e340e468e54cb80126353053b2e","impliedFormat":1},{"version":"ebb79f267a3bf2de5f8edc1995c5d31777b539935fab8b7d863e8efb06c8e9ea","impliedFormat":1},{"version":"ada033e6a4c7f4e147e6d76bb881069dc66750619f8cc2472d65beeec1100145","impliedFormat":1},{"version":"0c04cc14a807a5dc0e3752d18a3b2655a135fefbf76ddcdabd0c5df037530d41","impliedFormat":1},{"version":"605d29d619180fbec287d1701e8b1f51f2d16747ec308d20aba3e9a0dac43a0f","impliedFormat":1},{"version":"67c19848b442d77c767414084fc571ce118b08301c4ddff904889d318f3a3363","impliedFormat":1},{"version":"c704ff0e0cb86d1b791767a88af21dadfee259180720a14c12baee668d0eb8fb","impliedFormat":1},{"version":"195c50e15d5b3ea034e01fbdca6f8ad4b35ad47463805bb0360bdffd6fce3009","impliedFormat":1},{"version":"da665f00b6877ae4adb39cd548257f487a76e3d99e006a702a4f38b4b39431cb","impliedFormat":1},{"version":"2b82adc9eead34b824a3f4dad315203fbfa56bee0061ccf9b485820606564f70","impliedFormat":1},{"version":"eb47aaa5e1b0a69388bb48422a991b9364a9c206a97983e0227289a9e1fca178","impliedFormat":1},{"version":"d7a4309673b06223537bc9544b1a5fe9425628e1c8ab5605f3c5ebc27ecb8074","impliedFormat":1},{"version":"db2108aea36e7faa83c38f6fe8225b9ad40835c0cba7fa38e969768299b83173","impliedFormat":1},{"version":"3eadfd083d40777b403f4f4eecfa40f93876f2a01779157cc114b2565a7afb51","impliedFormat":1},{"version":"cb6789ce3eba018d5a7996ccbf50e27541d850e9b4ee97fdcb3cbd8c5093691f","impliedFormat":1},{"version":"a3684ea9719122f9477902acd08cd363a6f3cff6d493df89d4dc12fa58204e27","impliedFormat":1},{"version":"2828dabf17a6507d39ebcc58fef847e111dcf2d51b8e4ff0d32732c72be032b3","impliedFormat":1},{"version":"c0c46113b4cd5ec9e7cf56e6dbfb3930ef6cbba914c0883eeced396988ae8320","impliedFormat":1},{"version":"118ea3f4e7b9c12e92551be0766706f57a411b4f18a1b4762cfde3cd6d4f0a96","impliedFormat":1},{"version":"2ad163aaddfa29231a021de6838f59378a210501634f125ed04cfa7d066ffc53","impliedFormat":1},{"version":"6305acbe492b9882ec940f8f0c8e5d1e1395258852f99328efcb1cf1683ca817","impliedFormat":1},{"version":"7619b1f6087a4e9336b2c42bd784b05aa4a2204a364b60171e5a628f817a381e","impliedFormat":1},{"version":"15be9120572c9fbcd3c267bd93b4140354514c9e70734e6fcca65ff4a246f83a","impliedFormat":1},{"version":"412482ab85893cec1d6f26231359474d1f59f6339e2743c08da1b05fc1d12767","impliedFormat":1},{"version":"858e2315e58af0d28fcd7f141a2505aba6a76fd10378ba0ad169b0336fee33fc","impliedFormat":1},{"version":"02da6c1b34f4ae2120d70cf5f9268bf1aedf62e55529d34f5974f5a93655ce38","impliedFormat":1},{"version":"3ecf179ef1cc28f7f9b46c8d2e496d50b542c176e94ed0147bab147b4a961cd6","impliedFormat":1},{"version":"b145da03ce7e174af5ced2cbbd16e96d3d5c2212f9a90d3657b63a5650a73b7f","impliedFormat":1},{"version":"c7aadab66a2bc90eeb0ab145ca4daebcbc038e24359263de3b40e7b1c7affba6","impliedFormat":1},{"version":"99518dc06286877a7b716e0f22c1a72d3c62be42701324b49f27bcc03573efff","impliedFormat":1},{"version":"f4575fd196a7e33c7be9773a71bcc5fbe7182a2152be909f6b8e8e7ba2438f06","impliedFormat":1},{"version":"05cba5acd77a4384389b9c62739104b5a1693efd66e6abac6c5ffc53280ae777","impliedFormat":1},{"version":"acacda82ebd929fe2fe9e31a37f193fc8498a7393a1c31dc5ceb656e2b45b708","impliedFormat":1},{"version":"1b13e7c5c58ab894fe65b099b6d19bb8afae6d04252db1bf55fe6ba95a0af954","impliedFormat":1},{"version":"4355d326c3129e5853b56267903f294ad03e34cc28b75f96b80734882dedac80","impliedFormat":1},{"version":"37139a8d45342c05b6a5aa1698a2e8e882d6dca5fb9a77aa91f05ac04e92e70b","impliedFormat":1},{"version":"e37191297f1234d3ae54edbf174489f9a3091a05fe959724db36f8e58d21fb17","impliedFormat":1},{"version":"3fca8fb3aab1bc7abb9b1420f517e9012fdddcbe18803bea2dd48fad6c45e92e","impliedFormat":1},{"version":"d0b0779e0cac4809a9a3c764ba3bd68314de758765a8e3b9291fe1671bfeb8a1","impliedFormat":1},{"version":"d2116b5f989aa68e585ae261b9d6d836be6ed1be0b55b47336d9f3db34674e86","impliedFormat":1},{"version":"d79a227dd654be16d8006eac8b67212679d1df494dfe6da22ea0bd34a13e010c","impliedFormat":1},{"version":"b9c89b4a2435c171e0a9a56668f510a376cb7991eaecef08b619e6d484841735","impliedFormat":1},{"version":"44a298a6c52a7dab8e970e95a6dabe20972a7c31c340842e0dc57f2c822826eb","impliedFormat":1},{"version":"6a79b61f57699de0a381c8a13f4c4bcd120556bfab0b4576994b6917cb62948b","impliedFormat":1},{"version":"c5133d7bdec65f465df12f0b507fbc0d96c78bfa5a012b0eb322cf1ff654e733","impliedFormat":1},{"version":"00b9ff040025f6b00e0f4ac8305fea1809975b325af31541bd9d69fa3b5e57b1","impliedFormat":1},{"version":"9f96b9fd0362a7bfe6a3aa70baa883c47ae167469c904782c99ccc942f62f0dc","impliedFormat":1},{"version":"54d91053dc6a2936bfd01a130cc3b524e11aa0349da082e8ac03a8bf44250338","impliedFormat":1},{"version":"89049878a456b5e0870bb50289ea8ece28a2abd0255301a261fa8ab6a3e9a07d","impliedFormat":1},{"version":"55ae9554811525f24818e19bdc8779fa99df434be7c03e5fc47fa441315f0226","impliedFormat":1},{"version":"24abac81e9c60089a126704e936192b2309413b40a53d9da68dadd1dd107684e","impliedFormat":1},{"version":"f13310c360ecffddb3858dcb33a7619665369d465f55e7386c31d45dfc3847bf","impliedFormat":1},{"version":"e7bde95a05a0564ee1450bc9a53797b0ac7944bf24d87d6f645baca3aa60df48","impliedFormat":1},{"version":"62e68ce120914431a7d34232d3eca643a7ddd67584387936a5202ae1c4dd9a1b","impliedFormat":1},{"version":"91d695bba902cc2eda7edc076cd17c5c9340f7bb254597deb6679e343effadbb","impliedFormat":1},{"version":"e1cb8168c7e0bd4857a66558fe7fe6c66d08432a0a943c51bacdac83773d5745","impliedFormat":1},{"version":"a464510505f31a356e9833963d89ce39f37a098715fc2863e533255af4410525","impliedFormat":1},{"version":"ebbe6765a836bfa7f03181bc433c8984ca29626270ca1e240c009851222cb8a7","impliedFormat":1},{"version":"ac10457b51ee4a3173b7165c87c795eadd094e024f1d9f0b6f0c131126e3d903","impliedFormat":1},{"version":"468df9d24a6e2bc6b4351417e3b5b4c2ca08264d6d5045fe18eb42e7996e58b4","impliedFormat":1},{"version":"954523d1f4856180cbf79b35bd754e14d3b2aea06c7efd71b254c745976086e9","impliedFormat":1},{"version":"a8af4739274959d70f7da4bfdd64f71cfc08d825c2d5d3561bc7baed760b33ef","impliedFormat":1},{"version":"090fda1107e7d4f8f30a2b341834ed949f01737b5ec6021bb6981f8907330bdb","impliedFormat":1},{"version":"cc32874a27100c32e3706d347eb4f435d6dd5c0d83e547c157352f977bbc6385","impliedFormat":1},{"version":"e45b069d58c9ac341d371b8bc3db4fa7351b9eee1731bffd651cfc1eb622f844","impliedFormat":1},{"version":"7f3c74caad25bfb6dfbf78c6fe194efcf8f79d1703d785fc05cd606fe0270525","impliedFormat":1},{"version":"54f3f7ff36384ca5c9e1627118b43df3014b7e0f62c9722619d19cdb7e43d608","impliedFormat":1},{"version":"2f346f1233bae487f1f9a11025fc73a1bf9093ee47980a9f4a75b84ea0bb7021","impliedFormat":1},{"version":"013444d0b8c1f7b5115462c31573a699fee7458381b0611062a0069d3ef810e8","impliedFormat":1},{"version":"0612b149cabbc136cb25de9daf062659f306b67793edc5e39755c51c724e2949","impliedFormat":1},{"version":"2579b150b86b5f644d86a6d58f17e3b801772c78866c34d41f86f3fc9eb523fe","impliedFormat":1},{"version":"0353e05b0d8475c10ddd88056e0483b191aa5cdea00a25e0505b96e023f1a2d9","impliedFormat":1},{"version":"8c4df93dafcf06adc42a63477cc38b352565a3ed0a19dd8ef7dfacc253749327","impliedFormat":1},{"version":"22a35275abc67f8aba44efc52b2f4b1abc2c94e183d36647fdab5a5e7c1bdf23","impliedFormat":1},{"version":"99193bafaa9ce112889698de25c4b8c80b1209bb7402189aea1c7ada708a8a54","impliedFormat":1},{"version":"70473538c6eb9494d53bf1539fe69df68d87c348743d8f7244dcb02ca3619484","impliedFormat":1},{"version":"c48932ab06a4e7531bdca7b0f739ace5fa273f9a1b9009bcd26902f8c0b851f0","impliedFormat":1},{"version":"df6c83e574308f6540c19e3409370482a7d8f448d56c65790b4ac0ab6f6fedd8","impliedFormat":1},{"version":"32f19b665839b1382b21afc41917cda47a56e744cd3df9986b13a72746d1c522","impliedFormat":1},{"version":"8db1ed144dd2304b9bd6e41211e22bad5f4ab1d8006e6ac127b29599f4b36083","impliedFormat":1},{"version":"843a5e3737f2abbbbd43bf2014b70f1c69a80530814a27ae1f8be213ae9ec222","impliedFormat":1},{"version":"6fc1be224ad6b3f3ec11535820def2d21636a47205c2c9de32238ba1ac8d82e6","impliedFormat":1},{"version":"5a44788293f9165116c9c183be66cefef0dc5d718782a04847de53bf664f3cc1","impliedFormat":1},{"version":"afd653ae63ce07075b018ba5ce8f4e977b6055c81cc65998410b904b94003c0a","impliedFormat":1},{"version":"9172155acfeb17b9d75f65b84f36cb3eb0ff3cd763db3f0d1ad5f6d10d55662f","impliedFormat":1},{"version":"71807b208e5f15feffb3ff530bec5b46b1217af0d8cc96dde00d549353bcb864","impliedFormat":1},{"version":"1a6eca5c2bc446481046c01a54553c3ffb856f81607a074f9f0256c59dd0ab13","impliedFormat":1},{"version":"6ecc423e71318bafbd230e6059e082c377170dfc7e02fccfa600586f8604d452","impliedFormat":1},{"version":"772f9bdd2bf50c9c01b0506001545e9b878faa7394ad6e7d90b49b179a024584","impliedFormat":1},{"version":"f204b03cb07517d71715ac8bc7552542bfab395adb53e31c07fbc67de6856de1","impliedFormat":1},{"version":"7467736a77548887faa90a7d0e074459810a5db4bbc6de302a2be6c05287ccae","impliedFormat":1},{"version":"39504a2c1278ee4d0dc1a34e27c80e58b4c53c08c87e3a7fc924f18c936bebb5","impliedFormat":1},{"version":"cd1ccdd9fd7980d43dfede5d42ee3d18064baed98b136089cf7c8221d562f058","impliedFormat":1},{"version":"d60f9a4fd1e734e7b79517f02622426ea1000deb7d6549dfdece043353691a4e","impliedFormat":1},{"version":"403d28b5e5f8fcff795ac038902033ec5890143e950af45bd91a3ed231e8b59c","impliedFormat":1},{"version":"c73b59f91088c00886d44ca296d53a75c263c3bda31e3b2f37ceb137382282be","impliedFormat":1},{"version":"e7aa2c584edb0970cb4bb01eb10344200286055f9a22bc3dadcc5a1f9199af3e","impliedFormat":1},{"version":"bfeb476eb0049185cb94c2bfcadb3ce1190554bbcf170d2bf7c68ed9bb00458e","impliedFormat":1},{"version":"ae23a65a2b664ffe979b0a2a98842e10bdf3af67a356f14bbc9d77eb3ab13585","impliedFormat":1},{"version":"eccf6ad2a8624329653896e8dbd03f30756cbd902a81b5d3942d6cf0e1a21575","impliedFormat":1},{"version":"1930c964051c04b4b5475702613cd5a27fcc2d33057aa946ff52bfca990dbc84","impliedFormat":1},{"version":"2793d525d79404df346e4ef58a82f9b6d28a7650beeb17378cd121c45ba03f02","impliedFormat":1},{"version":"62463aa3d299ae0cdc5473d2ac32213a05753c3adce87a8801c6d2b114a64116","impliedFormat":1},{"version":"c9c2eabaad71c534d7de16385977f95184fdf3ddd0339dadbd5d599488d94f90","impliedFormat":1},{"version":"d0642c453e6af4c0700182bec4afc5b2cc9498fe27c9b1bcf2e6f75dd1892699","impliedFormat":1},{"version":"8f4469dd750d15f72ba66876c8bc429d3c9ce49599a13f868a427d6681d45351","impliedFormat":1},{"version":"d1e888a33faeb1f0e3c558bbe0ea4a55056318e0b2f8eba72ffd6729c3bbff4e","impliedFormat":1},{"version":"f689c0633e8c95f550d36af943d775f3fae3dac81a28714b45c7af0bbb76a980","impliedFormat":1},{"version":"fef736cfb404b4db9aa942f377dbbac6edb76d18aabd3b647713fa75da8939e9","impliedFormat":1},{"version":"45659c92e49dfca4601acc7e57fbb03a71513c69768984baf86ead8d20387a01","impliedFormat":1},{"version":"0239d8f6a3f51b26cbdbb9362f4fde35651c6bd0ff3d9fc09ee4a2da6065cb4e","impliedFormat":1},{"version":"6e5ab399ec7bd61d4f86421cc6074fd904379c3923706c899d15146e4f9a08c8","impliedFormat":1},{"version":"c9ffec02582eed74f518ae3e32a5dcf4ac835532e548300c5c5f950cdfeead5f","impliedFormat":1},{"version":"df343f5de08f5b607a3c7954ff1b512b7fa983d561e136cce0b6dc6849602a15","impliedFormat":1},{"version":"8fc97ef271771dc6f81a9c846d007ac4f0cb5779e3f441c1de54dfda5046fe7b","impliedFormat":1},{"version":"b5a060e2a4c54695076f871ddc0c91a0ff8eea1262177c4ede5593acbf1ca3bb","impliedFormat":1},{"version":"08ee70765d3fa7c5bad4afbbe1c542771e17f84bfd5e3e872ae1fdc5160836c8","impliedFormat":1},{"version":"1c225a18846203fafc4334658715b0d3fd3ee842c4cfd42e628a535eda17730d","impliedFormat":1},{"version":"7ce93da38595d1caf57452d57e0733474564c2b290459d34f6e9dcf66e2d8beb","impliedFormat":1},{"version":"d7b672c1c583e9e34ff6df2549d6a55d7ca3adaf72e6a05081ea9ee625dac59f","impliedFormat":1},{"version":"f3a2902e84ebdef6525ed6bf116387a1256ea9ae8eeb36c22f070b7c9ea4cf09","impliedFormat":1},{"version":"33bb0d96cea9782d701332e6b7390f8efae3af92fd3e2aa2ac45e4a610e705d6","impliedFormat":1},{"version":"ae3e98448468e46474d817b5ebe74db11ab22c2feb60e292d96ce1a4ee963623","impliedFormat":1},{"version":"f0a2fdee9e801ac9320a8660dd6b8a930bf8c5b658d390ae0feafdba8b633688","impliedFormat":1},{"version":"7beb7f04f6186bdac5e622d44e4cac38d9f2b9fcad984b10d3762e369524dd77","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"63101dfa598f051c5d6f49d35d17856878a72aecab7cf30db97a7442d90f8b9b","signature":"653711fba8904aa27fd8911b63cf526e7b334e13a292da4cefdbbe179ac3f3f2"},{"version":"76e2438e845f869f51838ca514366cadec60af84eef465f462297abc0bd167dc","signature":"ef022c91ea9e75ab4082f2e881f9c4db7b346be2da706e876b253bebce5e6140"},{"version":"b8ad793dc17938bc462812e3522bbd3d62519d91d9b4a6422bed1383c2d3eb42","impliedFormat":1},{"version":"8b0b6a4c032a56d5651f7dd02ba3f05fbfe4131c4095093633cda3cae0991972","impliedFormat":1},{"version":"ff3c48a17bf10dfbb62448152042e4a48a56c9972059997ab9e7ed03b191809b","impliedFormat":1},{"version":"192a0c215bffe5e4ac7b9ff1e90e94bf4dfdad4f0f69a5ae07fccc36435ebb87","impliedFormat":1},{"version":"3ef8565e3d254583cced37534f161c31e3a8f341ff005c98b582c6d8c9274538","impliedFormat":1},{"version":"d7e42a3800e287d2a1af8479c7dd58c8663e80a01686cb89e0068be6c777d687","impliedFormat":1},{"version":"1098034333d3eb3c1d974435cacba9bd5a625711453412b3a514774fec7ca748","impliedFormat":1},{"version":"f2388b97b898a93d5a864e85627e3af8638695ebfa6d732ecd39d382824f0e63","impliedFormat":1},{"version":"6c6bd91368169cfa94b4f8cc64ebca2b050685ec76bc4082c44ce125b5530cca","impliedFormat":1},{"version":"f477375e6f0bf2a638a71d4e7a3da8885e3a03f3e5350688541d136b10b762a6","impliedFormat":1},{"version":"a44d6ea4dc70c3d789e9cef3cc42b79c78d17d3ce07f5fd278a7e1cbe824da56","impliedFormat":1},{"version":"272af80940fcc0c8325e4a04322c50d11f8b8842f96ac66cbd440835e958dd14","impliedFormat":1},{"version":"1803e48a3ec919ccafbcafeef5e410776ca0644ae8c6c87beca4c92d8a964434","impliedFormat":1},{"version":"875c43c5409e197e72ee517cb1f8fd358406b4adf058dbdc1e50c8db93d68f26","impliedFormat":1},{"version":"8854713984b9588eac1cab69c9e2a6e1a33760d9a2d182169059991914dd8577","impliedFormat":1},{"version":"e333d487ca89f26eafb95ea4b59bea8ba26b357e9f2fd3728be81d999f9e8cf6","impliedFormat":1},{"version":"2f554c6798b731fc39ff4e3d86aadc932fdeaa063e3cbab025623ff5653c0031","impliedFormat":1},{"version":"fe4613c6c0d23edc04cd8585bdd86bc7337dc6265fb52037d11ca19eeb5e5aaf","impliedFormat":1},{"version":"53b26fbee1a21a6403cf4625d0e501a966b9ccf735754b854366cee8984b711c","impliedFormat":1},{"version":"c503be3ddb3990ab27ca20c6559d29b547d9f9413e05d2987dd7c4bcf52f3736","impliedFormat":1},{"version":"598b15f0ae9a73082631d14cb8297a1285150ca325dbce98fc29c4f0b7079443","impliedFormat":1},{"version":"8c59d8256086ed17676139ee43c1155673e357ab956fb9d00711a7cac73e059d","impliedFormat":1},{"version":"cfe88132f67aa055a3f49d59b01585fa8d890f5a66a0a13bb71973d57573eee7","impliedFormat":1},{"version":"53ce488a97f0b50686ade64252f60a1e491591dd7324f017b86d78239bd232ca","impliedFormat":1},{"version":"50fd11b764194f06977c162c37e5a70bcf0d3579bf82dd4de4eee3ac68d0f82f","impliedFormat":1},{"version":"e0ceb647dcdf6b27fd37e8b0406c7eafb8adfc99414837f3c9bfd28ffed6150a","impliedFormat":1},{"version":"99579aa074ed298e7a3d6a47e68f0cd099e92411212d5081ce88344a5b1b528d","impliedFormat":1},{"version":"c94c1aa80687a277396307b80774ca540d0559c2f7ba340168c2637c82b1f766","impliedFormat":1},{"version":"ce7dbf31739cc7bca35ca50e4f0cbd75cd31fd6c05c66841f8748e225dc73aaf","impliedFormat":1},{"version":"942ab34f62ac3f3d20014615b6442b6dc51815e30a878ebc390dd70e0dec63bf","impliedFormat":1},{"version":"7a671bf8b4ad81b8b8aea76213ca31b8a5de4ba39490fbdee249fc5ba974a622","impliedFormat":1},{"version":"8e07f13fb0f67e12863b096734f004e14c5ebfd34a524ed4c863c80354c25a44","impliedFormat":1},{"version":"6f6bdb523e5162216efc36ebba4f1ef8e845f1a9e55f15387df8e85206448aee","impliedFormat":1},{"version":"aa2d6531a04d6379318d29891de396f61ccc171bfd2f8448cc1649c184becdf2","impliedFormat":1},{"version":"d422f0c340060a53cb56d0db24dd170e31e236a808130ab106f7ab2c846f1cdb","impliedFormat":1},{"version":"424403ef35c4c97a7f00ea85f4a5e2f088659c731e75dbe0c546137cb64ef8d8","impliedFormat":1},{"version":"16900e9a60518461d7889be8efeca3fe2cbcd3f6ce6dee70fea81dfbf8990a76","impliedFormat":1},{"version":"6daf17b3bd9499bd0cc1733ab227267d48cd0145ed9967c983ccb8f52eb72d6e","impliedFormat":1},{"version":"e4177e6220d0fef2500432c723dbd2eb9a27dcb491344e6b342be58cc1379ec0","impliedFormat":1},{"version":"ab710f1ee2866e473454a348cffd8d5486e3c07c255f214e19e59a4f17eece4d","impliedFormat":1},{"version":"db7ff3459e80382c61441ea9171f183252b6acc82957ecb6285fff4dca55c585","impliedFormat":1},{"version":"4a168e11fe0f46918721d2f6fcdb676333395736371db1c113ae30b6fde9ccd2","impliedFormat":1},{"version":"2a899aef0c6c94cc3537fe93ec8047647e77a3f52ee7cacda95a8c956d3623fb","impliedFormat":1},{"version":"ef2c1585cad462bdf65f2640e7bcd75cd0dbc45bae297e75072e11fe3db017fa","impliedFormat":1},{"version":"6a52170a5e4600bbb47a94a1dd9522dca7348ce591d8cdbb7d4fe3e23bbea461","impliedFormat":1},{"version":"6f6eadb32844b0ec7b322293b011316486894f110443197c4c9fbcba01b3b2fa","impliedFormat":1},{"version":"a51e08f41e3e948c287268a275bfe652856a10f68ddd2bf3e3aaf5b8cdb9ef85","impliedFormat":1},{"version":"16c144a21cd99926eeba1605aec9984439e91aa864d1c210e176ca668f5f586a","impliedFormat":1},{"version":"af48a76b75041e2b3e7bd8eed786c07f39ea896bb2ff165e27e18208d09b8bee","impliedFormat":1},{"version":"fd4107bd5c899165a21ab93768904d5cfb3e98b952f91fbf5a12789a4c0744e6","impliedFormat":1},{"version":"deb092bc337b2cb0a1b14f3d43f56bc663e1447694e6d479d6df8296bdd452d6","impliedFormat":1},{"version":"041bc1c3620322cb6152183857601707ef6626e9d99f736e8780533689fb1bf9","impliedFormat":1},{"version":"77165b117f552be305d3bc2ef83424ff1e67afb22bfabd14ebebb3468c21fcaa","impliedFormat":1},{"version":"128e7c2ffd37aa29e05367400d718b0e4770cefb1e658d8783ec80a16bc0643a","impliedFormat":1},{"version":"076ac4f2d642c473fa7f01c8c1b7b4ef58f921130174d9cf78430651f44c43ec","impliedFormat":1},{"version":"396c1e5a39706999ec8cc582916e05fcb4f901631d2c192c1292e95089a494d9","impliedFormat":1},{"version":"89df75d28f34fc698fe261f9489125b4e5828fbd62d863bbe93373d3ed995056","impliedFormat":1},{"version":"8ccf5843249a042f4553a308816fe8a03aa423e55544637757d0cfa338bb5186","impliedFormat":1},{"version":"93b44aa4a7b27ba57d9e2bad6fb7943956de85c5cc330d2c3e30cd25b4583d44","impliedFormat":1},{"version":"a0c6216075f54cafdfa90412596b165ff85e2cadd319c49557cc8410f487b77c","impliedFormat":1},{"version":"3c359d811ec0097cba00fb2afd844b125a2ddf4cad88afaf864e88c8d3d358bd","impliedFormat":1},{"version":"d8ec19be7d6d3950992c3418f3a4aa2bcad144252bd7c0891462b5879f436e4e","impliedFormat":1},{"version":"db37aa3208b48bdcbc27c0c1ae3d1b86c0d5159e65543e8ab79cbfb37b1f2f34","impliedFormat":1},{"version":"d62f09256941e92a95b78ae2267e4cf5ff2ca8915d62b9561b1bc85af1baf428","impliedFormat":1},{"version":"e6223b7263dd7a49f4691bf8df2b1e69f764fb46972937e6f9b28538d050b1ba","impliedFormat":1},{"version":"2daf06d8e15cbca27baa6c106253b92dad96afd87af9996cf49a47103b97dc95","impliedFormat":1},{"version":"1db014db736a09668e0c0576585174dbcfd6471bb5e2d79f151a241e0d18d66b","impliedFormat":1},{"version":"8a153d30edde9cefd102e5523b5a9673c298fc7cf7af5173ae946cbb8dd48f11","impliedFormat":1},{"version":"abaaf8d606990f505ee5f76d0b45a44df60886a7d470820fcfb2c06eafa99659","impliedFormat":1},{"version":"8109e0580fc71dbefd6091b8825acf83209b6c07d3f54c33afeafab5e1f88844","impliedFormat":1},{"version":"d92a80c2c05cf974704088f9da904fe5eadc0b3ad49ddd1ef70ca8028b5adda1","impliedFormat":1},{"version":"fbd7450f20b4486c54f8a90486c395b14f76da66ba30a7d83590e199848f0660","impliedFormat":1},{"version":"ece5b0e45c865645ab65880854899a5422a0b76ada7baa49300c76d38a530ee1","impliedFormat":1},{"version":"62d89ac385aeab821e2d55b4f9a23a277d44f33c67fefe4859c17b80fdb397ea","impliedFormat":1},{"version":"f4dee11887c5564886026263c6ee65c0babc971b2b8848d85c35927af25da827","impliedFormat":1},{"version":"fb8dd49a4cd6d802be4554fbab193bb06e2035905779777f32326cb57cf6a2c2","impliedFormat":1},{"version":"df29ade4994de2d9327a5f44a706bbe6103022a8f40316839afa38d3e078ee06","impliedFormat":1},{"version":"82d3e00d56a71fc169f3cf9ec5f5ffcc92f6c0e67d4dfc130dafe9f1886d5515","impliedFormat":1},{"version":"d38f45cb868a830d130ac8b87d3f7e8caff4961a3a1feae055de5e538e20879a","impliedFormat":1},{"version":"4c30a5cb3097befb9704d16aa4670e64e39ea69c5964a1433b9ffd32e1a5a3a1","impliedFormat":1},{"version":"1b33478647aa1b771314745807397002a410c746480e9447db959110999873ce","impliedFormat":1},{"version":"7b3a5e25bf3c51af55cb2986b89949317aa0f6cbfb5317edd7d4037fa52219a9","impliedFormat":1},{"version":"3cd50f6a83629c0ec330fc482e587bfa96532d4c9ce85e6c3ddf9f52f63eee11","impliedFormat":1},{"version":"9fac6ebf3c60ced53dd21def30a679ec225fc3ff4b8d66b86326c285a4eebb5a","impliedFormat":1},{"version":"8cb83cb98c460cd716d2a98b64eb1a07a3a65c7362436550e02f5c2d212871d1","impliedFormat":1},{"version":"07bc8a3551e39e70c38e7293b1a09916867d728043e352b119f951742cb91624","impliedFormat":1},{"version":"e47adc2176f43c617c0ab47f2d9b2bb1706d9e0669bf349a30c3fe09ddd63261","impliedFormat":1},{"version":"7fec79dfd7319fec7456b1b53134edb54c411ba493a0aef350eee75a4f223eeb","impliedFormat":1},{"version":"189c489705bb96a308dcde9b3336011d08bfbca568bcaf5d5d55c05468e9de7a","impliedFormat":1},{"version":"98f4b1074567341764b580bf14c5aabe82a4390d11553780814f7e932970a6f7","impliedFormat":1},{"version":"dadfa5fd3d5c511ca6bfe240243b5cf2e0f87e44ea63e23c4b2fce253c0d4601","impliedFormat":1},{"version":"2e252235037a2cd8feebfbf74aa460f783e5d423895d13f29a934d7655a1f8be","impliedFormat":1},{"version":"763f4ac187891a6d71ae8821f45eef7ff915b5d687233349e2c8a76c22b3bf2a","impliedFormat":1},{"version":"dff93e0997c4e64ff29e9f70cad172c0b438c4f58c119f17a51c94d48164475a","impliedFormat":1},{"version":"fd1ddf926b323dfa439be49c1d41bbe233fe5656975a11183aeb3bf2addfa3bb","impliedFormat":1},{"version":"6dda11db28da6bcc7ff09242cd1866bdddd0ae91e2db3bea03ba66112399641a","impliedFormat":1},{"version":"ea4cd1e72af1aa49cf208b9cb4caf542437beb7a7a5b522f50a5f1b7480362ed","impliedFormat":1},{"version":"903a7d68a222d94da11a5a89449fdd5dd75d83cd95af34c0242e10b85ec33a93","impliedFormat":1},{"version":"e7fe2e7ed5c3a7beff60361632be19a8943e53466b7dd69c34f89faf473206d7","impliedFormat":1},{"version":"b4896cee83379e159f83021e262223354db79e439092e485611163e2082224ff","impliedFormat":1},{"version":"5243e79a643e41d9653011d6c66e95048fc0478eb8593dc079b70877a2e3990e","impliedFormat":1},{"version":"70521b6ab0dcba37539e5303104f29b721bfb2940b2776da4cc818c07e1fefc1","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab41ef1f2cdafb8df48be20cd969d875602483859dc194e9c97c8a576892c052","affectsGlobalScope":true,"impliedFormat":1},{"version":"d153a11543fd884b596587ccd97aebbeed950b26933ee000f94009f1ab142848","affectsGlobalScope":true,"impliedFormat":1},{"version":"21d819c173c0cf7cc3ce57c3276e77fd9a8a01d35a06ad87158781515c9a438a","impliedFormat":1},{"version":"98cffbf06d6bab333473c70a893770dbe990783904002c4f1a960447b4b53dca","affectsGlobalScope":true,"impliedFormat":1},{"version":"ba481bca06f37d3f2c137ce343c7d5937029b2468f8e26111f3c9d9963d6568d","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d9ef24f9a22a88e3e9b3b3d8c40ab1ddb0853f1bfbd5c843c37800138437b61","affectsGlobalScope":true,"impliedFormat":1},{"version":"1db0b7dca579049ca4193d034d835f6bfe73096c73663e5ef9a0b5779939f3d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"9798340ffb0d067d69b1ae5b32faa17ab31b82466a3fc00d8f2f2df0c8554aaa","affectsGlobalScope":true,"impliedFormat":1},{"version":"f26b11d8d8e4b8028f1c7d618b22274c892e4b0ef5b3678a8ccbad85419aef43","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e9c23ba78aabc2e0a27033f18737a6df754067731e69dc5f52823957d60a4b6","impliedFormat":1},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"1d6e127068ea8e104a912e42fc0a110e2aa5a66a356a917a163e8cf9a65e4a75","impliedFormat":1},{"version":"5ded6427296cdf3b9542de4471d2aa8d3983671d4cac0f4bf9c637208d1ced43","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"ea0148f897b45a76544ae179784c95af1bd6721b8610af9ffa467a518a086a43","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"dd3273ead9fbde62a72949c97dbec2247ea08e0c6952e701a483d74ef92d6a17","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"e61be3f894b41b7baa1fbd6a66893f2579bfad01d208b4ff61daef21493ef0a8","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"b52476feb4a0cbcb25e5931b930fc73cb6643fb1a5060bf8a3dda0eeae5b4b68","affectsGlobalScope":true,"impliedFormat":1},{"version":"e2677634fe27e87348825bb041651e22d50a613e2fdf6a4a3ade971d71bac37e","impliedFormat":1},{"version":"7394959e5a741b185456e1ef5d64599c36c60a323207450991e7a42e08911419","impliedFormat":1},{"version":"8c0bcd6c6b67b4b503c11e91a1fb91522ed585900eab2ab1f61bba7d7caa9d6f","impliedFormat":1},{"version":"8cd19276b6590b3ebbeeb030ac271871b9ed0afc3074ac88a94ed2449174b776","affectsGlobalScope":true,"impliedFormat":1},{"version":"696eb8d28f5949b87d894b26dc97318ef944c794a9a4e4f62360cd1d1958014b","impliedFormat":1},{"version":"3f8fa3061bd7402970b399300880d55257953ee6d3cd408722cb9ac20126460c","impliedFormat":1},{"version":"35ec8b6760fd7138bbf5809b84551e31028fb2ba7b6dc91d95d098bf212ca8b4","affectsGlobalScope":true,"impliedFormat":1},{"version":"5524481e56c48ff486f42926778c0a3cce1cc85dc46683b92b1271865bcf015a","impliedFormat":1},{"version":"68bd56c92c2bd7d2339457eb84d63e7de3bd56a69b25f3576e1568d21a162398","affectsGlobalScope":true,"impliedFormat":1},{"version":"3e93b123f7c2944969d291b35fed2af79a6e9e27fdd5faa99748a51c07c02d28","impliedFormat":1},{"version":"9d19808c8c291a9010a6c788e8532a2da70f811adb431c97520803e0ec649991","impliedFormat":1},{"version":"87aad3dd9752067dc875cfaa466fc44246451c0c560b820796bdd528e29bef40","impliedFormat":1},{"version":"4aacb0dd020eeaef65426153686cc639a78ec2885dc72ad220be1d25f1a439df","impliedFormat":1},{"version":"f0bd7e6d931657b59605c44112eaf8b980ba7f957a5051ed21cb93d978cf2f45","impliedFormat":1},{"version":"8db0ae9cb14d9955b14c214f34dae1b9ef2baee2fe4ce794a4cd3ac2531e3255","affectsGlobalScope":true,"impliedFormat":1},{"version":"15fc6f7512c86810273af28f224251a5a879e4261b4d4c7e532abfbfc3983134","impliedFormat":1},{"version":"58adba1a8ab2d10b54dc1dced4e41f4e7c9772cbbac40939c0dc8ce2cdb1d442","impliedFormat":1},{"version":"641942a78f9063caa5d6b777c99304b7d1dc7328076038c6d94d8a0b81fc95c1","impliedFormat":1},{"version":"714435130b9015fae551788df2a88038471a5a11eb471f27c4ede86552842bc9","impliedFormat":1},{"version":"855cd5f7eb396f5f1ab1bc0f8580339bff77b68a770f84c6b254e319bbfd1ac7","impliedFormat":1},{"version":"5650cf3dace09e7c25d384e3e6b818b938f68f4e8de96f52d9c5a1b3db068e86","impliedFormat":1},{"version":"1354ca5c38bd3fd3836a68e0f7c9f91f172582ba30ab15bb8c075891b91502b7","affectsGlobalScope":true,"impliedFormat":1},{"version":"7e20d899c28ca26a2a7afc98beaa69e63ff7fba0a8bc47b4e3bf3ede5e09e424","impliedFormat":1},{"version":"2d2fcaab481b31a5882065c7951255703ddbe1c0e507af56ea42d79ac3911201","impliedFormat":1},{"version":"a192fe8ec33f75edbc8d8f3ed79f768dfae11ff5735e7fe52bfa69956e46d78d","impliedFormat":1},{"version":"ca867399f7db82df981d6915bcbb2d81131d7d1ef683bc782b59f71dda59bc85","affectsGlobalScope":true,"impliedFormat":1},{"version":"372413016d17d804e1d139418aca0c68e47a83fb6669490857f4b318de8cccb3","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e043a1bc8fbf2a255bccf9bf27e0f1caf916c3b0518ea34aa72357c0afd42ec","impliedFormat":1},{"version":"b4f70ec656a11d570e1a9edce07d118cd58d9760239e2ece99306ee9dfe61d02","impliedFormat":1},{"version":"3bc2f1e2c95c04048212c569ed38e338873f6a8593930cf5a7ef24ffb38fc3b6","impliedFormat":1},{"version":"6e70e9570e98aae2b825b533aa6292b6abd542e8d9f6e9475e88e1d7ba17c866","impliedFormat":1},{"version":"f9d9d753d430ed050dc1bf2667a1bab711ccbb1c1507183d794cc195a5b085cc","impliedFormat":1},{"version":"9eece5e586312581ccd106d4853e861aaaa1a39f8e3ea672b8c3847eedd12f6e","impliedFormat":1},{"version":"085f552d005479e2e6a7311cdbbe5d8c55c497b4d19274285df161ee9684cd9c","impliedFormat":1},{"version":"37ba7b45141a45ce6e80e66f2a96c8a5ab1bcef0fc2d0f56bb58df96ec67e972","impliedFormat":1},{"version":"45650f47bfb376c8a8ed39d4bcda5902ab899a3150029684ee4c10676d9fbaee","impliedFormat":1},{"version":"007faacc9268357caa21d24169f3f3f2497af3e9241308df2d89f6e6d9bb3f2e","affectsGlobalScope":true,"impliedFormat":1},{"version":"74cf591a0f63db318651e0e04cb55f8791385f86e987a67fd4d2eaab8191f730","impliedFormat":1},{"version":"5eab9b3dc9b34f185417342436ec3f106898da5f4801992d8ff38ab3aff346b5","impliedFormat":1},{"version":"12ed4559eba17cd977aa0db658d25c4047067444b51acfdcbf38470630642b23","affectsGlobalScope":true,"impliedFormat":1},{"version":"f3ffabc95802521e1e4bcba4c88d8615176dc6e09111d920c7a213bdda6e1d65","impliedFormat":1},{"version":"809821b8a065e3234a55b3a9d7846231ed18d66dd749f2494c66288d890daf7f","impliedFormat":1},{"version":"ae56f65caf3be91108707bd8dfbccc2a57a91feb5daabf7165a06a945545ed26","impliedFormat":1},{"version":"a136d5de521da20f31631a0a96bf712370779d1c05b7015d7019a9b2a0446ca9","impliedFormat":1},{"version":"c3b41e74b9a84b88b1dca61ec39eee25c0dbc8e7d519ba11bb070918cfacf656","affectsGlobalScope":true,"impliedFormat":1},{"version":"4737a9dc24d0e68b734e6cfbcea0c15a2cfafeb493485e27905f7856988c6b29","affectsGlobalScope":true,"impliedFormat":1},{"version":"36d8d3e7506b631c9582c251a2c0b8a28855af3f76719b12b534c6edf952748d","impliedFormat":1},{"version":"1ca69210cc42729e7ca97d3a9ad48f2e9cb0042bada4075b588ae5387debd318","impliedFormat":1},{"version":"f5ebe66baaf7c552cfa59d75f2bfba679f329204847db3cec385acda245e574e","impliedFormat":1},{"version":"ed59add13139f84da271cafd32e2171876b0a0af2f798d0c663e8eeb867732cf","affectsGlobalScope":true,"impliedFormat":1},{"version":"b7c5e2ea4a9749097c347454805e933844ed207b6eefec6b7cfd418b5f5f7b28","impliedFormat":1},{"version":"b1810689b76fd473bd12cc9ee219f8e62f54a7d08019a235d07424afbf074d25","impliedFormat":1},{"version":"76e7352249c42b9d54fe1f9e1ebcef777da1cb2eb33038366af49469d433597b","impliedFormat":1},{"version":"88cb622dd0ec1ef860e5c27fa884e60d2eba5ae22c7907dff82c56a69bdd2c8a","impliedFormat":1},{"version":"eb234b3e285e8bc071bdddc1ec0460095e13ead6222d44b02c4e0869522f9ba3","impliedFormat":1},{"version":"c85114872760189e50fef131944427b0fb367f0cc0b6dce164bb427a6fd89381","impliedFormat":1},{"version":"5ad69b0d7e7bdbcd3adfdb6a3e306e935c9c2711b1c60493646504a2f991346e","impliedFormat":1},{"version":"a12a667efdeb03b529bd4ebb4032998ddd32743799f59f9f18b186f8e63a2cf1","impliedFormat":1},{"version":"cee7efa0ae4c58deab218d1df0d1bf84abfd5c356cff28bca1421489cba13a19","impliedFormat":1},{"version":"f9e034b1ae29825c00532e08ea852b0c72885c343ee48d2975db0a6481218ab3","impliedFormat":1},{"version":"1193f49cbb883f40326461fe379e58ffa4c18d15bf6d6a1974ad2894e4fb20f3","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"2e2bc02af7b535d267be8cecbc5831466dd71c5af294401821791b26cb363c47","impliedFormat":1},{"version":"986affe0f60331f20df7d708ee097056b0973d85422ec2ce754af19c1fa4e4b1","impliedFormat":1},{"version":"8f06c2807459f1958b297f4ad09c6612d7dbd7997c9ccfc6ea384f7538e0cea8","impliedFormat":1},{"version":"a7de30cd043d7299bfe9daaca3732b086e734341587c3e923b01f3fd74d31126","impliedFormat":1},{"version":"78f7fad319e4ac305ffe8e03027423279b53a8af4db305096aa75d446b1ec7af","impliedFormat":1},{"version":"3bf58923a1d27819745bdad52bca1bdced9fef12cc0c7f8a3fd5f4e0206b684a","impliedFormat":1},{"version":"8fc11f102df58f03d36fcbf0da3efa37c177f5f18f534c76179ceef0c3a672cd","impliedFormat":1},{"version":"e6935ab0f64a886e778c12a54ed6e9075ce7e7f44723ff0d52020a654b025a09","impliedFormat":1},{"version":"9829af7653a29f1b85d3dd688a6c6256087c0b737b85d84b630e7f93fd420faf","impliedFormat":1},{"version":"3d9d985d41e536fcf79fc95082925c2f1ae5ade75814ad2bd70c0944747f7ac4","impliedFormat":1},{"version":"03b419ce598d77fe4d1705c8281a797a908f57ce24a15d6174d7e7276d355a65","impliedFormat":1},{"version":"b0e6f1b1569779cf567317c2265d67460d1d3b4de4e79126533109d87dc16d50","impliedFormat":1},{"version":"18cb8be1326ffa4158abd8d84c9b0a189c0f52201f12f7af2d2af830c077f2bf","impliedFormat":1},{"version":"9c15e2b87cd3d8b18881bcc7d72b2d1dc6d5fe078b674ae12c12c19ec09a6a1a","impliedFormat":1},{"version":"0de68916e23c1e3df800f9f61cdd7c506ceb0656fcbc245ee9974aad26786781","impliedFormat":1},{"version":"80c538ee6a62249e77ba3de07efb23d4a7ca8946499c065261bf5079f1cd3cf0","impliedFormat":1},{"version":"ad4277862bdcbe1cf5c1e0d43b39770e1ccc033da92f5b9ff75ca8c3a03a569b","impliedFormat":1},{"version":"46a86c47400a564df04a1604fcac41cb599ebbada392527a1462c9dfe4713d78","impliedFormat":1},{"version":"f342dcb96ad26855757929a9f6632704b7013f65786573d4fdcd4da09f475923","impliedFormat":1},{"version":"dcd467dc444953a537502d9e140d4f2dc13010664d4216cc8e6977b3c5c3efa3","impliedFormat":1},{"version":"ca476924dfa6120b807a14e0a8aea7b061b8bdaa7eecdb303d7957c769102e96","impliedFormat":1},{"version":"848fe622fac070f8af9255e5d63fe829e3da079cae30be48fb6deb5dbf2c27c6","impliedFormat":1},{"version":"f3bb275073b5db8931c042d347fdce888775436a4774836221af57fdccec32ff","impliedFormat":1},{"version":"03cb8cb2f8ef002a5cac9b8c9a0c02e5fd09de128b9769c5b920a6cbfc080087","impliedFormat":1},{"version":"3e5ebc3a6a938a03a361f4cdb9a26c9f5a1bac82b46273e11d5d37cd8eccc918","impliedFormat":1},{"version":"a0a7800e71c504c21f3051a29f0f6f948f0b8296c9ebffeb67033822aabf92e0","impliedFormat":1},{"version":"6a219f12b3e853398d51192736707e320699a355052687bad4729784649ff519","impliedFormat":1},{"version":"4294a84634c56529e67301a3258448019e41c101de6b9646ea41c0ecdc70df92","impliedFormat":1},{"version":"80fc027e10234b809a9a40086114a8154657dcb8478d58c85ef850592d352870","impliedFormat":1},{"version":"27f24ba43083d406b372e9eff72dbc378afa0503dac1c1dd32499cc92fc9cb22","impliedFormat":1},{"version":"12594611a054ca7fe69962f690a4e79922d563b4b434716eb855d63a9d11a78f","impliedFormat":1},{"version":"1440eca2d8bc47ebdbc5a901b369de1b7b39c3297e5b4ac9631899f49ea9740b","impliedFormat":1},{"version":"fc9897fbada879bda954603ea204c6e5df913262a90ad848b5efaab182b58033","impliedFormat":1},{"version":"93443b2da120bea58eb48bd7da86559d4cf868dc2d581eebf9b48b51ba1e8894","impliedFormat":1},{"version":"94be5c5f8cf26bbf53554cba4b112e30134349b14f3c0fd0ede3b51ec25a7174","impliedFormat":1},{"version":"c2956026078814be6dc01515213aeb1eb816e81715085952bbc97b7c81fe3f6d","impliedFormat":1},{"version":"ac3a69c529ab256532825b08902aec65d0d88c66963e39ae19a3d214953aedc5","impliedFormat":1},{"version":"fe29108f3ddf7030c3d573c5226ebe03213170b3beca5200ca7cb33755184017","impliedFormat":1},{"version":"04d5bfb0a0eecd66c0b3f522477bf69065a9703be8300fbea5566a0fc4a97b9d","impliedFormat":1},{"version":"d5e3e13faca961679bed01d80bc38b3336e7de598ebf9b03ec7d31081af735ad","impliedFormat":1},{"version":"de05a488fb501de32c1ec0af2a6ddfe0fdef46935b9f4ffb3922d355b15da674","impliedFormat":1},{"version":"9f00f2bc49f0c10275a52cb4f9e2991860d8b7b0922bfab6eafe14178377aa72","impliedFormat":1},{"version":"7bd94408358caf1794ad24546ca0aa56f9be6be2d3245d0972fcb924b84a81fd","impliedFormat":1},{"version":"0e7c3660d1df392b6f6ae7fa697f0629ae4404e5b7bac05dd81136247aff32d5","impliedFormat":1},{"version":"b0b3636502dc0c50295f67747968f202f7b775eac5016329606d1bc2888d5dd9","impliedFormat":1},{"version":"f9ede7ea553dc197fd5d2604f62cda1be1aea50024ed73237d9e3144f0c93608","impliedFormat":1},{"version":"b1005ae67226fd9b7b65333d9a351917f517d421a0c63b7cde59bec3b8e3562f","impliedFormat":1},{"version":"c6688fd4c2a8a24c9b80da3660a7a06b93ed37d12d84f3ba4aa071ffc125e75f","impliedFormat":1},{"version":"20efc25890a0b2f09e4d224afaaf84917baa77b1aee60d9dfd11ff8078d73f93","impliedFormat":1},{"version":"d00b48096854d711cee688e7ff1ca796c1bf0d27ca509633c2a98b85cc23d47d","impliedFormat":1},{"version":"30f116226d0e53c6cbbdbc967479d5c8036935f771b2af51987c2e8d4cc7fc6a","impliedFormat":1},{"version":"8be98ffc3c54fb40b220796b796388f8ade50c8ba813a811bffccf98006566d5","impliedFormat":1},{"version":"4e82eed3c1b5084132708ce030f8ec90b69e4b7bb844dcaacd808045ae24c0e2","impliedFormat":1},{"version":"eae8c7cbcb175b997ce8e76cd6e770eca5dba07228f6cb4a44e1b0a11eb87685","impliedFormat":1},{"version":"b3ded8e50b3cdf548d7c8d3b3b5b2105932b04a2f08b392564f4bc499407e4e5","impliedFormat":1},{"version":"4ed2d8fb4c598719985b8fbef65f7de9c3f5ae6a233fc0fe20bd00193c490908","impliedFormat":1},{"version":"6da51da9b74383988b89e17298ceca510357f63830f78b40f72afe4d5a9cee3e","impliedFormat":1},{"version":"512a079a1a3de2492c80aa599e173b2ea8cc6afb2800e3e99f14330b34155fe1","impliedFormat":1},{"version":"f281f20b801830f2f94b2bc0b18aba01d4fb50c2f4a847ffcadff39de31c8b80","impliedFormat":1},{"version":"738ddac5ab5b61d70d3466f3906d6b3c83c8786e922c6e726a6597296181ae87","impliedFormat":1},{"version":"90d202ace592f7b51b131a5890ec93e4df774c8677a485391c280cef0ea53f48","impliedFormat":1},{"version":"b34e1861949a545916696ef40f4a7fe71793661e72dd4db5e04cacc60ef23f7a","impliedFormat":1},{"version":"dd3f42651cfa863ded8fa0b0608fb892b826e254a0a6cbc447388cb5e11bffd5","impliedFormat":1},{"version":"8e3842ba15690ab4b340893a4552a8c3670b8f347fbb835afe14be98891eef10","impliedFormat":1},{"version":"e7b9673dcd3d1825dbd70ad1d1f848d68189afc302ecdafc6eb30cbe7bd420b5","impliedFormat":1},{"version":"15911b87a2ad4b65b30c445802d55fa6186c66068603113042e8c3dfa4a35e2a","impliedFormat":1},{"version":"a9dc7b8d06b1f69d219f61fa3f7ac621e6e3a8d5a430e800cd7d1a755cc058c3","impliedFormat":1},{"version":"f8c496656cb5fd737931b4d6c60bd72a97c48f37c07dcb74a593dd24ac3f684a","impliedFormat":1},{"version":"f2cf1d33c458ac091983e5dac1613f264d48a69b281e43c5b055321320082358","impliedFormat":1},{"version":"0fa43815d4b05eafe97c056dae73c313f23a9f00b559f1e942d042c7a04db93c","impliedFormat":1},{"version":"e769097e5ea39d2ed548eeb9c093e90f26dde167f95eb80fbdd4efb041778387","impliedFormat":1},{"version":"a02db6aabaa291a85cf52b0c3f02a75301b80be856db63d44af4feea2179f37b","impliedFormat":1},{"version":"e1e94e41f47a4496566a9f40e815687a2eca1e7b7910b67704813cf61248b869","impliedFormat":1},{"version":"557ba6713b2a6fefd943399d5fb6c64e315dc461e9e05eaa6300fdbeeda5d0a1","impliedFormat":1},{"version":"1f7eeb69504ad94d16f4731f707d2af879adc7487dc35b146e2d86825bb779b4","impliedFormat":1},{"version":"c1b5c480e4d38377c82f9f517c12014d3d4475c0e607c4845e0836e0e89bbf7d","impliedFormat":1},{"version":"1a014a8365354f37ea245349a4361d3b46589be7921fe7f1dbf408cc0f084bab","impliedFormat":1},{"version":"87fc4a324b9fa5c9b93a13b5ae1b55ea390929ec1b0450afebff9620921a9cc1","impliedFormat":1},{"version":"73c0b8df0e282e26a53820f53502847a043bd77a9cda78782207d5349842fba2","impliedFormat":1},{"version":"5c7391307b9a7c540d678f015d687c277269aa9171f441467e20bab15694db40","impliedFormat":1},{"version":"082aa8710bbf3d16b877e798341c69599fdd487b4dc34d374ab3e3ec6d46f690","impliedFormat":1},{"version":"acb9367f45f12526ea808d6da48ab77eee1ceb2b6fe47ab02bbcc7cce4c972b0","impliedFormat":1},{"version":"d6db974317fd9ff66a923555464850dcf87976054a7adacf09d53323f64686d1","impliedFormat":1},{"version":"79f4812dffe8f933c12c341d68eee731cb6dd7f2a4bb20097c411560c97a6263","impliedFormat":1},{"version":"c446e8f3bd5b16e121252e05ba7696524ca95ec3f819c12fb8c37e7836744769","impliedFormat":1},{"version":"23386bb0bcb20fcb367149f22f5c6468b53f1987e86fd25de875ffb769e4d241","impliedFormat":1},{"version":"3913806467307a4bd874b105ac3e79ac261ab986fbdce7f0feea26cbcee95765","impliedFormat":1},{"version":"a9417a980a4300048d179d0295e5b7dd76e4db7b566344779ee576cbd084b3c4","impliedFormat":1},{"version":"b96760c030c41fa078b35ea05fc3e7e4d2a81710a8329271d42b6abc110d5dbe","impliedFormat":1},{"version":"ef8ff23609cec5eb95e2beb98132ad90c0c5075415b50228b12f89ffaf981a4a","impliedFormat":1},{"version":"80bbc9365ca8398c69eae77cdf7284d07192a17dacf1904095ab4c89f4520a5d","impliedFormat":1},{"version":"174a3381f98fc78c451528cb1aa1baaa37a51852ec6fa90d42efd876301537c1","impliedFormat":1},{"version":"2c0de27d99a9331cfac8bc5c6bbd174e0593628bf3df268faa6c4188962a9549","impliedFormat":1},{"version":"1a17bcbc124a098987f7b1adbbcd412f8372ecb37e352b1c50165dac439eee5e","impliedFormat":1},{"version":"0ef49170735d9e5902f55b72465accadd0db93cae52544e3c469cbc8fbdbf654","impliedFormat":1},{"version":"f68a30e88dfa7d12d8dd4609bc9d5226a31d260bf3526de5554feed3f0bf0cb6","impliedFormat":1},{"version":"d8acc6f92c85e784acbbc72036156a4c1168a18cba5390c7d363040479c39396","impliedFormat":1},{"version":"1fffef141820a0556f60aa6050eccb17dbcdc29ecd8a17ee4366573fd9c96ce3","impliedFormat":1},{"version":"d2598c755c11170e3b5f85cd0c237033e783fd4896070c06c35b2246879612b8","impliedFormat":1},{"version":"8d2044a28963c6c85a2cf4e334eb49bb6f3dd0c0dfe316233148a9be74510a0e","impliedFormat":1},{"version":"2660eb7dba5976c2dcbea02ec146b1f27109e7bee323392db584f8c78a6477dd","impliedFormat":1},{"version":"54a4f21be5428d7bff9240efb4e8cae3cb771cad37f46911978e013ff7289238","impliedFormat":1},{"version":"10837df0382365c2544fb75cb9a8f6e481e68c64915362941b4ea4468fd0ef61","impliedFormat":1},{"version":"cc4483c79688bd3f69c11cb3299a07d5dcf87646c35b869c77cde553c42893cf","impliedFormat":1},{"version":"faf76eeb5dd5d4d1e37c6eb875d114fa97297c2b50b10e25066fed09e325a77a","impliedFormat":1},{"version":"b741703daf465b44177ef31cc637bde5cd5345e6c048d5807108e6e868182b01","impliedFormat":1},{"version":"9c3e59360437a3e2a22f7f1032559a4c24aba697365b62fb4816b7c8c66035b8","impliedFormat":1},{"version":"393446ab3f0dd3449ad6fd4c8abd0c82b711c514b9e8dfbf75222bbc48eb0cb6","impliedFormat":1},{"version":"ea02a962453ec628e886a6c5d0fc03bf4da9dfa38e1f8d42e65e07b2651edd85","impliedFormat":1},{"version":"5eb09226bfa1928721a438e37c004647fc19d8d1f4817bddcc350e57fb32935f","impliedFormat":1},{"version":"5994ed389d7fc28c03dad647ecb62e5349160bde443b0c7a54e0e10d6368bcbd","impliedFormat":1},{"version":"e1ff7df643e1aa1dbf1863113a913358844ed66f1af452e774834b0008e578b2","impliedFormat":1},{"version":"c5114285d0283d05e09cd959e605a4f76e5816c2fbe712241993fd66496083e5","impliedFormat":1},{"version":"2752e949c871f2cbd146efa21ebc34e4693c0ac8020401f90a45d4e150682181","impliedFormat":1},{"version":"c349cea980e28566998972522156daac849af8a9e4a9d59074845e319b975f5d","impliedFormat":1},{"version":"0370682454d1d243b75a7c7031bc8589531a472e927b67854c1b53b55ee496ea","impliedFormat":1},{"version":"cf6b4dbb5a1ac9ece24761c3a08682029851b292b67113a93b5e2bfd2e64e49d","impliedFormat":1},{"version":"baa9fbd480342a1d5e3e11ba3629f2826d18d4a765f1f9693ab87bfb3ce54adb","impliedFormat":1},{"version":"cb2fea712720bb7951d7e5d63db8670bf4a400d3e0fb197bceb6ef44efe36ec3","impliedFormat":1},{"version":"1b4fcfc691980d63a730d47d5309d9f85cdddc18a4c83f6e3af20936d103e3ff","impliedFormat":1},{"version":"ef19d5fe42541f8b529bccd10f488d12caefa3b57a0deb1ed6143219cba716b4","impliedFormat":1},{"version":"84b5e6269d7cf53008a479eeb533ef09d025eafb4febe3729301b8d4daf37ff2","impliedFormat":1},{"version":"04196b5d9edd60b9648daa329c3355d7c95f33b7e520e7835eb21002174a8b8c","impliedFormat":1},{"version":"637c0d7d8cedbc64a3c228c3fa6bef884746f7a16a631e7532f9828c9ac06b8a","impliedFormat":1},{"version":"9e665aea79b702fd612ffb7ac741e4160d35d8d696a789129ebcbaea003beb3d","impliedFormat":1},{"version":"c8eeffebe6c2c6800f73aa59d1436d4dadbad7f3ddda02a831ffa66114c3122d","impliedFormat":1},{"version":"caf3f141f93cbf527ad18ecce326311d70342fe1e16ce93e5ce8d6bcdf02bd48","impliedFormat":1},{"version":"4283d88023e6e9645626475e392565464eae99068f17e324cfc40a27d10fe94f","impliedFormat":1},{"version":"51e3b73dea24e2a9638345fb7a2a7ef5d3aa2e7a285ad6bd446b45fab826def1","impliedFormat":1},{"version":"77c4c9f71f3736ed179043a72c4fad9832023855804fbe5261a956428b26a7a6","impliedFormat":1},{"version":"7232467057ec57666b884924f84fd21cd3a79cc826430c312e61a5bc5758f879","impliedFormat":1},{"version":"624f5dbfd76f2d77f20ace318e8cb918608a296106e55587fb443ef3030c595d","impliedFormat":1},{"version":"c78bb1275f640e4902ad5c3383ab4f54f73322a59c95924ab671125ba9546294","impliedFormat":1},{"version":"1cb0838371e8213ce116a1497bb86bcf01a11a755b77587980ee7cfb2d625ece","impliedFormat":1},{"version":"f5d29fd7099274774c203d94d8c0238770ab411b922b978be15a2c3ec8ab845c","impliedFormat":1},{"version":"6d99b5b226a65890ce27796e086d58c6351f601757c1e9f217a69e944d05e7e6","impliedFormat":1},{"version":"10b322f5bc001bec9bf08513c978c120adb0abe3c82793b11bdaf75873426c05","impliedFormat":1},{"version":"51b4efdc8dc92bc6ae2c44d4edad265decad70e8577d5653fc7f85200cbf6c6e","impliedFormat":1},{"version":"c3fa40ac56aa2598d9133c90b115eeb39bbad56c6dfca350dc8435b8b107fe26","impliedFormat":1},{"version":"cc542183b68b048a8cf64eb6231b3d0852f7f4d0191d4637c9d1d4c3f44b83b5","impliedFormat":1},{"version":"4b954a3d432dca82c787c06d2f1cca0fe673a4b440c5e0195429bd1fe43b324a","impliedFormat":1},{"version":"c6fd975d319a70d6ba90bf38c34ac8efebe531214038fe561a27f89f2203f78e","impliedFormat":1},{"version":"a818204639081cf07d80885b88aff5120e5a4135211162f5e08cfc00ef3bf5b6","impliedFormat":1},{"version":"c194ca06da86829b836bb188dffc05543bbea3cbda797667c7a7cade2f907646","impliedFormat":1},{"version":"6df6afb0424a7c7581ee98a9333d30e893b943d0a4709b88f18c252ddc3101b4","impliedFormat":1},{"version":"59c2cbf84c22fae87f4f506f36a7258a72b931b602115067dfd6008ee526f8c0","impliedFormat":1},{"version":"1e09cd1bc6b6baa0733e1e799c4533105ea79cbb109937c71e8c870e14693216","impliedFormat":1},{"version":"0b60cfcd94fa9bd9fa58176650c7e4c72f99b9d30a50d0b55aa08b510276af96","impliedFormat":1},{"version":"ba25681012e5117866a2456dd3557e24aa5a946ed641126aa4469880db526883","impliedFormat":1},{"version":"2b1e058a8c3944890c7ce7c712ecfd0f2645420ee67537ac031d7afe6feda6e0","impliedFormat":1},{"version":"175dbcd1f226eebd93fd9628e9180fb537bb1171489b33db7b388ef0f4e73b37","impliedFormat":1},{"version":"69ec6331ee3a7cd6bade5d5f683f1705c1041ff77432aa18c50d2097e61f93db","impliedFormat":1},{"version":"06f34a0f2151b619314fc8a54e4352a40fd5606bda50623c326c3be365cc1ef9","impliedFormat":1},{"version":"6c6dcb49af3d72d823334f74a554b2f9917e3a59b3219934b7ae9e6b03a3e8b4","impliedFormat":1},{"version":"9628be9799a060a3f7fe2e1f08fab2b21cdd7e97a2bbc3ef2f0029be46e0d7da","impliedFormat":1},{"version":"3d24aec533fe2f035b0675ba1c0e55e8680a714fff2a517e0fb388279476701c","impliedFormat":1},{"version":"224e2edff4c1e67d9c5179aa70e31d0dc7dd4ea5a9e80ffde121df9e5254eef2","impliedFormat":1},{"version":"e324c3b2058f9525cf5c11915284f9dfdf7550c98f103429b271fe723c4f8e14","impliedFormat":1},{"version":"70a3659d557bb683091f9d318762a330a3acb3954f5e89e5134d24c9272192f1","impliedFormat":1},{"version":"d9fe2c804f7db2f19e4323601278b748dc2984798f265c37cd37bb84e6c88ab8","impliedFormat":1},{"version":"3525647a73ae2124fa8f353f0a078b44ff1ee6f82958c2bb507de61575f12fff","impliedFormat":1},{"version":"d7238315cbd18ebeed93f41ad756a0ed9759824b9b158c3d7a1e0b71682d8966","impliedFormat":1},{"version":"eeba7376ce9721610d3282a4159f3c60154b7b3877fb251f7b3211b085cfdc18","impliedFormat":1},{"version":"643efb9d7747ee1dd50ff5bd4b7a87351157e55988c7d2f90ffbdf124f063931","impliedFormat":1},{"version":"788c870cac6b39980a5cc41bf610b1873952ecdd339b781f0687d42682ffc5dc","impliedFormat":1},{"version":"d51a2e050c8a131b13ec9330a0869e5ac75b9ac4ebde52d5f474e819510b5263","impliedFormat":1},{"version":"b694593470a9bf370987e5b0757d5a9a88a46a703c9cf7921969f3379ce16148","impliedFormat":1},{"version":"6c034655fa83236bd779cacfc1d5b469d6e2150a1993e66ecca92376a8b2c6a7","impliedFormat":1},{"version":"6bd6933efe9d6263d9f1a534a28a8f88b1e4c331b95d85d39350cf02eca8dce0","impliedFormat":1},{"version":"658cf468a05b2b591fcd5455a76d9927face59ac4a21b4965982b3c234f5d289","impliedFormat":1},{"version":"6bf893d1b824bde22ee5880c0c760c1dd0a5163c38d22311441a3341b6965d2d","impliedFormat":1},{"version":"579d9d3c25058b854a6f7cc6368a473efcaa0740f45db13cb508761d35fc0156","impliedFormat":1},{"version":"68705604f0666ba3862670153eb4f965c3079415e7ab30a35b3126e36277dc9e","impliedFormat":1},{"version":"28b415e70f9da0346545b7d2bcf361844a8e5778bd6b45bc1a2859f99700ff5b","impliedFormat":1},{"version":"a905f2f6785e3971bd97c42191394209d97f2aefb11841f7353dd9789821fa8c","impliedFormat":1},{"version":"e099c5ebddf80ae7285d380c7dd3b5d49c1347346ced51ae121b846833a8d102","impliedFormat":1},{"version":"aec91730b9f4d83758b4a45596317d34d6ecdbe9330a44629f53af47641b96ee","impliedFormat":1},{"version":"2321197343254570a8d4c868572059bfdfb683cf9d4099b6d4694250dac69471","impliedFormat":1},{"version":"18a3be03c31356b60ea1090bcc905d99e4983ca911cc70b34ad0b9b4d4e050c3","impliedFormat":1},{"version":"9833a67663f960dc2d1908a19365ddde55c0651235596ac60d7078a9be6f6e56","impliedFormat":1},{"version":"2bcb8920601b80911430979b6db4a58a7908a31334e74e4e22b75c65edce3587","impliedFormat":1},{"version":"c3186dc74d62d0fb6fba29841ccbf995614992526c37fac5c082d0f28b351e54","impliedFormat":1},{"version":"2306daed18f7f59542a99857a678ef818058eefa30c2a556af123a1cf53889cd","impliedFormat":1},{"version":"b41ed9285a09710807ce2c423e038dfe538e46e9183c0c05aadc27bfb9ae256a","impliedFormat":1},{"version":"56b9f9de03f28eb5922750a213d3f47b21a4f00a48c7c9b89bf1733623873d3a","impliedFormat":1},{"version":"2bdd736078e445858cb1d9df809ff3a2f00445d78664dd70b6794fb2156bdd53","impliedFormat":1},{"version":"2653fb2893a65c610ec17d0e454e2b16726f16118425f0bc8a38c801943ef7f5","impliedFormat":1},{"version":"74ffa4541a56571f379060acaf9ab86da6c889dfe1f588425807e0117e62bba5","impliedFormat":1},{"version":"cf4dc15ca9dc6c0995dd2a9264e5ec37d09d9d551c85f395034e812abdf60a99","impliedFormat":1},{"version":"73e8b003f39c7ce46d2811749dab1dd1b309235fd5c277bd672c30a98b5cf90f","impliedFormat":1},{"version":"4cb49e79595c6413fcb01af55a8a574705bf385bd2ec5cf8b777778952e2914a","impliedFormat":1},{"version":"d6b44382b2670f38c8473e7c16b6e8a9bfa546b396b920afc4c53410eeb22abf","impliedFormat":1},{"version":"3b5c6f451b7ad87e3fcd2008d3a6cb69bd33803e541e9c0fe35754201389158f","impliedFormat":1},{"version":"8329556a2e85e3c3ff3dff43141790ff624b0f5138cedec5bb793164cf8b088f","impliedFormat":1},{"version":"4c889ce7e61ca7f3b7733e0d2be80b3af373e080c922e04639aa25f22963ae63","impliedFormat":1},{"version":"2239a8cd90c48e0b5c075e51099e7e3b4fc3d4741e4d9cc4410d2544d4216946","impliedFormat":1},{"version":"f5aa57712223d7438799be67b0c4a0e5ac3841f6397b5e692673944374f58a83","impliedFormat":1},{"version":"774c37f8faed74c238915868ccc36d0afedfbafb1d2329d6a230966457f57cbd","impliedFormat":1},{"version":"bc41b711477270e8d6f1110d57863284d084b089a22592c7c09df8d4cc3d1d20","impliedFormat":1},{"version":"0c792fe4e5f383b4f085a0033553fb84ed9322b7923fd59d4575aa43135e050d","impliedFormat":1},{"version":"228ed3721f42cc25bfebceef33754ce4766414d975ff71d012f01f141dbe3549","impliedFormat":1},{"version":"08985cdb65bbfe3c70d0037794a3d0f0a5613f55c278c77277a7acc17205db57","impliedFormat":1},{"version":"22bdefb6b2107006ab203073218566443a52ab65eb5e4e8e86c3d38efe776588","impliedFormat":1},{"version":"0f01b48cee64391fabef3f344e6e86197dc921f0f88a6d45d133ac58283d9690","impliedFormat":1},{"version":"c86fea295c21ea01c93410eba2ec6e4f918b97d0c3bf9f1bb1960eabe417e7eb","impliedFormat":1},{"version":"05d41b3e7789381ff4d7f06d8739bf54cc8e75b835cb28f22e59c1d212e48ff3","impliedFormat":1},{"version":"6fbcfc270125b77808679b682663c7c6ad36518f5a528c5f7258bcd635096770","impliedFormat":1},{"version":"9d3bd4ee558de42e9d8434f7293b404c4b7a09b344e77c36bbe959696328d594","impliedFormat":1},{"version":"f63be9b46a22ee5894316cf71a4ba7581809dd98cf046109060a1214ee9e2977","impliedFormat":1},{"version":"dd3cc41b5764c9435b7cae3cc830be4ee6071f41a607188e43aa1edeba4fbb3e","impliedFormat":1},{"version":"b2dbb9485701a1d8250d9a35b74afd41b9a403c32484ed40ed195e8aa369ae70","impliedFormat":1},{"version":"5aa7565991c306061181bd0148c458bcce3472d912e2af6a98a0a54904cd84fc","impliedFormat":1},{"version":"9629e70ae80485928a562adb978890c53c7be47c3b3624dbb82641e1da48fd2f","impliedFormat":1},{"version":"c33d86e1d4753d035c4ea8d0fdb2377043bc894e4227be3ceabc8e6a5411ab2e","impliedFormat":1},{"version":"f9ec74382c95cbc85804daf0e9dabed56511a6dfb72f8a2868aa46a0b9b5eafc","impliedFormat":1},{"version":"1ff7a67731e575e9f31837883ddfc6bfcef4a09630267e433bc5aea65ad2ced4","impliedFormat":1},{"version":"0c4f6b6eb73b0fa4d27ce6eef6c2f1e7bd93d953b941e486b55d5d4b22883350","impliedFormat":1},{"version":"af9692ce3b9db8b94dcfbaa672cb6a87472f8c909b83b5aeea043d6e53e8b107","impliedFormat":1},{"version":"782f2628a998fd03f4ccbe9884da532b8c9be645077556e235149ca9e6bd8c7d","impliedFormat":1},{"version":"269b7db8b769d5677f8d5d219e74ea2390b72ea2c65676b307e172e8f605a74a","impliedFormat":1},{"version":"ae731d469fae328ba73d6928e4466b72e3966f92f14cd1a711f9a489c6f93839","impliedFormat":1},{"version":"90878ed33999d4ff8da72bd2ca3efb1cde76d81940767adc8c229a70eb9332b2","impliedFormat":1},{"version":"d7236656e70e3a7005dba52aa27b2c989ba676aff1cab0863795ac6185f8d54f","impliedFormat":1},{"version":"e327901e9f31d1ad13928a95d95604ee4917d72ad96092da65612879d89aba42","impliedFormat":1},{"version":"868914e3630910e58d4ad917f44b045d05303adc113931e4b197357f59c3e93e","impliedFormat":1},{"version":"7d59adb080be18e595f1ce421fc50facd0073672b8e67abac5665ba7376b29b9","impliedFormat":1},{"version":"275344839c4df9f991bcf5d99c98d61ef3ce3425421e63eeb4641f544cb76e25","impliedFormat":1},{"version":"c4f1cc0bd56665694e010a6096a1d31b689fa33a4dd2e3aa591c4e343dd5181c","impliedFormat":1},{"version":"81c3d9b4d90902aa6b3cbd22e4d956b6eb5c46c4ea2d42c8ff63201c3e9676da","impliedFormat":1},{"version":"5bfc3a4bd84a6f4b992b3d285193a8140c80bbb49d50a98c4f28ad14d10e0acc","impliedFormat":1},{"version":"a7cf6a2391061ca613649bc3497596f96c1e933f7b166fa9b6856022b68783ab","impliedFormat":1},{"version":"864c844c424536df0f6f745101d90d69dd14b36aa8bd6dde11268bb91e7de88e","impliedFormat":1},{"version":"c74a70a215bbd8b763610f195459193ab05c877b3654e74f6c8881848b9ddb7f","impliedFormat":1},{"version":"3fa94513af13055cd79ea0b70078521e4484e576f8973e0712db9aab2f5dd436","impliedFormat":1},{"version":"48ffc1a6b67d61110c44d786d520a0cba81bb89667c7cdc35d4157263bfb7175","impliedFormat":1},{"version":"7cb4007e1e7b6192af196dc1dacd29a0c3adc44df23190752bef6cbbc94b5e0b","impliedFormat":1},{"version":"3d409649b4e73004b7561219ce791874818239913cac47accc083fad58f4f985","impliedFormat":1},{"version":"051908114dee3ca6d0250aacb0a4a201e60f458085177d5eda1fc3cde2e570f3","impliedFormat":1},{"version":"3e8240b75f97eb4495679f6031fb02ad889a43017cae4b17d572324513559372","impliedFormat":1},{"version":"d82609394127fb33eed0b58e33f8a0f55b62b21c2b6c10f1d7348b4781e392cb","impliedFormat":1},{"version":"b0f8a6436fbaf3fb7b707e2551b3029650bfaeb51d4b98e089e9a104d5b559b5","impliedFormat":1},{"version":"eae0ac4f87d56dcf9fbcf9314540cc1447e7a206eee8371b44afa3e2911e520c","impliedFormat":1},{"version":"b585e7131070c77b28cc682f9b1be6710e5506c196a4b6b94c3028eb865de4a7","impliedFormat":1},{"version":"b92ac4cc40d551450a87f9154a8d088e31cff02c36e81db2976d9ff070ba9929","impliedFormat":1},{"version":"6f99b4a552fbdc6afd36d695201712901d9b3f009e340db8b8d1d3415f2776f5","impliedFormat":1},{"version":"43700e8832b12f82e6f519b56fae2695e93bb18dddb485ddea6583a0d1482992","impliedFormat":1},{"version":"e8165ea64af5de7f400d851aeea5703a3b8ac021c08bebc958859d341fa53387","impliedFormat":1},{"version":"6db546ea3ced87efda943e6016c2a748e150941a0704af013dfe535936e820e1","impliedFormat":1},{"version":"f521c4293b6d8f097e885be50c2fef97de3dd512ad26f978360bb70c766e7eae","impliedFormat":1},{"version":"a0666dfd499f319cc51a1e6d9722ed9c830b040801427bbdd2984b73f98d292a","impliedFormat":1},{"version":"a7d86611d7882643dd8c529d56d2e2b698afd3a13a5adc2d9e8157b57927c0da","impliedFormat":1},{"version":"7e4615c366c93399f288c7bfbaa00a1dc123578be9d8ac96b15d489efc3f4851","impliedFormat":1},{"version":"f2e6c87a2c322ee1473cb0bd776eb20ee7bff041bc56619e5d245134ab73e83d","impliedFormat":1},{"version":"ee89bc94431b2dfaf6a7e690f8d9a5473b9d61de4ddcb637217d11229fe5b69f","impliedFormat":1},{"version":"a19c1014936f60281156dd4798395ad4ab26b7578b5a6a062b344a3e924a4333","impliedFormat":1},{"version":"5608be84dd2ca55fc6d9b6da43f67194182f40af00291198b6487229403a98fe","impliedFormat":1},{"version":"4a800f1d740379122c473c18343058f4bd63c3dffdef4d0edba668caa9c75f54","impliedFormat":1},{"version":"8e6868a58ca21e92e09017440fdb42ebfe78361803be2c1e7f49883b7113fdc2","impliedFormat":1},{"version":"2fbb72a22faefa3c9ae0dfb2a7e83d7b3d82ec625a74a8800a9da973511b0672","impliedFormat":1},{"version":"3e8c1a811bad9e5cd313c3d90c39a99867befa746098cdad81a9578ac3392541","impliedFormat":1},{"version":"d88f78b4e272864f414d98e5ed0996cd09f7a3bb01c5b7528320386f7383153d","impliedFormat":1},{"version":"0b9c34da2c6f0170e6a357112b91f2351712c5a537b76e42adfee9a91308b122","impliedFormat":1},{"version":"47adac87ec85a52ed2562cb4a3b441383551727ed802e471aa05c12e7cc7e27e","impliedFormat":1},{"version":"d1cacf181763c5d0960986f6d0abd1a36fc58fc06a707c9f5060b6b5526179ca","impliedFormat":1},{"version":"92610d503212366ff87801c2b9dc2d1bccfa427f175261a5c11331bc3588bb3f","impliedFormat":1},{"version":"805e2737ce5d94d7da549ed51dfa2e27c2f06114b19573687e9bde355a20f0ff","impliedFormat":1},{"version":"a37b576e17cf09938090a0e7feaec52d5091a1d2bbd73d7335d350e5f0e8be95","impliedFormat":1},{"version":"98971aa63683469692fef990fcba8b7ba3bae3077de26ac4be3e1545d09874b8","impliedFormat":1},{"version":"c6d36fa611917b6177e9c103a2719a61421044fb81cdd0accd19eba08d1b54de","impliedFormat":1},{"version":"088592cf2e218b99b02a5029ed8d1a763a3856cd25e012cfbb536b7494f08971","impliedFormat":1},{"version":"5eb39c56462b29c90cb373676a9a9a179f348a8684b85990367b3bbc6be5a6e9","impliedFormat":1},{"version":"52252b11bcbfaeb4c04dc9ec92ea3f1481684eee62c0c913e8ff1421dc0807e5","impliedFormat":1},{"version":"731d07940d9b4313122e6cc58829ea57dcc5748003df9a0cad7eb444b0644685","impliedFormat":1},{"version":"b3ead4874138ce39966238b97f758fdb06f56a14df3f5e538d77596195ece0b5","impliedFormat":1},{"version":"032b40b5529f2ecce0524974dbec04e9c674278ae39760b2ee0d7fce1bb0b165","impliedFormat":1},{"version":"c25736b0cb086cd2afa4206c11959cb8141cea9700f95a766ad37c2712b7772b","impliedFormat":1},{"version":"033c269cd9631b3f56bb69a9f912c1f0d6f83cf2cff4d436ee1c98f6e655e3b5","impliedFormat":1},{"version":"bd6d692a4a950abbfabe29131420abe804e7f3cc187c3c451f9811e9cf4408ce","impliedFormat":1},{"version":"a9b6411417d4bffd9a89c41dc9dedda7d39fb4fa378eaa0ab55ec9ea1a94eb6a","impliedFormat":1},{"version":"1329e7cd7aca4d223ef5a088d82bc3f6f302ce70581c8d3823a050ea155eec3b","impliedFormat":1},{"version":"09248c76437c5b1efce189b4050c398f76a9385135af75c5fb46308b0d1432e0","impliedFormat":1},{"version":"b8df115bf7b30cceeb4550c0be507082b9930ee6268539a1a1aaffb0791cc299","impliedFormat":1},{"version":"dde00f41a2d2b1e70df6df8ac33de7cb3a658956212c7bee326245cc01c990c2","impliedFormat":1},{"version":"115d092e2748990ff0f67f376f47e9a45a2f21f7c7784102419c14b32c4362d1","impliedFormat":1},{"version":"4ba068163c800094cd81b237f86f22c3a33c23cf2a70b9252aca373cfdf59677","impliedFormat":1},{"version":"53e65282ab040a9f535f4ad2e3c8d8346034d8d69941370886d17055874b348d","impliedFormat":1},{"version":"e6db934da4b03c1f4f1da6f4165a981ec004e9e7d956c585775326b392d4d886","impliedFormat":1},{"version":"6ecb85c8cbb289fe72e1d302684e659cc01ef76ae8e0ad01e8b2203706af1d56","impliedFormat":1},{"version":"fca410876e0302680190982f2fc5102d896e65e4f4f20547a185b60364838910","impliedFormat":1},{"version":"601bc70ff67ae9855fc65bad9bb2d135f72147cf22e2490f58ea0d209d95f2ee","impliedFormat":1},{"version":"5cd5a999e218c635ea6c3e0d64da34a0f112757e793f29bc097fd18b5267f427","impliedFormat":1},{"version":"de8a12540370f9f18b160a07ed57917d69fe24525d360531d42d4b1b5d0d9f0f","impliedFormat":1},{"version":"4a397c8a3d1cccf28751bcca469d57faeb637e76b74f6826e76ad66a3c57c7b8","impliedFormat":1},{"version":"34c1bb0d4cf216f2acb3d013ad2c79f906fe89ce829e23a899029dfa738f97e0","impliedFormat":1},{"version":"5c744f3cc0a266dd95b5769a70ddc85c8b6019adbb0954d4de61f89182202ce3","impliedFormat":1},{"version":"b50f05738b1e82cbb7318eb35a7aaf25036f5585b75bbf4377cfa2bad15c40bf","impliedFormat":1},{"version":"c682cb23f38a786bb37901b3f64727bd3c6210292f5bb36f3b11b63fbe2b23ee","impliedFormat":1},{"version":"d6592cf10dc7797d138af32800d53ff4707fdcd6e053812ce701404f5f533351","impliedFormat":1},{"version":"997f6604cd3d35281083706aa2862e8181ed1929a6cbb004c087557d6c7f23c4","impliedFormat":1},{"version":"9584dd669a3bf285e079502ebbb683e7da0bf7f7c1eb3d63f6ef929350667541","impliedFormat":1},{"version":"41a10e2db052a8bf53ed4d933d9b4f5caa30bdaee5a9d978af95f6641ce44860","impliedFormat":1},{"version":"d84761f8a994b5444529c7c294b194de6fd5350ccda974929ea7e8b3893b753a","impliedFormat":1},{"version":"652e51858bafd77e1abcc4d4e9d5e48cc4426c3dd2910021abd8cc664961e135","impliedFormat":1},{"version":"8c5c602045ffdfebeffc7a71cd2bf201fe147a371274b5fcbded765a92f2af78","impliedFormat":1},{"version":"6392ce794eef6f9b57818264bb0eeb24a46cf923f7695a957c15d3d087fbb6cc","impliedFormat":1},{"version":"b10f123e8100aa98723c133af16f1226a6360ec5b6990a0fe82b165d289549db","impliedFormat":1},{"version":"93d20368cdb5fff7f7398bfc9b2b474b2a2d5867277a0631a33b7db7fd53d5b4","impliedFormat":1},{"version":"b1e69b9834104482fabf7fba40e86a282ee10e0600ffd75123622f4610b0ef9e","impliedFormat":1},{"version":"ad5bb6c450cb574289db945ff82be103ed5d0ad8ee8c76164cee7999c695ae01","impliedFormat":1},{"version":"217761e8a5482b3ad20588a801521c2f5f9f7fb2fbb416d4eff3aff9b57f8471","impliedFormat":1},{"version":"7ad780687331f05998c62277d73b6f15ee3e8045b0187a515ffc49c0ad993606","impliedFormat":1},{"version":"e9aa5ccb42e118f5418721d2ac8c0ebdebeb9502007db9b4c1b7c9b8d493013e","impliedFormat":1},{"version":"d300868212b3cc4d13228f5dc2e9880d5959dc742c0c55be2fc43bcda8504c8f","impliedFormat":1},{"version":"0c55daad827669843bd2401f1ddd163b74d9f922680b08ae6e162ceb6c11b078","impliedFormat":1},{"version":"fe45a9bc654dfd1550c9466c0dad9c8017f2626476ed9d25c65ddfc1943f6b74","impliedFormat":1},{"version":"03abcbc7b5b68887525be71a194dd7f9f68276b5fb5b8989abae9a91585ddc33","impliedFormat":1},{"version":"5055e86e689cfe39104ab71298757e5aac839c2ea9d1f12299e76fa79303d47d","impliedFormat":1},{"version":"42266c387025558423c19d624f671352aac3e449c23906cb636f9ae317b72d7e","impliedFormat":1},{"version":"e578a36b3683d233e045a85c9adb0f10e83d2b48f777b9c05fbc363ccc6bdd34","impliedFormat":1},{"version":"0235d0ba0c7b64244d4703b7d6cabd88ba809abeb01da0c13e9ed111bf5e7059","impliedFormat":1},{"version":"9b21e8a79f4213c1cf29f3c408f85a622f9eb6f4902549ccb9a2c00717a0b220","impliedFormat":1},{"version":"d556e498591413e254793f9d64d3108b369a97bd50f9dd4015b5552888e975ef","impliedFormat":1},{"version":"e2c652c7a45072e408c1749908ca39528d3a9a0eb6634a8999b8cf0e35ef20c8","impliedFormat":1},{"version":"ec08224b320739d26aaf61cead7f1e0f82e6581df0216f6fe048aa6f5042cb8c","impliedFormat":1},{"version":"4eadaa271acca9bd20fc6ac1ea5e4bf9ab6698b8ccf3ec07c33df4970f8130f1","impliedFormat":1},{"version":"3238d2eee64423c8d41972c88673b0327d8b40174a78ea346bcd10954a8f3373","impliedFormat":1},{"version":"8f773ddff9070d725dd23f5cf6c8e62bd86984a57b5d5e3fc7583010b48cd8ac","impliedFormat":1},{"version":"5ecd8fdeb6c87db9c320eefbfa9ea27efccbdce853ed38d5ba58e2da482edf1f","impliedFormat":1},{"version":"19a4d116285e7d77e91411966930761a2204ce2d20915afdb12652681a4a88d7","impliedFormat":1},{"version":"c30ca82112586c5dae7477d7e82cc91a7e0d1e658c581f9ec3df07c4485bba84","impliedFormat":1},{"version":"68fca1813d17ee736f41124ccc958d0364cdef79ad1222951bfacc36b2630a58","impliedFormat":1},{"version":"7813329e568df1d42e5a6c52312b1a7c69700e35a561cf085158c345be155b22","impliedFormat":1},{"version":"561067dc7b6b7635277d3cad0a0e11f698d377063dd2c15dfac43ef78847eef4","impliedFormat":1},{"version":"438247e782a8a9b9abdce618e963667cf95157cc6d3f5194a452d3c7d9e9655c","impliedFormat":1},{"version":"0c293195f800014f1fa3ffacf979002c8c1886ab71750432813fb590738eeef5","impliedFormat":1},{"version":"7673348e0cc2f4e33d1db02ecda02f39e66e56ab2cc3c5602246e5532f2715ab","impliedFormat":1},{"version":"83724b26b711d85d6cfc9dd92fd5d666ffaae27fcfb1a0110401b98814ea26c0","impliedFormat":1},{"version":"869a27c929366c3c864013a991fd4c4c86af73eba25513e8ae915f814d3d349c","impliedFormat":1},{"version":"bfa105c32ed586b227188f7b568776d03202dc7aa4c3af2746579450c7d5e7f2","impliedFormat":1},{"version":"756e3f41a7f2501a34e1a070283c7f5550e200eeb43fed3c806e3f2edd924a75","impliedFormat":1},{"version":"59935cc13dcb7c3c7825e770a61e6696bfd11b65e3e47c28acc410dbdf8461c0","impliedFormat":1},{"version":"85e2808cc73ab3ac07774802b34a6ff0d7e1e46c26de7bc2dbe08e04b3340edb","impliedFormat":1},{"version":"f766e5cdea938e0c9d214533fd4501ab0ee23ab4efca9edba334fa02d2869f11","impliedFormat":1},{"version":"eb380820a3a1feda3a182a3d078da18e0d5b7da08ae531ce11133a84b479678c","impliedFormat":1},{"version":"7fba5cc3088ad9acada3daeff52dae0f2cac8d84d19508abd78af5924dc96bea","impliedFormat":1},{"version":"14176cfdbc3d1d633ad9b5daf044ab4c7d0d73be61ca2f14388800e21f0989cd","impliedFormat":1},{"version":"a24f510afe4d938d625a4b5a5374ac0478e56305e8743dd7d37d86d709754286","impliedFormat":1},{"version":"648acdbcbcd01b1a91e8b0ad390ed59fada685977f44b90e148b65bd8159dfe8","impliedFormat":1},{"version":"8309898ba0ac6f2856a94a11723d499091253a6d5df34ddebc6149d43480bfd2","impliedFormat":1},{"version":"a317ae0eb092da3fd799d1717a2da319a74abebe85e2914cb259222969f95705","impliedFormat":1},{"version":"36d76e2dbd5f5243bd566b018c589e2ba707e34b24ec7d285feb11ba6bf23fbe","impliedFormat":1},{"version":"f780879a2ca63dbb59b36f772bc28dccd2840f1377d8d632e8c978b99c26a45f","impliedFormat":1},{"version":"335c2e013b572967a9a282a70f9dded38631189b992381f1df50e966c7f315d6","impliedFormat":1},{"version":"8b7a519edbd0b7654491300d8e3cbd2cb3ef921003569ca39ebd33e77479bb99","impliedFormat":1},{"version":"c90f8038c75600e55db93d97bab73c0ab8fb618d75392d1d1ad32e2f6e9c7908","impliedFormat":1},{"version":"ca083f3bf68e813b5bded56ecbf177636aa75833eb86c7b40e3d75b8ce4c2f78","impliedFormat":1},{"version":"3c8bf00283ef468da8389119d3f5662c81106e302c8810f40ea86b1018df647e","impliedFormat":1},{"version":"67b248e4bac845c5139898b44cbd3e1213674bcc9831039701b5f0f957243a24","impliedFormat":1},{"version":"63d49516f359186f7b3e3115f2c829ed75c319b34022c97b56beead032a073b7","impliedFormat":1},{"version":"9f5f256c7b5cc4a98ef557ea9720f81e96319d569f731c897ddb4514936242b4","impliedFormat":1},{"version":"a20ded6c920f6e566537e93d69cbad79bc57d7e3ce85686003078cf88c1c9cfc","impliedFormat":1},{"version":"40b2d781df7b4a76d33454cb917c3883655ec1d8d05424b7a80d01610ad5082f","impliedFormat":1},{"version":"703ea2acd8b4741248897a5709cd46e22fcd9d13f01ff3481322a86505f0b77c","impliedFormat":1},{"version":"e09c56f8c446225e061b53cb2f95fcbbc8555483ab29165f6b0f39bc82c8d773","impliedFormat":1},{"version":"a571973bc2e34c898c3202452f957e6757f0c08cb66d50d6785f4a9042d74bad","impliedFormat":1},{"version":"a6a059446e66fbf5072eccce94eb5587cef2f99aa04d4bbd4ebe63d0a6592a4f","impliedFormat":1},{"version":"6e2533e27eba5ff02d6eed37e0a7eb69ae7982e0f72fd8f74c90ab201f061867","impliedFormat":1},{"version":"9c10dd3d85b7620ed3105b3f018125d0bb54198bf5847e39622afb22c651a1ad","impliedFormat":1},{"version":"58c62e415bf74b1423bf443587e33d7951a8bf19d7b03073f26e86d9b43ba9ea","impliedFormat":1},{"version":"dd6ec67ad168e92b8bf79ba975c6e0be8c60e403ba704d1c1b31a6059c12f967","impliedFormat":1},{"version":"bcaf468eea143f8e68ca40e5da58d640656b4f36697170c339042500be78ac5d","impliedFormat":1},{"version":"92de961d1db5fe075db8c0b6414a6eec430adaf9022465fe9d0a23f437aafcb3","impliedFormat":1},{"version":"7610ecdae59cea1a8db7580941ebc24d522d8ac1751ce718a6af22d41e1a1279","impliedFormat":1},{"version":"3e55a65822875e85f96e444b79787f619b9473e36c143dedc6d5441a2544b8ab","impliedFormat":1},{"version":"d49275f9098a8e7a5df7c55321b0242cef0bfdde51018b7b2709c4dc74917822","impliedFormat":1},{"version":"b25556c4111afad4cb174aa4674db2e5b23a6b191dc6a3e42c7c3417ea446a68","impliedFormat":1},{"version":"f9568a3a6c74013aee8b09d73ef04175596b51ce6f5d9dcd4885418170fe9306","impliedFormat":1},{"version":"bd3910ccd4fcd05ebd83fbfeb62f5a82a6674c85c6c0e4755c16298df7abe4d7","impliedFormat":1},{"version":"7c0541d0addc3007e5f5776023d5e6e44f96eae0684cdabe59ef04f2a294b116","impliedFormat":1},{"version":"70137204b720e4dd1b81260a70578f0f4f417c53837f8a13859b2f58e20d7150","impliedFormat":1},{"version":"b28b6875a761fd153ebf120fecb359660de80fd36e90c9b3d72a12318bd5d789","impliedFormat":1},{"version":"56d092bd6225f6e67d9acab3fd65ce0a4edb36cadba2f0370e67322e2f6f1bc8","impliedFormat":1},{"version":"a4709d5d466ad8dcf4ddccb905ad95348131df1616f964185be9739f96526bde","impliedFormat":1},{"version":"73b0fd6255f24e82be861f800a264f0175984062b6ccca3052578b03ed6f397b","impliedFormat":1},{"version":"4a3f7c6f02cb01eb7a9800548b41cfa03a57e476fc92a72869983f37efa8067a","impliedFormat":1},{"version":"ee428614fde0db27939b30094352fc85b58d1bdba6d63be6d8b2aec5929fd9a2","impliedFormat":1},{"version":"bc0b17d3fd0e34083fbc886367ed53563b569d1d05214f60b21117e2dbfb7fdd","impliedFormat":1},{"version":"6120bbd2dbac7d6bb7005b3e00ddb8e6acb9592a37e2bedf32218ad21da915e5","impliedFormat":1},{"version":"72fa257966dec421bf308d15ccf5ee43c588309942d51dd6330250bb0ab39891","impliedFormat":1},{"version":"9a7638d62db8cfa1466093d7d413fdf85c5e4a7c663ed76f2bfc8739c8e01505","impliedFormat":1},{"version":"058709777c09f2ef9b91ec305d4fd84cfa44eb4a0e39e39a3c759924b352f194","impliedFormat":1},{"version":"c338859b98f8a11f80e3e47e33767299e7a4facdf0870c01c8694fa8fa048d16","impliedFormat":1},{"version":"a6f9821e4b5f28264f61f7a8fccbedb30edf194313088242687c31ddf6c7a336","impliedFormat":1},{"version":"b113e9770d5be136c5e2add9e6cdf40d85051762ff2391f71d552975e66b1500","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"03464dcca517bcfb982cefdc316afe821aae8bbe02dcd4765dfa25bc2aecd097","impliedFormat":1},{"version":"59bdc8b3c0ca88ace4d08cf703a52a14f91ce05e3d66235df792915ea54f67c9","impliedFormat":1},{"version":"8a90c628f293590574bbeb66092271849d180a7f4812cb05233a2c4cb30e0c04","impliedFormat":1},{"version":"d2ab468a72716e9a385b9c0188ddd17045efb781ce90fd9f00141729cdc867e6","impliedFormat":1},{"version":"c3fbb898f4185e04b223a3c406f71be2ce89b58816b95096e91bd40bf74d2a08","impliedFormat":1},{"version":"7bac41f2fcdc718cb06a0caee8796305de3f435a1c3d5a700305f9cb26ab3041","impliedFormat":1},{"version":"e46abaadffe51343e4b50115f22ec40c55efc952e1a5ad8ea83a379e68fdc41b","impliedFormat":1},{"version":"56a44eae80f744ff0ed0ae54ed2c98873d9efaeb94b23102ce3882cbf3c80c87","impliedFormat":1},{"version":"c1608564db1e63ec542694ce8a173bb84f6b6a797c5baf2fdd05de87d96a087f","impliedFormat":1},{"version":"4205f1615444f90977138e01f4c6becc1ae84e09767b84c5a22185ddea2b8ffe","impliedFormat":1},{"version":"823fcbdb4319180e3f9094bc859d85c393200b9568c66f45ba4d5596ace5641d","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"0972ae3e0217c3591053f8db589e40b1bab85f7c126e5cf6cc6f016e757a0d09","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"165181dcaf69484f3a83fef9637de9d56cfa40ee31d88e1a6c3a802d349d32b2","impliedFormat":1},{"version":"823fcbdb4319180e3f9094bc859d85c393200b9568c66f45ba4d5596ace5641d","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"8e517fddbe9660901d0c741161c1ee6674967aaa83c0c84916058a2c21a47feb","impliedFormat":1},{"version":"30f2b1e9cecf6e992ee38c89f95d41aebdb14a109164dd47d7e2aa2a97d16ea9","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"f44bf6387b8c7ab8b6a4f9f82f0c455b33ca7abc499b950d0ef2a6b4af396c2a","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"0a7a83acf2bd8ece46aff92a9dedb6c4f9319de598764d96074534927774223a","impliedFormat":1},{"version":"4f9142ccaefd919a8fe0b084b572940c7c87b39f2fd2c69ecb30ca9275666b3d","impliedFormat":1},{"version":"b80840cbfda90fd14082608e38e9b9c5fde7a0263792c544cddc0034f0247726","impliedFormat":1},{"version":"dcd34efd697cf0e0275eb0889bdd54ca2c9032a162a8b01b328358233a8bcd49","impliedFormat":1},{"version":"98ca8492ccc686190021638219e1a172236690a8b706755abb8f9ff7bb97b63e","impliedFormat":1},{"version":"b61f91617641d713f3ab4da7fdda0ecef11906664550c2487b0ffa8bfbdc7106","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"725d0451e136578def8263b9f5631d45b7c7c54e72a6ce3b524a1fd5bf6a31f5","impliedFormat":1},{"version":"61cc5aabafaa95e33f20f2c7d3289cf4cab048fc139b62b8b7832c98c18de9ef","impliedFormat":1},{"version":"811273181a8489d26cfa0c1d611178ddbeef85ced1faec1a04f62202697a38a5","impliedFormat":1},{"version":"487d2e38f52af45f6c183407858ea3e0a894fb3723c972140436f40878a27e85","impliedFormat":1},{"version":"15e56c8cb8c5515fe9794c5d223ca5c37a302c62183137a595ba657f5d961527","impliedFormat":1},{"version":"fda3db70b49ad94d08ec58caf0ca052e51d38c51d0461a28669a419c67edb396","impliedFormat":1},{"version":"bb7dd4601aaf41b0313503ffc43142a566a87224cc1720cbbc39ff9e26696d55","impliedFormat":1},{"version":"5ef05c11e0fe4120fb0413b18ca56c78e7fe5843682731fe89c6d35f46d0a4ae","impliedFormat":1},{"version":"02c3a89952ea1b30a3573246649c474cd27b17a26d532abed1e152d5981a6b97","impliedFormat":1},{"version":"d2873a33f67fd7d843ead8cebaeebd51ada53f5fc70d4a61e1874c5d2e3fde4b","impliedFormat":1},{"version":"94c6e873b76d2b5094bd2fddd026db85264bc24faa9cb23db9375f1a770312b5","impliedFormat":1},{"version":"2e8e67d756f97ff13764c81f098b9de13ff91e31028890f3dabe9e8d354f7e47","impliedFormat":1},{"version":"a3476600ff22e7d4845d951dbd0548f8d118f2bfe236aaa6ccd695f041f7a1fc","impliedFormat":1},{"version":"02c3a89952ea1b30a3573246649c474cd27b17a26d532abed1e152d5981a6b97","impliedFormat":1},{"version":"a86a43e07633b88d9b015042b9ea799661fe341834f2b9b6484cfa18a3183c74","impliedFormat":1},{"version":"8994f4c217d03e50957cc4693ae5fd35fd15c60c7d77a31528d90cbeb89311df","impliedFormat":1},{"version":"f5db90ab2b03fc1bc55b4d46df4aa6d4cacdbdd1491bcba0a3cf1a73777204d7","impliedFormat":1},{"version":"9fd04134a11f62f6b1523168945b42a74c35ffe1ea94dfdb08ecddf32218c5c2","impliedFormat":1},{"version":"dbe0161c1a41397e79211136cc6d595b10117aa23ac2f17f7484702ada81bc13","impliedFormat":1},{"version":"b21e6c15895ef16c12925295ebbb39f6731a0c74116f7bfdf5a9085040178bac","impliedFormat":1},{"version":"ea9911c1ac347d631cd840485aef26a8079f0ab64019cc90ae6c97d97dd65034","impliedFormat":1},{"version":"e9ff90fbab735e28c091315b542c620141a76f91bb0d56a14178908905e51b35","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"6fcdcc891e7f13ad8bd34c4de33d76d96c84f06d9ab6629620c8cf08d0cc6bea","impliedFormat":1},{"version":"16a187924c639631e4aab3d6ea031492dc0a5973bae7e1026b6a34116bd9ff5c","impliedFormat":1},{"version":"cd78f65631ff21afa0d2d72f47bd7783126e48c986ff47df22d1dc31347730e5","impliedFormat":1},{"version":"f5db90ab2b03fc1bc55b4d46df4aa6d4cacdbdd1491bcba0a3cf1a73777204d7","impliedFormat":1},{"version":"ad068305ead33649eb11b390392e091dbf5f77a81a4c538e02b67b18eb2c23b3","impliedFormat":1},{"version":"8994f4c217d03e50957cc4693ae5fd35fd15c60c7d77a31528d90cbeb89311df","impliedFormat":1},{"version":"caa292653f273a1cee0b22df63ce67417dbc84b795867bf3cd69f7386bb0f73c","impliedFormat":1},{"version":"cbe901efe10faaa15e14472d89b3a47892afc862b91f7a3d6e31abeb3546a453","impliedFormat":1},{"version":"717b25e589f53597f65f42e0ccff891cd22743511c79b50d534d2fa548484937","impliedFormat":1},{"version":"79d5d086cfd15de8c973783e166e689aa29100d0906ccfef52928504949cf8c2","impliedFormat":1},{"version":"15ecea8b0870ebf135faa352b43b8385f5a809e321bb171062da7ad257c9fd08","impliedFormat":1},{"version":"df9712034821067a7a2a0cf49c7bb90778dc39907083fa47b20c3e22c4e62da5","impliedFormat":1},{"version":"6b2394ca4ae40e0a6e693ad721e59f5c64c2d64b3a6271b4f20b27fce6d3c9c2","impliedFormat":1},{"version":"27ea6d85f1ba97aa339451165cae6992c8a6a7b17d3c8468e3d8dce1c97d16cd","impliedFormat":1},{"version":"05751acbcbf5d3ff3d565e17589834a70feb5638ae7ee3077de76f6442b9e857","impliedFormat":1},{"version":"54edf55c5a377ee749d8c48ca5132944906c09f68b86d1d7db4acc53eea70d57","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"bd0923e7cd1c54c64d7396fbd284983003f0e757bd67f3d6cf3a4e5d394128d7","impliedFormat":1},{"version":"b80840cbfda90fd14082608e38e9b9c5fde7a0263792c544cddc0034f0247726","impliedFormat":1},{"version":"4628d6640af9591f1671e0737b3b7de3abe790ff92686a46d6ca5b2e867162c1","impliedFormat":1},{"version":"50145df9cc9bdb77ac65e4622d11fb896b4730f6f727ffd42337a4fdcd2346da","impliedFormat":1},{"version":"0211a096d47b00b5ba4f6a2557184c649db02cb13a8d63f671428c09818b6df8","impliedFormat":1},{"version":"d32d132c14387d64aa1b776f426a5c3ddcf8211d8764526380dda04f9f4dd776","impliedFormat":1},{"version":"af1c879f74fa27f97cf8ae59ed33421826b7d00647c601cafbbeea129ed5ef5b","impliedFormat":1},{"version":"3b47ab89a1b5a0d3943aace80a68b9af7ae671e359836679ff07536c56ada3fa","impliedFormat":1},{"version":"99c0975f5d575eb40fdf0b43fc3e9e8538aa89f47fdf1c20b06bdea609bafc60","impliedFormat":1},{"version":"ae66752cf1b4d08f0b1870dd7c848e491f078116e6395ee5171323c7ec30e92b","impliedFormat":1},{"version":"14a9ec5df1f55a6b37f36d5d91699092119dba1d81defd12151eb0069a26069d","impliedFormat":1},{"version":"ff49d78bd5a137f76e23cc9629105c1d216c43bf68f545acf3f997e838a47ba3","impliedFormat":1},{"version":"842f200637a0e0f390a6512e3e80c8f47c0193bbdff19b5700b070b6b29f1787","impliedFormat":1},{"version":"26a06ef0d60229641de4f9d0ac8566a471b99a3c124e567405a82e77116bee2a","impliedFormat":1},{"version":"f4f34cdbe509c0ae1a7830757a16c1ccb50093b3303af2c301c0007ec2ddf7e0","impliedFormat":1},{"version":"59ba962250bec0cde8c3823fd49a6a25dea113d19e23e0785b05afde795fad20","impliedFormat":1},{"version":"ea930c3c5a401f876daaec88bfc494d0f257e433eaa5f77208cc59e43d29c373","impliedFormat":1},{"version":"318ba92f9fcec5a9533d511ee430f1536e3e833ffe3ea8665d54fe73e28b1ad4","impliedFormat":1},{"version":"adc45c05969fc43d8b5eaac9d5cb96eccf87a6a1bd94498ddd675ea48f1ba450","impliedFormat":1},{"version":"5691d5365f48ff9de556f5883901586f2c9c428bcf75d6eff79615ae1fb67da6","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"a67a76d1886745066bd45956fdc5842812786be2a47285d2c59424882cefd6cf","impliedFormat":1},{"version":"66adf84e776d039acb0207f079934f389147264385fc8847b56481253da99fad","impliedFormat":1},{"version":"d2eee6a9d0b2f4123aba65f6e1bc4df3f973f73a7bdeaa9f76c3c0d3f369bef8","impliedFormat":1},{"version":"8f47038a38222bcbc8551a017ae2e32933ca4e6d2a4ec5cfa01179f1facfa975","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"73c82b8dd8ac2916e7cc44856da0dc795ca9952bb63baa220743d31f62b278e5","impliedFormat":1},{"version":"9e302a99187359decbfba11a58c6c1186722b956f90098bb34d8b161bc342a0d","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"9a06d96357b472809d65ea00b724b4309ba8c9bc1c73eadd3c465e1c336a1e2f","impliedFormat":1},{"version":"ac2b056c5c243b64e85fb8291efd5a1a5481f0bc246b92ea40827ed426ff408c","impliedFormat":1},{"version":"be78757555b38025ba2619c8eb9a3b2be294a2b7331f1f0c88e09bf94db54f3c","impliedFormat":1},{"version":"d68d6551207bf833d92fb7cda4d9428182f8c84eed1743d9a1e7135003e8e188","impliedFormat":1},{"version":"99394e8924c382a628f360a881171304a30e12ac3a26a82aba93c59c53a74a21","impliedFormat":1},{"version":"ed1f01a7eb4058da6d2cde3de9e8463da4351dbab110f50b55e6a7e6261e5e86","impliedFormat":1},{"version":"19ee405d4f1ae4cbacf4361f9a03092a9d69daa3b4ec147c346049d196b5656d","impliedFormat":1},{"version":"6d82ce2eadb900816fb1fa8b62eb4fcf375322bd1fe326b57ef521a0cac3c189","impliedFormat":1},{"version":"19ee405d4f1ae4cbacf4361f9a03092a9d69daa3b4ec147c346049d196b5656d","impliedFormat":1},{"version":"9d344fa3362148f3b55d059f2c03aa2650d5e030b4e8318596ee9bd083b9cf05","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"bfea7300ed7996fd03c8325ce6993eed134984b4bb994b0db8560b206c96f1f7","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"ca87e8ccd63c92b34fc734eee15d8ab2d64f0ffb85d762018bc0df29ca7185b4","impliedFormat":1},{"version":"4628d6640af9591f1671e0737b3b7de3abe790ff92686a46d6ca5b2e867162c1","impliedFormat":1},{"version":"a3913393d42c709b4faea550820241a262a4ba3577f9a00e2f8727eaa92be535","impliedFormat":1},{"version":"5e424456e19df83a4befc6cd24561c2564b7a846b7025a164ce7076ee43828ee","impliedFormat":1},{"version":"887dec57d4c44eaf8f5275c9f5e02721b55c0a34f21f5b6ed08a1414743d8fd9","impliedFormat":1},{"version":"2d53acf155ccbc6b7dca2cfdb01bac84e3571865d925411d2f08ff0445667ea8","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"a7161c3e94028388a80f7091eb2f7f60d2bdde6a58f76876ab30f66c26f6128e","impliedFormat":1},{"version":"381936e93d01e5697c8835df25019a7279b6383197b37126568b2e1dfa63bc14","impliedFormat":1},{"version":"9944093cbb81cc75243b5c779aebfb81fe859b1e465d50cd5331e35f35ef263a","impliedFormat":1},{"version":"fb19163944642017fcdcbdc61999ab21c108334c8b63377184a2a1095698889a","impliedFormat":1},{"version":"839ebe64509ec88e2d7e48cc36bed2b0f52e68b02818478a6e18a88b041ed78a","impliedFormat":1},{"version":"1bd91f5355283c8fa33ad3b3aace6c4ebb499372943a49f57276f29f55fd62c4","impliedFormat":1},{"version":"6535056b39d5e025505b36ec189302e15af7d197a6afd9a3c853187eb1bea7b5","impliedFormat":1},{"version":"34f97cabd716ba01042042f6523183149c573b8fb15a08a3a9524bf1950216ef","impliedFormat":1},{"version":"01911dee2f91c28782c46d57e2e19e250f7c9db4388f8e9945476379e9392d56","impliedFormat":1},{"version":"95ce7b12742f82bddb85134d8ee20a759c698e5d8beefd559fd6e87112fbf72f","impliedFormat":1},{"version":"0b464435da3dd6473694a2128d49f37c9cf43951455c56f0aa5a940f290c69d2","impliedFormat":1},{"version":"75a5fcf80ec969763cb4a31d2cf8b8531b076d6f1ef8699bd9dacca43d34b571","impliedFormat":1},{"version":"b27117352bfa4f1e6fa6874c3f5518252ae2ff30e345d9e505409a75a232372c","impliedFormat":1},{"version":"d21630c0cd7409e8078cc0aeebf3cf8b915888553d7c9c2d9debd918bfd4bebb","impliedFormat":1},{"version":"7e7a2691f49c7d2623b8a531c9eb4005c22daa57e7789f1982c19fe3c1bf55eb","impliedFormat":1},{"version":"80c54f1d257a28de68ec6c23ca7da374071646182d9a2d2106a91606ebc15f52","impliedFormat":1},{"version":"55ba9e8cb3701eff791fccbe92ef441d19bc267b8aab1f93d4cac0d16fffa26a","impliedFormat":1},{"version":"a40e9367d94ec1db62a406d6e1cb589107ea6ad457af08b544e18d206a6ae893","impliedFormat":1},{"version":"12b260ecee756ba93760308b75a8445f2fe6a1cff3f918cf7e256e3d6d1066cc","impliedFormat":1},{"version":"181de508acbe6fe1b6302b8c4088d15548fb553cb00456081d1e8d0e9d284a24","impliedFormat":1},{"version":"ead149a41e9675c986e6d87c9309e751a8c2d0521839a1902f05ec92b2cba50b","impliedFormat":1},{"version":"d15a8152e6df11bfad2d6813f4517aa8664f6551b0200eca7388e5c143cd200d","impliedFormat":1},{"version":"98884645b61ad1aa2a0b6b208ebaab133f9dd331077a0af4ec395e9492c8d275","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"f660100bff4ca8c12762518ba1c1d62dd72ee1daa7ea42f7eae2f72e993bec6f","impliedFormat":1},{"version":"fd7140ce6b8fc050547d7da8696ed2bcdf4cabc4e65f40f4ac1b080f694711d8","impliedFormat":1},{"version":"8689dabe861fb0bdb3f577bdd9cca3990b14244d1d524c7bdb8d89e229c903a6","impliedFormat":1},{"version":"15d728b5790c39ce9abbd1363e0a5ed03ee6b59a38ee3c4d9d25476641baa7a5","impliedFormat":1},{"version":"95159570a0fc2b007b1a46ed8caf145ad6711030c0c4727cee979a3b770b0634","impliedFormat":1},{"version":"e5446a2b0c44d21a4e2ed885bbdb40a4e39a184f9155f13717993782e313bc7e","impliedFormat":1},{"version":"8683b5b593a5fd2cf99212195ba25106e61a546169068626c8a3745ec6e94bed","impliedFormat":1},{"version":"3f72337d957fd6c87b5c8628c85633d7314b8539cc641ea71a6f93a71f7533c2","impliedFormat":1},{"version":"5d0975641e296dba1ebaf16bb987a2b3abe0a62d18fa1396f57c9d4aaead48e8","impliedFormat":1},{"version":"7b08a55fd84cf8bbee204fa09e8ea402996a648c5af38b52d27231c60d9c8e4d","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"60d3271e8f6a7e952844b716a5f9f71744cb8d6fbeb9adaf35f1735ff7e44aa0","impliedFormat":1},{"version":"632e473a59bfaff109a4405851b56c61aab4a82cedd2a658b37931f98f64ba91","impliedFormat":1},{"version":"178871c23f0cac1cb358aa23f0ba3b1650ec3e962f575e82d33bce7550e55cce","impliedFormat":1},{"version":"94386e32c1da2a3dbff53bfa3aca55ef89397f09bfbb7546890031f246d65716","impliedFormat":1},{"version":"2b96e9789937d863abbb5e33861c941da0d0607fa548f965cdf4e0cf984579ce","impliedFormat":1},{"version":"ea80ad7543efdaeb5ee48a3951f5a32adaa8814fb2a8b9f8296170aa31083455","impliedFormat":1},{"version":"72aad439f7b0cf1c9b28cba809c6b818c72d09f8eeb5978f626d088c2d520f18","impliedFormat":1},{"version":"40d4add4a758635ba84308ecf486090c2f04d4d3524262c13bfb86c8979fac4e","impliedFormat":1},{"version":"72aad439f7b0cf1c9b28cba809c6b818c72d09f8eeb5978f626d088c2d520f18","impliedFormat":1},{"version":"f44c61ac2e275304f62aace3ebc52b844a154c3230f9e5b5206198496128e098","impliedFormat":1},{"version":"924f76dc7507df1c4140262ea2a2d8ef99b8c31e995edefc8271928a3e4807a6","impliedFormat":1},{"version":"3ffc5226ff4a96e2f1a1b12720f0f8c97ac958ac8dd73822bedf6f3ed3c35769","impliedFormat":1},{"version":"924f76dc7507df1c4140262ea2a2d8ef99b8c31e995edefc8271928a3e4807a6","impliedFormat":1},{"version":"9df26a86871f5e0959d47f10bff32add294bf75b8d5a4f77a19dfc41694649d2","impliedFormat":1},{"version":"bfdd4ae390e0cad6e6b23f5c78b8b04daef9b19aa6bb3d4e971f5d245c15eb9a","impliedFormat":1},{"version":"369364a0984af880b8d53e7abb35d61a4b997b15211c701f7ea84a866f97aa67","impliedFormat":1},{"version":"7143d8e984680f794ba7fb0aa815749f2900837fb142436fe9b6090130437230","impliedFormat":1},{"version":"f7b9862117ae65bea787d8baf317dcc7b749c49efeada037c42199f675d56b7b","impliedFormat":1},{"version":"78a29d3f67ea404727199efc678567919ecebbfdc3f7f7951f24e1014b722b46","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"e53b2d245026cefec043621d6648fab344fd04415b47270da9eb4e6796d2a9f4","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"f10a10d90bd1e3e12e1d7d027086a716dd6fa03d251597af77210e7a3081ac0b","impliedFormat":1},{"version":"b2bd6911e91dbb008938121d0fd7df51f00148652090bc9ccde4dc704f36f011","impliedFormat":1},{"version":"1bbdf84753428ed6f1533eabb066f9b467fade05180797e39cb32b4be4ba7d5d","impliedFormat":1},{"version":"e52d0f3e5073519a3a0a69fb0090c180f219fa04fc4053bb2bc5453a61296acd","impliedFormat":1},{"version":"24b30db28923568ff5274ec77c4c70c3e18a62e055f207633b95981ba94b0dee","impliedFormat":1},{"version":"e285a018fca2bcd32f25e2e048076b135086b3bd0d6215b1f72716129dce44ad","impliedFormat":1},{"version":"d9901d27accf8b30a3db21c9537e516427f55abd13ca53283c8237711bd37c16","impliedFormat":1},{"version":"46ded89297bd3856f536a6a990d64831ea69976626669e9371fe12e47a263ceb","impliedFormat":1},{"version":"823f27e48b1e7ff551b90d15351912470ab3cd0fa133bc2e1ddc22bea6c07d23","impliedFormat":1},{"version":"189abcb612878978d45a513656690710591b93860bc9cc2d2bf58c5f2ea9b3ae","impliedFormat":1},{"version":"e6251b50929025156877155e58eff37840da58c85d094e3f128b4f07e03aa66d","impliedFormat":1},{"version":"e6251b50929025156877155e58eff37840da58c85d094e3f128b4f07e03aa66d","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"657bfa91b3233a36081f7030fa35a16728be10e90b926a9e8ae218e9078a5e75","impliedFormat":1},{"version":"c6b1f54c34ab08126f8594801908410a93a64e0dff66df8a226a9b5460054f19","impliedFormat":1},{"version":"ca969c350e570c5fa395c4fb88ea52dfe50014890c445d2834e4f1fe96e93c2d","impliedFormat":1},{"version":"a6f374e4c41a9aaa10213ba98f7d1e520f4cc314c2f20770145124e2f207f11c","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"1481094055c14f5976d55446330cca137adf0b2a39dcae164f1d6460862e5e5b","impliedFormat":1},{"version":"914912142f2648f12b831ad10bcfacfbc02876161de095c479a1ae308067f646","impliedFormat":1},{"version":"b5f7732acfd56640a680acbd12caff991c839c3dfd5a4b48ad90bd7a730d501d","impliedFormat":1},{"version":"8b801973d33012fc9b97dcb37cfd2d5d30eed228b4d342ae3563972ba1004279","impliedFormat":1},{"version":"09c3bb9dac02114c00586e82c825655ea0c5031097667855544d436063322760","impliedFormat":1},{"version":"14e64ceb540cc27093ba1a04948aec14707da94a6ff1d9675efca976e10fea49","impliedFormat":1},{"version":"da6e2dde5747e6e71bdc00a26978fe29027a9e59afe7c375e2c040a07ef9ff25","impliedFormat":1},{"version":"5d6ddacf1e9cc6fd92ae992eb6eb00910cfe3fe95f6e29b44f0730c710b2def5","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"da20ac2b80ec650f4c36df8ebff9493625634329eb0f901a0971dd6619e0978c","impliedFormat":1},{"version":"ef51ac3ae8d6ddc8ee29937a039cbb4a9bfe6ab34267d4c9d998645e73f91237","impliedFormat":1},{"version":"cc45a177fe3864f8a5579ddb987cb5db0ee47c4d39335832635c241b5f98337e","impliedFormat":1},{"version":"3aaf74018283ef4c49f52bcab37f09cd6ec57fff27503090bc4bb75194fd68a8","impliedFormat":1},{"version":"69578d34fa63a8314823b04f6f57a60671755666055a9990b070f5403f21d417","impliedFormat":1},{"version":"c9aa17bf9f1d631f01764ad9087de52f8c7e263313d79ac023f7cd15967b85cb","impliedFormat":1},{"version":"78d05f11e878fe195255ac49d0c2414a1c7fa786b24e8d35c0659d5650d37441","impliedFormat":1},{"version":"b93a1522b0ae997d2b4dc0e058c1d34f029b34370ee110b49654deeef5829a41","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"ae2104bdc52ab3722b5c0cfa26aa65b077e09d7288695f9e0ee9ffde08721b3d","impliedFormat":1},{"version":"a4038d37487d8535f99ba99adc4a01b08f038515dd939e57bd80a3743c0e5662","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"483095dc7d04bc24cc55e72a807fa8d786a52981068c6f484947f63956b0fa92","impliedFormat":1},{"version":"4539884fadd3b91977560c64de4e5a2f894a656a9288882e1307ba11c47db82e","impliedFormat":1},{"version":"430016e60c428c9c8bfa340826ff7ed5988e522348838700f3c529dc48376c10","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"2e1b0586468b145f432257bfc0dc8d40a82b04ebd00c5f92efdde426d14d122b","impliedFormat":1},{"version":"976d79fce50c222b3aa23d34e4165e1c8424060c3744a4a5b5834bbc644e64a6","impliedFormat":1},{"version":"d61d7221ed4b74db0568ffae7765f6c2a48afc64a076dd627e98dfecd1ad9897","impliedFormat":1},{"version":"89ac12f3bd077e0d31abc0142b41a3dbbdb7ae510c6976f0a957a1f3ca8c46c9","impliedFormat":1},{"version":"694d279f9a6012c39bba6411e08b27706e0d31ea6049c69ff59d39a50de331cc","impliedFormat":1},{"version":"e27f95d214610d9d7831fdeccba54fbe463ae7e89bd1783d828668072c2d2c92","impliedFormat":1},{"version":"ed48328b38a82b98abf873153e939c9baed42cbd5d5289830dd832c552db5024","impliedFormat":1},{"version":"6ca43ca6b5f1794be3eee4993c66f15083c3b47ee45615163ee49f450e4b464a","impliedFormat":1},{"version":"8d8381e00cd14cf97b708210657e10683f7d53a4eddcfc3f022be2c9bdf591dd","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"a37d882a1490198571664d4d06e584d226f8c62445b25696f3f9efff776b2a0b","impliedFormat":1},{"version":"ec85bf4283c2ec8108b0b6161f155aeedfc770f42dca27bb6fca2cfb0abf1a8a","impliedFormat":1},{"version":"ec2ba248e2ad73cfd1989cb7f53ff1df5612f63b628e03a472308c1bab10c0f9","impliedFormat":1},{"version":"ea763067ac7adab4741f87de9fec3fc154ac1f3578b7e3bc0c64b42c6f6c912e","impliedFormat":1},{"version":"a6add93dcdbb7c0b119b363ba421fb530d7fd68814be4a8314ec9aee486478f9","impliedFormat":1},{"version":"d54fa16b15959ed42cd81ad92a09109fadbb94f748823e2f6b4ad2fbbee6e01f","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"2e2ffb8593c9db471bac9f97c0b1f1c7ef524946a462936e5e68858ac3e71566","impliedFormat":1},{"version":"d4c081ae5c343c754ac0dd7212f6308d07f55ab398cee4586ee0a76480517ae5","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"a4f2c605bbc73124b1bb76faa66be28937ccfb7f5b77c45cd8022071bd53696c","impliedFormat":1},{"version":"be4c58de8fd3ddd0e84076c26416ce5ffcf193a1238704692e495bc32e0a6ec5","impliedFormat":1},{"version":"af9491fcc19d5157b074871bdceafc18dd61972020fb8778c7d3cd789cd8186a","impliedFormat":1},{"version":"64da3dee7d98bdc4b99b24de094a08ffb2dda8aa14270cd51fc936dc8af1cdb2","impliedFormat":1},{"version":"a4038d37487d8535f99ba99adc4a01b08f038515dd939e57bd80a3743c0e5662","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"152532087c2a91adb4527e96ccd7b3640f1b08c92301fa2f41ed6a53130bda67","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"549f38b7fc2753d95809f16c29e8f86cf6f9d99cb17d8eb53f0132bc92192a2b","impliedFormat":1},{"version":"6e39d03aa07f268eed05dd88e1bd493cb10429c1d2809e1aaa61fbcd33978196","impliedFormat":1},{"version":"aa7384441d37522532179359964184e5c8cf649db32a419542e7b5605208b45c","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"36d27819ece3bf0eefe61ecda9e3aa2e86b5949c89dba79f17dd78a2c4587a61","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"18a20ae79049147b460771dfd6b63b3b477772d763c26b367efa499c98e9fb5f","impliedFormat":1},{"version":"4c91908ebcc1b1c91f5c9cd7e9ffff83fc443e6926013b0b0082a6c2778b729e","impliedFormat":1},{"version":"ee51a4032beba0b38ff75838b386627a38c53008b8ca350bb42f192d0fb3cf58","impliedFormat":1},{"version":"b14b8756b166914ab1cb68c44bb579566833449d5e9d68655726f6ffc6d5e457","impliedFormat":1},{"version":"a09ae8631b5e442bbcdb93e3b60d6f71a54d192452af841616e2b49c5a03fb26","impliedFormat":1},{"version":"7a254103740333c7fb870f95ab9a26fb028cb298478f43e4750b8eddefafa11f","impliedFormat":1},{"version":"d54b449b0eff66bc26e09593df44512725b9e9fce4d86ea436bed9e7af721ff1","impliedFormat":1},{"version":"91991180db9a4d848bd9813c38a56d819a41376a039a53f0e7461cc3d1a83532","impliedFormat":1},{"version":"4e5f8c9d9655d5cedd160d50dc0d04f78fafb2c21db87e5b0c87105050445d91","impliedFormat":1},{"version":"637ffc16aeaadb1e822bffc463fcc2ca39691dea13f40829c1750747974c43d4","impliedFormat":1},{"version":"7955f3e66404ff9a4ac41f40b09457fe1c0e135bde49e4d77c3ea838956041bf","impliedFormat":1},{"version":"f6d23ab8669e32c22f28bdbdf0c673ba783df651cafcbdcc2ead0ff37ba9b2b5","impliedFormat":1},{"version":"c90ef12b8d68de871f4f0044336237f1393e93059d70e685a72846e6f0ebbbff","impliedFormat":1},{"version":"ecefe0dd407a894413d721b9bc8a68c01462382c4a6c075b9d4ca15d99613341","impliedFormat":1},{"version":"9ec3ba749a7d20528af88160c4f988ad061d826a6dd6d2f196e39628e488ccd8","impliedFormat":1},{"version":"71ce93d8e614b04d49be0251fb1d5102bb248777f64c08078ace07449700e207","impliedFormat":1},{"version":"9560571cf48c84114027d57b34d769cd4e9a6cfaac7919bfbdcd6ad0801bc73c","impliedFormat":1},{"version":"4818c918c84e9d304e6e23fdd9bea0e580f5f447f3c93d82a100184b018e50f5","impliedFormat":1},{"version":"6e39d03aa07f268eed05dd88e1bd493cb10429c1d2809e1aaa61fbcd33978196","impliedFormat":1},{"version":"eab3b41a54d5bc0e17a61b7b09639dc0d8640440e3b43715a3621d7fa721ae85","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"da31c5275a923bb601a84bd648fd24cc9009860fd5901351f32e686e69bfd432","impliedFormat":1},{"version":"36d27819ece3bf0eefe61ecda9e3aa2e86b5949c89dba79f17dd78a2c4587a61","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"ce8eb80dad72ac672d0021c9a3e8ab202b4d8bccb08fa19ca06a6852efedd711","impliedFormat":1},{"version":"a336b950cd64c3c3dd2503384bf2915a5ea03d694672bfedabd71fafdae34ebe","impliedFormat":1},{"version":"d12e9c3d5e2686b5c82f274fb06227748fc71b3a6f58f7b3a6f88f4b8f6921fb","impliedFormat":1},{"version":"5f9a490be2c894ac65814a1a9e465b99882490ed3bce88c895362dc848f74a8d","impliedFormat":1},{"version":"2d5935948312241d3195b5e24df67775c6736dec1e1373efb1b6f04447106867","impliedFormat":1},{"version":"686ccf874ccbf999a155208a7ec8358a718d211f779980c2fe7cca176025d769","impliedFormat":1},{"version":"48bf56f3c8b3d0b27f94587996400c129773ab9c4810354d89850b0bee92b3d7","impliedFormat":1},{"version":"e6e9bdd2f65408a0b52d8e8ca9ddb7827c5f3496561788c974e4f2fb485427eb","impliedFormat":1},{"version":"193772121770797ee600739d86de128cd7244e3e3e101684473eb49590dbfce1","impliedFormat":1},{"version":"7a6208fa971deb77dbd7c59d56f7eb5b2516d76a3372a55917b75fc931c44483","impliedFormat":1},{"version":"b9aa4ed5dc603ad443dac26b9c27b0680b1cf4614f321b8d3663e26c1b7ef552","impliedFormat":1},{"version":"8613d707dc7f47e2d344236136010f32440bebfdf8d750baccfb9fad895769ee","impliedFormat":1},{"version":"59ebb6007bce20a540e273422e64b83c2d6cddfd263837ddcbadbbb07aa28fcc","impliedFormat":1},{"version":"23d8df00c021a96d2a612475396e9b7995e0b43cd408e519a5fb7e09374b9359","impliedFormat":1},{"version":"9a3c859c8d0789fd17d7c2a9cd0b4d32d2554ce8bb14490a3c43aba879d17ffb","impliedFormat":1},{"version":"431dc894a90414a26143bbf4ca49e75b15be5ee2faa8ba6fcc9815e0ce38dd51","impliedFormat":1},{"version":"5d5af5ceb55b5ec182463fe0ffb28c5c0c757417cbed081f4afd258c53a816c5","impliedFormat":1},{"version":"f43eee09ead80ae4dcfc55ba395fe3988d8eb490770080d0c8f1c55b1bd1ef67","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"4c9784ca0ab39916b498c54db858ea27c929777f161a2450f8712a27cec1b017","impliedFormat":1},{"version":"9c92db9255eab1e3d218bdeca593b99355bbf41fa2a73a9c508ad232a76cda96","impliedFormat":1},{"version":"bf2cc5b962f3823a8af297abe2e849227dbfb3a39a7f7301c2be1c0a2ecb8d32","impliedFormat":1},{"version":"eaed6473e830677fd1b883d81c51110fcb5e8c87a3da7a0f326e9d01bf1812ff","impliedFormat":1},{"version":"3ac0952821b7a43a494a093b77190a3945c12f6b34b19f2392f20c644ac8d234","impliedFormat":1},{"version":"ed5877de964660653409f2561c5d0a1440777b2ef49df2d145332c31d56b4144","impliedFormat":1},{"version":"c05da4dd89702a3cc3247b839824bdf00a3b6d4f76577fcb85911f14c17deae5","impliedFormat":1},{"version":"f91967f4b1ff12d26ad02b1589535ebe8f0d53ec318c57c34029ee68470ad4a3","impliedFormat":1},{"version":"f6ac182bf5439ec39b1d9e32a73d23e10a03fe7ec48c8c9ace781b464ecc57c3","impliedFormat":1},{"version":"eec377e6bfd366a64f9641e80ff1e7ab5fa58963364a9d6a76a11365dccd87d3","impliedFormat":1},{"version":"687b26db97685fcadeb8e575b6bc252ea621fef8217acd2bb788ce781a4b05b3","impliedFormat":1},{"version":"e4a88ca598bf561ec253c0701eea34a9487766c69a8d8e1b80cf67e60dcc10d7","impliedFormat":1},{"version":"281cf6513fcf7b7d88f2d69e433ebbd9248d1e1f7571715dd54ca15676be482e","impliedFormat":1},{"version":"dc9f827f956827ec240cec3573e7215dc08ed812c907363c6653a874b0f5cabb","impliedFormat":1},{"version":"baa40541bd9b31a6f6b311d662252e46bad8927d1233d67e105b291d62ace6e6","impliedFormat":1},{"version":"d3fa2e4b6160be0ab7f1bc4501bf0c969faa59c6b0f765dc8ca1000ca8172b18","impliedFormat":1},{"version":"cf24c5c94e5e14349df49a69fb963bee9cd2df39f29ddd1d4d153d7a22dfb23f","impliedFormat":1},{"version":"18a20ae79049147b460771dfd6b63b3b477772d763c26b367efa499c98e9fb5f","impliedFormat":1},{"version":"c5ad2bd5f2243c6fade8a71a752b4333b0ba85ae3ea97d5323f7d938b743cb26","impliedFormat":1},{"version":"cf1e804f283ae1ca710f90dba66404c397b7b39682dbdfa436a6b8cc0b52b0ab","impliedFormat":1},{"version":"25fd641b32d4f7d6811cec4b00c0c9a74cb8822ec216f3b74bae205a32b1de08","impliedFormat":1},{"version":"658f07f1b7c327ecc8b18ed95ada19a90f9fc3f0282d536ca9d6cd2d597631f4","impliedFormat":1},{"version":"35c8e20c61bffc19a0391f42db2fe8f7bb77caa414bd2145a8891826bfdb9667","impliedFormat":1},{"version":"658f07f1b7c327ecc8b18ed95ada19a90f9fc3f0282d536ca9d6cd2d597631f4","impliedFormat":1},{"version":"b3279a079db8ea0c8b76f7f3098f4b10266c3bb24fa21e5838fe6008e3d40043","impliedFormat":1},{"version":"803e5b05c612513cf773d7826c8556eb30ff4a92ba33e9c9dde5ab4cfc342cf9","impliedFormat":1},{"version":"8aec152ae554311c39f87fc5ec3c1f4c5d5d44e1145704782a4fdd6b16c2f1d7","impliedFormat":1},{"version":"9b4a1b563bc6d3d02a4a9d3e72bf699d486a6b117fdcf29199d49d3650abe122","impliedFormat":1},{"version":"803e87c5c27720886ff9f591a47e3281b02bf737f6c67964d72a4d8e7b905a21","impliedFormat":1},{"version":"ce762eb7d3137473f6b50c2cd5e5f44be81334550d9eb624dadb553342e9c6ed","impliedFormat":1},{"version":"3a4d63e0d514e2b34487f84356984bd4720a2f496e0b77231825a14086fb05c1","impliedFormat":1},{"version":"22856706f994dec08d66fcbf303a763f351bc07394fb9e1375f0f36847f6d7a5","impliedFormat":1},{"version":"1f2b07381e5e78133e999e7711b84a5d65b1ab50413f99a17ffccfc95b3f5847","impliedFormat":1},{"version":"39aa109cb3f83642b99d9f47bf18824f74eaaa04f2664395b0875a03d4fc429a","impliedFormat":1},{"version":"15ca7cf99d213ac6a059a5f81ff17dd2c0d4e31260821719ef7e78ea6163f518","impliedFormat":1},{"version":"ee130bd48bc1fb67a0be58ab5708906f8dc836a431b0e3f48732a82ad546792e","impliedFormat":1},{"version":"9d32f274f0b2388e27a83b6b88b33616a4b73b4d045c00d814e942c07a5c9a57","impliedFormat":1},{"version":"06a6defbd61ec1f028c44c647c7b8a5424d652b3330ff4f6e28925507e8fde35","impliedFormat":1},{"version":"9d32f274f0b2388e27a83b6b88b33616a4b73b4d045c00d814e942c07a5c9a57","impliedFormat":1},{"version":"15ca7cf99d213ac6a059a5f81ff17dd2c0d4e31260821719ef7e78ea6163f518","impliedFormat":1},{"version":"9df4d5273810ea069628b1efd0ea6ca9932af9694bfbc8dcea17c8253f1790c2","impliedFormat":1},{"version":"9b3ca716ad96d961aa8f2bab5fbd6752637af2da898f54c8d4021ef8ab2607d2","impliedFormat":1},{"version":"60d53d724e5854f545fd4753881466043628eb886159a73568878f18b3020afe","impliedFormat":1},{"version":"c53d0b758384bd45cd3a051a5227805b57eae8f2142e906d65ae97c8868fd45f","impliedFormat":1},{"version":"a844bbf1cb0bb844743b2d78eee9bdc78df80a98989deab32ff8cd3228b41289","impliedFormat":1},{"version":"b641f9357511425b12ad981f9ba66d964fc114b78a5761ead8595599f036a22f","impliedFormat":1},{"version":"3537c3f024e3bed94fedcce3444fca3c1bce744942912a5a4857f7050ab25429","impliedFormat":1},{"version":"96a5c70389556c62902487f56bb34259ef57439a4cba6c9bdbbbb55225b32e63","impliedFormat":1},{"version":"54895ba2b529f7c369600228dbb88c842c311d1fb7de4ccbc43123b357c26a90","impliedFormat":1},{"version":"9d0050ae8481d6e0731ed80b55f6b475ae3a1cffbc61140e92816a0933dba206","impliedFormat":1},{"version":"68867d1d1560d31165f817de3fceb4b2bedbd41e39acdf7ae9af171cdc056c47","impliedFormat":1},{"version":"1c193e68e159296fded0267475b7172231c94e66b3d2f6f4eb42ffde67111cc5","impliedFormat":1},{"version":"f025c51bcc3c7dacbedb4b9a398815f4d5c6f4c645db40880cee4ac6f89588de","impliedFormat":1},{"version":"b94704c662a31e0d061abb006d38f6211ade97422f0ae45d751ef33d46ce3042","impliedFormat":1},{"version":"c3e2f2b328bd55ae9a401673bd33f86d25a7d53a4f5e1fad216f5071c86c0b79","impliedFormat":1},{"version":"5f6e56ac166b7a5bde756afd2e573af1e38fdd5f10ddb72e46bc44f3c0a42369","impliedFormat":1},{"version":"9b65fd7edfcf3c4c6538d735d269647edc14856dc062e9dde80412c45ff2cf29","impliedFormat":1},{"version":"fbb26af430ebc8743161f6026a0722a4cee3df8c08bdc2610a1d037f733fa823","impliedFormat":1},{"version":"65de396834768bf2b3548447b84b774310f83f33d00f9fb951c1b338dd9b5395","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"75b022f6a48640ca4e048da35132eef2cb9445680c7e1080021ccc15f4d2bf59","impliedFormat":1},{"version":"ea7c9f9c4b1cd2573d49dd628d446fa7611052e00ea1a3aa385a83a7b07c7fbb","impliedFormat":1},{"version":"a74eec58a6011f6ba3d6bbe4eacea0935f7fce9ad34f8c8bd8ed8872ae68f826","impliedFormat":1},{"version":"6bd326162475f1661612f9bb68aa7833e548c7a726940f042e354086cd9b7c2d","impliedFormat":1},{"version":"4b3d55b3d962f8773ea297be1b7f04093a5e5f0ea71cb8b28cef89d3d66f39b0","impliedFormat":1},{"version":"39d7517763d726ce19f25aacf1ccb48ec4f1339978c529abdf88c863418b9316","impliedFormat":1},{"version":"4ce8ae09e963394e7ffe3a5189007f00a54e2b18295585bb0dae31c7d55c1b3f","impliedFormat":1},{"version":"b29b65017a631dff06b789071cdf7a69f67be35238b79f05e5f33523e178feaf","impliedFormat":1},{"version":"58cb40faa82010f10f754e9839e009766e4914586bdb7a4cceff83765fa5e46c","impliedFormat":1},{"version":"efa190d15d9b3f8a75496c9f7c95905fca255a7ce554f4f0b91ba917b61c3b7e","impliedFormat":1},{"version":"303fd31bbed55c8cdf2d3d9851668f4e67746f0a79861a3b4d947a6c1c9e35c5","impliedFormat":1},{"version":"0fe6e8d738df018108bd3ca0e208dfa771d4e34641242b45423eca7d7ade80a7","impliedFormat":1},{"version":"8210e3bdbeeb9f747efdf7dad7c0ed6db9d13cd0acd9a31aa9db59ddbbac5a15","impliedFormat":1},{"version":"d6791734d0fce30014c94846a05cb43560bce15cfdc42827a4d42c0c5dafa416","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"8c4f5b888d7d2fc1283b7ce16164817499c58180177989d4b2bd0c3ebd0197f7","impliedFormat":1},{"version":"58c97efc183a6465be046e3c59ff1164b9930c25f080f5462d4b103760757d97","impliedFormat":1},{"version":"ea7c9f9c4b1cd2573d49dd628d446fa7611052e00ea1a3aa385a83a7b07c7fbb","impliedFormat":1},{"version":"3108920603f7f0bbf0cebce04bcaf90595131c9170adb84dc797e3948f7b6d06","impliedFormat":1},{"version":"8aded022b77ae3c07af72765bca9421f2d990814e0f4bfca0aa97395aa4c9010","impliedFormat":1},{"version":"f817987f543a452afa3035a00aa92800dbd7ff3246fcbe4cecb29bc18552b081","impliedFormat":1},{"version":"6ab1e8b5d0a0f4123b82158ea498222a5eacbffa1354abe8770030ba722c13b7","impliedFormat":1},{"version":"3cda89b540ed1ea9a3d1e302a489a4157a98b62b71c7abb34f8f15c13da9717a","impliedFormat":1},{"version":"a1ebece06e1ac47fb3a1b07997e57aa2e6a8f5ece26ea3c4a4fcb591e05d1e05","impliedFormat":1},{"version":"8aded022b77ae3c07af72765bca9421f2d990814e0f4bfca0aa97395aa4c9010","impliedFormat":1},{"version":"fb3b5ff3f5fe7767c07b755f2c22ce73ba46d98e6bc4a4603fde8888eed14e19","impliedFormat":1},{"version":"41c53632da296cf700f8553a48522e993949ea8499ceac4a483d1813beed3017","impliedFormat":1},{"version":"03b97deb8a168b27af94dca96eba747e19faf077445102d52c618210829cb85f","impliedFormat":1},{"version":"6a3589af6b9ec75cd87d9516ccfb9b06ab6be6f938790aeb4b1cd4dbaef92c45","impliedFormat":1},{"version":"722a667fe3b290be746d3ea6db20965ec669614e1f6f2558da3d922f4559d9c4","impliedFormat":1},{"version":"0f1c68ddd4573b2e135748377c3705a96d6a6c123910b00d0c7e8dc2edcd7f6b","impliedFormat":1},{"version":"a63781a8662205b9b6d2c7c5f3bad1747a28e2327804477463ebb15e506508e1","impliedFormat":1},{"version":"0f1c68ddd4573b2e135748377c3705a96d6a6c123910b00d0c7e8dc2edcd7f6b","impliedFormat":1},{"version":"80d8f42128925d6f1c82268a3f0119f64fd522eec706c5925b389325fb5256de","impliedFormat":1},{"version":"b4c189c9be8cf4a7cce177fc49678e29d170e67279195207f36a4f4d184d60f2","impliedFormat":1},{"version":"d16a18dfc505a7174b98f598d1b02b0bf518c8a9c0f5131d2bd62cfcaaa50051","impliedFormat":1},{"version":"b4c189c9be8cf4a7cce177fc49678e29d170e67279195207f36a4f4d184d60f2","impliedFormat":1},{"version":"d3ceb0f254de2c13ffe0059a9a01ab295ccf80941c5429600ffdbaaec57410a7","impliedFormat":1},{"version":"8e172ba46195a56e4252721b0b2b780bf8dc9e06759d15bc6c9ad4b5bb23401d","impliedFormat":1},{"version":"41c53632da296cf700f8553a48522e993949ea8499ceac4a483d1813beed3017","impliedFormat":1},{"version":"0fe5f22bc0361f3e8eacf2af64b00d11cfa4ed0eacbf2f4a67e5805afd2599bc","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"226dc98afab126f5b99f016ec709f74c3bcc5c0275958613033e527a621ad062","impliedFormat":1},{"version":"ec7197e94ffb2c4506d476df56c2e33ff52d4455373ecb95e472bb4cedb87a65","impliedFormat":1},{"version":"343865d96df4ab228ff8c1cc83869b54d55fa764155bea7db784c976704e93ec","impliedFormat":1},{"version":"f3f8a9b59a169e0456a69f5c188fb57982af2d79ec052bf3115c43600f5b09e4","impliedFormat":1},{"version":"e2898fa86354ef00ff2c0967a79b4f809477ec4471528aa96e192251b9f81d0c","impliedFormat":1},{"version":"15ddffc9b89470a955c0db3a04aec1f844d3f67e430b244236171877bdb40e50","impliedFormat":1},{"version":"7ca1ed0b7bd39d6912d810562413fb0dad45300d189521c3ca9641a5912119a5","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"74766ac445b27ae31cc47f8338fd0d316a103dd4d9eb766d54b468cb9aacbf0e","impliedFormat":1},{"version":"65873070c21b3ce2ccdf220fe9790d8a053035a25c189f686454353d00d660f9","impliedFormat":1},{"version":"d767c3cc8b1e117a3416dda1d088c35b046b82a8a7df524a177814b315bde2e3","impliedFormat":1},{"version":"bf834cd64464f9217cb642a48c2f5f5f1cd509e13088adac6773715fb8536212","impliedFormat":1},{"version":"40258ea27675f7891614c8bd2b3e4ee69416731718f35ec28c0b1a68f6d86cd6","impliedFormat":1},{"version":"bf834cd64464f9217cb642a48c2f5f5f1cd509e13088adac6773715fb8536212","impliedFormat":1},{"version":"c61aa5b694977909ef7e4a3fdad86b3c8cd413c8d8e05b74a2def595165ba7ce","impliedFormat":1},{"version":"bfef3048352341739d810997dcd32f78527c3c426fac1bbb2b8c14293e1fa505","impliedFormat":1},{"version":"1dd31462ed165900a141c2e159157be0e8701ce2a2ed0977636f1d021894887d","impliedFormat":1},{"version":"872321f2e59009fad1f2efde489b20508a3631e16a86860740044e9c83d4b149","impliedFormat":1},{"version":"fa381c11f336210a8c10d442c270c35165dcf6e76492618ee468dba325a3fc98","impliedFormat":1},{"version":"857857dbb4d949686de80a138aeab8e669d23397100dc1e645190ff8be5787de","impliedFormat":1},{"version":"d6a9fe9c13a14a8d930bb90f3461dc50945fa7152e1a20a1f5d740d32f50b313","impliedFormat":1},{"version":"4162a1f26148c75d9c007dd106bd81f1da7975256f99c64f5e1d860601307dad","impliedFormat":1},{"version":"63f1d9ad68e55d988c46dab1cbc2564957fcbd01f6385958a6b6f327a67d5ff4","impliedFormat":1},{"version":"8df3b96fbafb9324e46b2731bb267e274e516951fbf6c26165a894cae6fd0142","impliedFormat":1},{"version":"822e61c3598579070f6da4275624f34db9eb4af4c27a2f152a467b4a54f4302f","impliedFormat":1},{"version":"a8f83bf864a5dea43d30c9035d74069b1820f0c49824960764cf21d6bfbb8e66","impliedFormat":1},{"version":"f9449f2b807f14c9ff9db943e322385875cca5faa26775f64a137e4d1a21b158","impliedFormat":1},{"version":"8855c7125e06a2001f726b4f2f9905e916d122377f7d938936fb49606ccb55c5","impliedFormat":1},{"version":"8855c7125e06a2001f726b4f2f9905e916d122377f7d938936fb49606ccb55c5","impliedFormat":1},{"version":"d24f0b133a979dc915411e1c76d2dada47e3624b42d5838e9d6b9eef1f067cc7","impliedFormat":1},{"version":"755611714dbab5b9b351b51e7875195f83bb26169ae6b31486dcb1e6654ed14c","impliedFormat":1},{"version":"a82213450f0f56aab5e498eaae787cf0071c5296ea4847e523cf7754a6239c99","impliedFormat":1},{"version":"f2882c5afda246fa0c63489d1c1dff62bf4ddf66c065b4285935d03edaec3e71","impliedFormat":1},{"version":"d38c1b0fd8bc7e301fd467a2afd6d32b2457813c48c16afabc06d2ca5b6bda41","impliedFormat":1},{"version":"d38c1b0fd8bc7e301fd467a2afd6d32b2457813c48c16afabc06d2ca5b6bda41","impliedFormat":1},{"version":"4ed8f12983c82690e8fecd9b24f143d4a7c86d3156be7b2bff73e0761f820c8c","impliedFormat":1},{"version":"1d920699becb8e60a0cbbc916d8559a3579b204dd21655dd242c98fd8ae986ea","impliedFormat":1},{"version":"c278288183ec3690f63e50eb8b550ef0aa5a7f526337df62474f47efea57382b","impliedFormat":1},{"version":"3c0486004f75de2873a34714069f34d6af431b9b335fa7d003be61743ecb1d0a","impliedFormat":1},{"version":"99300e785760d84c7e16773ee29ac660ed92b73120545120c31b72166099a0e4","impliedFormat":1},{"version":"8056212dad7fd2da940c54aeb7dfbf51f1eb3f0d4fe1e7e057daa16f73c3e840","impliedFormat":1},{"version":"e58efb03ad4182311950d2ee203807913e2ee298b50e5e595729c181f4c07ce3","impliedFormat":1},{"version":"67b16e7fa0ef44b102cc4c10718a97687dabfa1a4c0ba5afe861d6d307400e00","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"f29c608ba395980d345144c0052c6513615c0ab0528b67d74cacbfac2639f1d4","impliedFormat":1},{"version":"e094afe0a81b08444016e3532fbf8fae9f406cdb9da8dbe8199ba936e859ced7","impliedFormat":1},{"version":"e4bcab0b250b3beb978b4a09539a9dfe866626a78b6df03f21ae6be485bc06e2","impliedFormat":1},{"version":"a89246c1a4c0966359bbbf1892f4437ff9159b781482630c011bb2f29c69638f","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"0a87a56e75de872e21997cec18ecda36abb5cac0d18690659b588e271099b589","impliedFormat":1},{"version":"98ca77869347d75cd0bb3d657b6dcd082798ef2419f1ab629ccf8c900f82d371","impliedFormat":1},{"version":"73acfe8f7f57f1976d448d9569b345f907a6cf1027a08028fe5b8bb905ef8718","impliedFormat":1},{"version":"ed8a781d8b568d8a425869029379d8abc967c7f74d6fe78c53600d6a5da73413","impliedFormat":1},{"version":"90ead73acfd0f21314e8cbef2b99658d88cc82124cfc20f565d0bdda35e3310a","impliedFormat":1},{"version":"8ecfec0e00878d6d26a496cf5afc715b72c3da465494081851da85269b0aef8e","impliedFormat":1},{"version":"4c78fccd1c5cd8eebde42cc078e7332f3d9b4eb1a542d9a5ec66899dfd71b93e","impliedFormat":1},{"version":"4c78fccd1c5cd8eebde42cc078e7332f3d9b4eb1a542d9a5ec66899dfd71b93e","impliedFormat":1},{"version":"e54b165a2a5a5fbcf4bcd09176e4388b514ca70a20635841937f1cc36e37fbef","impliedFormat":1},{"version":"6eb0dcefcf4cc9088174209028db705572e7fb7e38f3f93275bf6778afa2cd19","impliedFormat":1},{"version":"fa572fa0d1b1b1a7d356d5942b1d57f342880a68d1bf1ab5d00490221c471c18","impliedFormat":1},{"version":"17694dd0223346fa0a17e87e9ce00335569166368357b9963571aa623c5e3c27","impliedFormat":1},{"version":"207d46e6e557df62460be9021502fc3af96c927cef0cc5add32cb6f2d60b2e23","impliedFormat":1},{"version":"cf0cf6556adc9178a6251d9b12837e5d514b805cebe8de6d7a16e1e4248ec1ef","impliedFormat":1},{"version":"3d3d28a294ca0d5caea84d58eec474891dd1df7015f8fb2ee4dabf96d938333c","impliedFormat":1},{"version":"0b5b95f3b76e6cc9b716e08274d0f7486bee9d99e42dd6a99c55e4cb4ff5569e","impliedFormat":1},{"version":"94fb6c136acee366e3a4893df5ddbecadde49738de3c4d61a2923c6ada93e917","impliedFormat":1},{"version":"95669998e1e807d41471cebed41ede155911da4b63511345571f5b7e13cbef9c","impliedFormat":1},{"version":"48cca9861e6f91bde2435e5336b18bdc9ed3e83a6e7ea4cf6561e7f2fee4bad6","impliedFormat":1},{"version":"b6b8be8a70f487d6a2fd80b17c4b524b632f25c6c19e76e45a19ad1130209d64","impliedFormat":1},{"version":"76d7fadbb4ff94093be6dd97ea81a0b330a3a41fc840c84a2a127b32311200e6","impliedFormat":1},{"version":"856a8b0060b0e835bccba7909190776f14d8871b8170b186d507d3e12688086d","impliedFormat":1},{"version":"e39aaeef0aea93bdda6f00d27ca9ebda885f233ecc52b40e32db459916f24183","impliedFormat":1},{"version":"14f3c0b1b5e6adac892607ecefc1d053c50bc8a5f14d05f24e89e87073d2f7e3","impliedFormat":1},{"version":"f877dcc12cc620dede9c200625692cf614b06aadc026f6b59e5967cd2e30cbc4","impliedFormat":1},{"version":"5a37547f8a18bc0738e670b5043819321ae96aee8b6552266f26d8ce8f921d17","impliedFormat":1},{"version":"4d3e13a9f94ac21806a8e10983abcf8f5b8c2d62a02e7621c88815a3a77b55ae","impliedFormat":1},{"version":"938cb78a2ad0894a22e7d7ebd98cdc1719ee180235c4390283b279ea8616e2a9","impliedFormat":1},{"version":"84ba4c2edb231b1568dae0820f82aca1256a04599d398ec526615c8a066f69ec","impliedFormat":1},{"version":"cd80a8f16c92fe9f03899f19c93783dce3775ef4c8cdf927ac6313354765a4f2","impliedFormat":1},{"version":"25df98970954ccd743fe5e68c99b47d0e02720e2bf6584a6de60e805395b6bf7","impliedFormat":1},{"version":"251983cb99df8c624ca1abd6335ca5d44d0dd7cdcab3ef9c765b4acc79fae8fb","impliedFormat":1},{"version":"7c4965812974ebd1333cb09f95c4a3669e19008dfbb1e931321e08ae1f7cff09","impliedFormat":1},{"version":"31d3f4757bece74c888df52c8bdc4373e3f58deb518000051cadb5e85deb54de","impliedFormat":1},{"version":"a2be4cad298b3b474a0a71c1dd78a8bfc70b322f44704cf4329aecb873687a3a","impliedFormat":1},{"version":"ca8b04bea4ba551b47ddea18e385e76e555a9f7ff823dcae668d05e255fdc241","impliedFormat":1},{"version":"de0d160ecc8e643727bb93018015ae89510d59b7bdad4550f4318fba0a0ce2e6","impliedFormat":1},{"version":"acf3fff2afb5ceb54bd5ddb697b1d337338e3c23b93385f100a2046cfa700184","impliedFormat":1},{"version":"a2be4cad298b3b474a0a71c1dd78a8bfc70b322f44704cf4329aecb873687a3a","impliedFormat":1},{"version":"15c7f60f69f663374a7bc57afe164e70e3b6310bd1ee476ba911646b09c7852b","impliedFormat":1},{"version":"d71becf074ceaa0e91558fe51ed8640fa83a0fbf45a31e8069716edbf38de99a","impliedFormat":1},{"version":"ef681b070e9f3b9b28f1886bbe67faa12237c8d4691604a1f1cba614a10ef2e1","impliedFormat":1},{"version":"b15f5e077245fef1ecf45327fd94aa67fc4da288bfd42bf1b8a80f297afd561e","impliedFormat":1},{"version":"b7091d79a6e7be7bb10ca9477b6c71db4cf7b44f155912266ecfba92c1a126c1","impliedFormat":1},{"version":"e585a113e0abcaf3022f5cf1318e17f299f0935d7b389a23dcad9074c3922946","impliedFormat":1},{"version":"ae545310dfa53a7b33f574e621b14f423373dea930218d2ad290b4da0c5e8e50","impliedFormat":1},{"version":"ae545310dfa53a7b33f574e621b14f423373dea930218d2ad290b4da0c5e8e50","impliedFormat":1},{"version":"4428e4d440e1914859e8aee558f90b4829c6a45b717078490dfc2d297dcef46c","impliedFormat":1},{"version":"ad205fc7116808509e19ee71277d8da74157751d7388f0134d91c009b987f69f","impliedFormat":1},{"version":"4428e4d440e1914859e8aee558f90b4829c6a45b717078490dfc2d297dcef46c","impliedFormat":1},{"version":"8900bf61f4ce9517567cc6c9e41638a5bd0c4a0e9cc094190bc07644bbeedf24","impliedFormat":1},{"version":"cf5414a97c345c8f3294e0513a7613f5a263e1b56b3a61b810ba8279716fd38c","impliedFormat":1},{"version":"7778bc213be81351a01867789728c7780467c84e3ec94cfcef53a4e2dccf1b57","impliedFormat":1},{"version":"41a934d2efbb6cb08b205a76206fb015ebda692db4d78382ec5bec9689d6f4ac","impliedFormat":1},{"version":"67881ba5e3ec9746193075b020876fa94a1437fd78b2e0ad6351d11e1062941a","impliedFormat":1},{"version":"953cbf62815703fa9970c9cfec3c8d033da04a90c2409af6070dcc6858cf6b98","impliedFormat":1},{"version":"68065ce3af3ef8599af8338068cf336be35249eff281ee393186a0ef40db3abf","impliedFormat":1},{"version":"5339f84dfcb7b04aa1c2b4d7713d6128039381447f07abc2e48d36685e2eef44","impliedFormat":1},{"version":"fb35a61a39c933d31b5b2549d906b2c932a1486622958586f662dbd4b2fe72e6","impliedFormat":1},{"version":"24e2728268be1ad2407bab004549d2753a49b2acb0f117a04c4e28ffb3ecdd4f","impliedFormat":1},{"version":"aff159b14eba59afe98a88fe6f57881ba02895fb9763512dda9083497bdcd0e6","impliedFormat":1},{"version":"1f2bddea07543ccda708134cca0600b4d9ac9bd774ec1ede0a69935b04df1496","impliedFormat":1},{"version":"6e8997d08f6798d0a9416df24312cafd084e6184a205d9283eba95ef56f8ef8b","impliedFormat":1},{"version":"ac6968717607889d24d6e407effb48dd5af82005925b4725b1d9eb52a8a047e2","impliedFormat":1},{"version":"26080058b725ac0b480241751255b4391f722263778e84e66a62068705aafd3c","impliedFormat":1},{"version":"46afbf46c3d62eac2afead3a2011d506637bf4f2c05e1fd64bbf7e2bb2947b7c","impliedFormat":1},{"version":"84d02daa32c7a8bff4946bbc7d878ffb7114c19879f7bfceeeb39bef48e93c42","impliedFormat":1},{"version":"29723e0bc48036a127c3b8874f3abe9b695c56103f685f2b817fc532b8995e33","impliedFormat":1},{"version":"991cf4ed946cdf4c140ccaad45c61fc36a25b238a8fa95af51e93cb20c4b0503","impliedFormat":1},{"version":"81ef252ff5df76bccf7863bb355ccbb8af69f7d1064b3ef87b2b01c30fb2c1f4","impliedFormat":1},{"version":"0f17f5f14a5f53e5709404b5b59fe816eaad15a469412b73330e6f69834234e0","impliedFormat":1},{"version":"01edea77be9c2bef3a5f3fc46324c5e420e5bd72b499c5dec217c91866be5a99","impliedFormat":1},{"version":"39209d2b85d238810ef19ab3905c9498918343bc8f72a1dcae7fc0b08270d9a0","impliedFormat":1},{"version":"92a130d875262e78c581f98faa07c62f4510885df6d98213c72f3b83a1be93c1","impliedFormat":1},{"version":"6029af83ecb3d7815daf2dfb53b978087288d018e0681043b1c8c586e787af4b","impliedFormat":1},{"version":"0aa14ffe353b8bab88046e64a92efa5cd039f095759fe884d188702956e2cba2","impliedFormat":1},{"version":"68d3eee1d509f45625e39ba325a72c6ce1d2116e3d5c3a40f513472e66622e02","impliedFormat":1},{"version":"4e5f1234308de112f09920e0a0b99f35a9780b3abbc13a84445f32a490d0bb87","impliedFormat":1},{"version":"9ac0e5aea87c4a1d37b4677145e9a75bc8e13bf887bd1148a4acb21ab7398d00","impliedFormat":1},{"version":"625b802ecd18feb6a9d69ef8ef58d6c08c9c9022b8105cdeaa3fc77acaab5667","impliedFormat":1},{"version":"2ac33d7f6999e0fb363d1e483d80f087d3e7d712ff6fcc2b4f7b18b5dab92f37","impliedFormat":1},{"version":"195749d135be639001a554e4b4025b66b3c5c627d90b68266c14399bde120cec","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"e4ffb6aa67b82aca99315bd54365892ece9ed76ad37667a8dea478b7ac9a755a","impliedFormat":1},{"version":"d90644de98a3157e1f12d4c253c41397e0ae2ec6988eaa7300cdec025cb00fe0","signature":"9eaa9dacd62f2658bef1fec8bdd30f6ce058d68e06191a028970f1ec89590ea1"},{"version":"d34aa8df2d0b18fb56b1d772ff9b3c7aea7256cf0d692f969be6e1d27b74d660","impliedFormat":1},{"version":"f4db16820c99b6db923ab18af5fecb02331d785c4c2a8a88373a0cfc08256589","impliedFormat":1},{"version":"2f5747b1508ccf83fad0c251ba1e5da2f5a30b78b09ffa1cfaf633045160afed","impliedFormat":1},{"version":"90407bbaa24977b8a6a90861148ac98d8652afe69992a90d823f29e9807fe2d7","affectsGlobalScope":true,"impliedFormat":1},{"version":"b71c603a539078a5e3a039b20f2b0a0d1708967530cf97dec8850a9ca45baa2b","impliedFormat":1},{"version":"168d88e14e0d81fe170e0dadd38ae9d217476c11435ea640ddb9b7382bdb6c1f","impliedFormat":1},{"version":"104c67f0da1bdf0d94865419247e20eded83ce7f9911a1aa75fc675c077ca66e","impliedFormat":1},{"version":"cc0d0b339f31ce0ab3b7a5b714d8e578ce698f1e13d7f8c60bfb766baeb1d35c","impliedFormat":1},{"version":"6823ccc7b5b77bbf898d878dbcad18aa45e0fa96bdd0abd0de98d514845d9ed9","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e04cf0688e0d921111659c2b55851957017148fa7b977b02727477d155b3c47","impliedFormat":1},{"version":"469fb8ca997ce086fb40fc58ea6924bd8526973343c5c9db707f909b977cac5f","signature":"094d933b983c5574a83721f6736c58cfcd58caf13f610997bc773676dd199ac9"},{"version":"22ba044f5f85969a40a4221426b4a594fd1a2a1637ba30d79fca014ee49bc0ec","signature":"039458f9177c3889ea7211f64efc67d89245f2529c620d5434377672da0d77e4"},{"version":"1c5887bcdf376d0dfa02e00ce51c97ea8af6bea80c3eb87b498ac6bd84eec450","signature":"3aae3ebd3a23e43ed66895f59599aacaeb822e0bfe579a3c8c0fbbf626837f03"},{"version":"daaaa3a5f397d51d1522701044f68f708248091fd88f42e57f008f1947016125","signature":"8d6fa708e87319955ae8faa649a3e609138acba9f84300c5131c4277ef3cadae"},{"version":"0143bfcf5f88b33deeb4d7406b113cee45c23aa526e7474f310f1abafbf280b0","signature":"16b5e6a4d6aaca2f6966bd1e07a33d57e5670e40732971b21710f14f98c4c9ba"},{"version":"3fbf6b6768c99a67dc52d73289f568afbf0198d1b4fd7b21e2420e7b1ec0ef93","signature":"ab1ce0f541503f3ada8761aafeab3df510e967aa44fa9f46b5f1e39d6ef048f0"},{"version":"041c45a91e87a949c5a1ba8ce3326d16bd2878298414aa1e82789723a005a443","signature":"4dee7385e7cec9fdb3e81332bb40f60f58f192fc963f02d1c9a2fe87e6f16f08"},{"version":"aa7addf882ff8e198814167c85537da0944a63f516553f3f17d6fc5fabdc90a4","signature":"3343d01142bde8825e4461d49cedb7bc10fad39302771ce07f77404ca030a7fe"},{"version":"3522d3ae2715f78b1ed212219d3e9d0a292ba2cc9dc5761eb87f60c8c5e9a6ed","signature":"71cdafa91c307a9732b5080b13e7cd0a76856a742e556cc4334164b97dbbea36"},{"version":"20329bd349a4a67bbe1cff56bd969675eba20db02ecfe9b690dfb1d19819a12f","signature":"92819bc609bb118d7f6b235c47f19bed45872fedd9acd1daad02acc17f5292e1"},{"version":"ee0ec7d00f5c0d8a19f24e1b65a0de9ba7df174588265751c9562933c1083549","signature":"98a7b93cca530e2f2b999cce9028639baa129441cbc612c7fb827f5a6861bd7d"},{"version":"99bd15a7bfee003671ca1b8f9159266621ce75d1ea6c920982cec262b60601c7","signature":"42d9f96d012825cd0b9f23ef8437e8971492237c52e7ac96746cac97ad710d04"},{"version":"38605a8cb4292e03b24aa9e902edb9f0621cfe671b3fce7755b73b6611d5dda9","signature":"a7ad670f373dd5665c73c7dc09a8e28b976a63bd7ed08def7c1c13456e87d3dc"},{"version":"6c1b497aeb9135ac66891d783a34dec6d4df347126ebe9c3841110f0a614e0c6","impliedFormat":1},{"version":"cef73ddf0336cb343be88b61a0448483029d438dd66ca21722aeabc66223bded","impliedFormat":1},{"version":"8cb6c8db9e27d0c6dba28bf0fcd7ef7603d0b5b2b3dce6fffc86f3827a8a00e9","impliedFormat":1},{"version":"d07ef5953b1499ae335c75147c658d9c037fc649544a4c85883f10eb9e5878e8","impliedFormat":1},{"version":"34714fae00aa0544916ade4018d18a04432db2b4b49c6cd066825ac31734eb40","impliedFormat":1},{"version":"5cb3b7b2b0997e451f91ab009ff2d66e7cd5f77838dc729a2e335554fa098a12","impliedFormat":1},{"version":"bdbe3e5d1f1f3dd035c551b6f94883212ccdbe9b3610f65f49138980e0efc0d7","impliedFormat":1},{"version":"eadae8542e5f360490f84d8da987529e415e265da584dd12e3e7c07a74db2fc9","impliedFormat":1},{"version":"9a82178f67affe7ca9c8b20035956d1ad5b25d25b42b6265820232ba16ba0768","impliedFormat":1},{"version":"332680a9475bd631519399f9796c59502aa499aa6f6771734eec82fa40c6d654","impliedFormat":1},{"version":"191bee6605de2b5210f29f22df04f5b5e6bdcc1f6e21fb07091d40eeeb75fd72","impliedFormat":1},{"version":"d83f3c0362467589b3a65d3a83088c068099c665a39061bf9b477f16708fa0f9","impliedFormat":1},{"version":"180e527dbc1f5ae2bbb79d0a3db1ada49258783d7e6299559e0f2ed663b4afec","impliedFormat":1},{"version":"29994a97447d10d003957bcc0c9355c272d8cf0f97143eb1ade331676e860945","impliedFormat":1},{"version":"f4260022f7af38e533d364ea62eb7ae01b0a32050033d7f6772073e1dc908025","impliedFormat":1},{"version":"9cddf06f2bc6753a8628670a737754b5c7e93e2cfe982a300a0b43cf98a7d032","impliedFormat":1},{"version":"3f8e68bd94e82fe4362553aa03030fcf94c381716ce3599d242535b0d9953e49","impliedFormat":1},{"version":"63e628515ec7017458620e1624c594c9bd76382f606890c8eebf2532bcab3b7c","impliedFormat":1},{"version":"355d5e2ba58012bc059e347a70aa8b72d18d82f0c3491e9660adaf852648f032","impliedFormat":1},{"version":"311cc121259b3e0c3c08304fc25b525aa02ba0f9bf55b3e7c60b0dbb7422014e","impliedFormat":1},{"version":"74c269b43d39e5ece20b2cca49c14e64c05b01e46407200d7558301d0fcaabf4","impliedFormat":1},{"version":"ec09bd95866efe38cd00ebb79dfa7a26563d600fa4a30db0f7c6d68f8f6d2b06","impliedFormat":1},{"version":"482d0ac70d56aa79941be30da6df28e926a007f835eed70cf7b5f3135368d1f6","impliedFormat":1},{"version":"7dd19397d5a090c9f8cd762bae67bd0ad6f782abe422594fb71168fb578673b0","impliedFormat":1},{"version":"84cbf6204ada0ee2f80493e55e45befa079954788718efd6dcc103183104e3c0","impliedFormat":1},{"version":"ed849d616865076f44a41c87f27698f7cdf230290c44bafc71d7c2bc6919b202","impliedFormat":1},{"version":"9a0a0af04065ddfecc29d2b090659fce57f46f64c7a04a9ba63835ef2b2d0efa","impliedFormat":1},{"version":"10297d22a9209a718b9883a384db19249b206a0897e95f2b9afeed3144601cb0","impliedFormat":1},{"version":"034b8b5912823744c986986f24432bf3fa7bfa671e69316b672f3f2db5166ce4","impliedFormat":1},{"version":"34d206f6ba993e601dade2791944bdf742ab0f7a8caccc661106c87438f4f904","impliedFormat":1},{"version":"05ca49cc7ba9111f6c816ecfadb9305fffeb579840961ee8286cc89749f06ebd","impliedFormat":1},{"version":"2151db9166dfd90feaa67f0c3a07efcab39e1640f1b26abc81632d8e1bf95fcb","impliedFormat":1},{"version":"2f6fda81a8ee3f78205dcf8a69e6f5afd8ee577b8f423fed0a72c346509937a0","impliedFormat":1},{"version":"37389b1222c65e82f8e2670d586d788911f317548c3ead5c5535d2495fd08572","impliedFormat":1},{"version":"1ce0883eaaeea383c10e3274b4e5189915beaf4ec3f74fe609afd44d16bba02e","impliedFormat":1},{"version":"55a525e18db580413a78fa364a54faf071c028888d5432fcf015d229c5fafc28","impliedFormat":1},{"version":"9190c744aa6c9c2d69b5e283f5559a5543a201c518b2fe14ea4e3eb7e42f33e0","impliedFormat":1},{"version":"057ac92d0a839b0a3bd91d86b9288991ff6ac9ea72b64877464f2b12b1d117a1","impliedFormat":1},{"version":"528f3448c98e09174ca4186540000bf77f81fddcb587e0db9626ea825dead0ab","impliedFormat":1},{"version":"305af5e4e2f1f3b99d43e285d8dde8a39073dd9e40a2c5682839e7f19d66688c","impliedFormat":1},{"version":"5cc755647e5205f5acd69ad6fbb5bb41c150ea5ce229ab8ee34fc769ffecb7ac","impliedFormat":1},{"version":"aac2532c6e0b1183dd1f7d746013af0dcde78ad82879ac599f17c5563bb2f1ec","impliedFormat":1},{"version":"9ca2b093001037c017169b747ec9616b26b42ac9d8fd60aa2b0d9eb6b8c1cf95","impliedFormat":1},{"version":"5e8f7bf17b0f8382eff0d93f1ffaafc4f7fd15293b4a90edd517dca2a3ff6020","impliedFormat":1},{"version":"e7705224440c17c183317a861a0f1392a9c0746c3a06549c09e2d511a8c0c32a","impliedFormat":1},{"version":"eaccaaec4001b7c7e672f371d466e96fcbe3782cdbb7eb8ecbd132060515573f","impliedFormat":1},{"version":"f3eca6b9a668c7872bb132fafe6750c582771c40a66606745c2c01dbec8d4c5d","impliedFormat":1},{"version":"8f4e60cefea06a80cdd3a6a19fd2070910aa819fa934b58e9dc4ac726f1ff74f","impliedFormat":1},{"version":"325574dd9f2eb185c39a1ee5bb5bf656cd448954ca5485e0987d29574abdc699","impliedFormat":1},{"version":"529e1aa76e7ed000983d12eec4eca0f6d38045e9d0bc8440083d86aa4897549a","impliedFormat":1},{"version":"452234c0b8169349b658a4b5e2b271608879b3914fcc325735ed21b9cb88d58d","impliedFormat":1},{"version":"3860b1088a3d0edbe82d07b7fb13a0d04b5f23653e70714892d58e847e37bb13","impliedFormat":1},{"version":"02e6216fe46b07bbfdc787254cf085fe383ad957fe7a26aab34cb4a6e0f969b6","impliedFormat":1},{"version":"1481128ac360e7a5fc5944efc36b7634b8e5eea8870d3e5cef6647af83f98c8c","impliedFormat":1},{"version":"b5b9340f337ae17e2b59afc4c70a45b698a0227a81daf16f4bdea22757d7ba74","impliedFormat":1},{"version":"3aec561fe42dc4beb19e50b9711580620d5b0988ca0295ad0f4060a5669ee3ba","impliedFormat":1},{"version":"801e735da27b1fcb22b4d79bbe1240f211889d633026cbbd1469f941245ab419","impliedFormat":1},{"version":"5265fd19af035a75b0ea228cdd98820babea56b2b79c75517c0158ad022ae16c","impliedFormat":1},{"version":"d9fdea96fc90cc8d970044bb7bbd75766899f06a6214383bbc3b95c061bdf733","impliedFormat":1},{"version":"b3952aed8c195a401b42a8995800b5c1ea4d9d390c1a5e3521a1a3c3653f9b71","impliedFormat":1},{"version":"69c63d594f437c04b4971e171b8b3eff3d926141b87c4a898cc139b39ac86666","impliedFormat":1},{"version":"3284e33a45d6aa8324691ac5737d08695e35e99b5f69fdc9ef21b3c7e7fd8449","impliedFormat":1},{"version":"46899ea33977cc9709846fa0df32edbaa610d261a7020e487c09c5b499723634","impliedFormat":1},{"version":"df2ba32dfae996beb1face17a5bba909d7fb8f6fb80ac554e7cae50e8b00a4c7","impliedFormat":1},{"version":"b4a8d900684e3167a5251e7614843bc889a307bd79226054124286743475f2fa","impliedFormat":1},{"version":"5feab6c5b5962b943354bafc10e79ab8a495786c1141358f2a44fe2108003828","impliedFormat":1},{"version":"91dc8945750895c8ee8cc8549f81b4f0a7a6248af72e48359f8cbbc5b8bec77a","impliedFormat":1},{"version":"67e1ae275bb047700f4384559e596bcdc46a9e7ba1ef6ab275e60b8059f077ce","impliedFormat":1},{"version":"eeb24fa259f000f6b51a1fe89123f55de081eb2a0ef8d8f847afd67af49cfb68","impliedFormat":1},{"version":"9d9b52c50efdfc1d23d4aea99074009a932068cb786992776c233913bff1eeb2","impliedFormat":1},{"version":"e21bb2cfbcdd8ce7eebb72422f3660806724f2b16cd6ce126d527511abb3a379","impliedFormat":1},{"version":"c04146836a55ea071b435298335e47f569db0e4d3ae420e35c83e448f944192f","impliedFormat":1},{"version":"31f71fe23daabea143fc8bd21dae0d5908227180fcda38ad3674df70351f9761","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"d023752daf2a5c2e27a2a850aedc10a48a42fb507dceae37db91dc8294aafdec","impliedFormat":1},{"version":"15b0d7351662b9b8216b165108b690d779969d5aa0b8fea1406a05fffa623748","signature":"ff0fde5b05a4ea984546de6973bf80f184d152a069667b3d91f6328c189249dc"},{"version":"ba63131c5e91f797736444933af16ffa42f9f8c150d859ec65f568f037a416ea","impliedFormat":1},{"version":"44372b8b42e8916b0ab379da38dcf4de11227bad4221aba3e2dbe718999bdfab","impliedFormat":1},{"version":"43ebfcc5a9e9a9306ea4de9fda3abdd9e018040e246434b48ad56d93b14d4a3d","impliedFormat":1},{"version":"0e9aa853b5eb2ca09e0e3e3eb94cbd1d5fb3d682ab69817d4d11fe225953fc57","impliedFormat":1},{"version":"179683df1e78572988152d598f44297da79ac302545770710bba87563ce53e06","impliedFormat":1},{"version":"793c353144f16601da994fa4e62c09b7525836ce999c44f69c28929072ca206a","impliedFormat":1},{"version":"ff155930718467b27e379e4a195e4607ce277f805cad9d2fa5f4fd5dec224df6","affectsGlobalScope":true,"impliedFormat":1},{"version":"599ac4a84b7aa6a298731179ec1663a623ff8ac324cdc1dabb9c73c1259dc854","impliedFormat":1},{"version":"95c2ab3597d7d38e990bf212231a6def6f6af7e3d12b3bb1b67c15fc8bfd4f4a","impliedFormat":1},{"version":"585bc61f439c027640754dd26e480afa202f33e51db41ee283311a59c12c62e7","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"4e4f0301f1961c82daf04f96df0a52cc7018410e048b4db9eb9d25f887f4d32c","signature":"43301d028420b51cf11dc0e375dee7d03c5a908e49ce85e16793dbd1d5acabea"},{"version":"0046758a8237181581869afab50e21b5287c6e0ea98e04f5843168a48a278168","signature":"e9629b89f6cd33a040577695d89edf882e66287d30c87121f5fc07f404decd25"},{"version":"2e19656c513ded3efe9d292e55d3661b47f21f48f9c7b22003b8522d6d78e42f","impliedFormat":1},{"version":"ddecf238214bfa352f7fb8ed748a7ec6c80f1edcb45053af466a4aa6a2b85ffe","impliedFormat":1},{"version":"896eec3b830d89bc3fb20a38589c111bbe4183dd422e61c6c985d6ccec46a1e9","impliedFormat":1},{"version":"907dab3492fc59404ecf40f9ad655251741c5f2e471bb0376d11dae3e27cb1d8","impliedFormat":1},{"version":"8629340be5692664c52a0e242705616c92b21330cb20acf23425fff401ac771f","impliedFormat":1},{"version":"81477bb2c9b97a9dd5ce7750ab4ae655e74172f0d536d637be345ba76b41cd92","impliedFormat":1},{"version":"55a6b0318ec658ff37bc88e18a93e5f10ddad7257b379b71abf39e6868b8d4d2","impliedFormat":1},{"version":"b7d85dc2de8db4ca983d848c8cfad6cf4d743f8cb35afe1957bedf997c858052","impliedFormat":1},{"version":"83daad5d7ae60a0aede88ea6b9e40853abcbe279c10187342b25e96e35bc9f78","impliedFormat":1},{"version":"3a4e276e678bae861d453944cf92178deaf9b6dcd363c8d10d5dd89d81b74a0c","impliedFormat":1},{"version":"db9661c9bca73e5be82c90359e6217540fd3fd674f0b9403edf04a619a57d563","impliedFormat":1},{"version":"f7a5ab7b54bdc6a13cf1015e1b5d6eeb31d765d54045281bfeefcdfcc982a37c","impliedFormat":1},{"version":"ec99a3d23510a4cb5bdc996b9f2170c78cde2bfa89a5aee4ca2c009a5f122310","impliedFormat":1},{"version":"37c2d356ebbcf0d3d33b9213fd1922b47eba5c15ec0cb15a411c5dc4d35499bf","impliedFormat":1},{"version":"4557295358e5ced5f8ba3b8383a299e615093c50ece06af13d9531a536c3fa8b","signature":"41a23f47c8dfeed537a9ce175a069a8aafc2ef70753d027a7d23d631bc1c3034"},{"version":"add1ce7a69db6093250244751f696161479d8e616e2949d6f4d267b7d5309d0b","signature":"a7f2275200766f1ce46682fc4e39615f05e78e709aec9574f049c68fc490ea83"},{"version":"a8050669599b16534362b91c04c7eeb1c5ac7cbddcb244537ec2035d7ed32d2f","signature":"3448bdd3307aead3cd722b655ea79b81df44b83e6410318d4407efdf7069c2f0"},{"version":"8692e63cad9009b5ba22b4148597d76ddecbe9fd77dd6a40840606024166ba44","signature":"b07d5051f8f98f199765e80d433dc9294dd867433bedbf1ee068f1a5c6add20d"},{"version":"0229dcc6fdcbaa1defa323f48a5d4f45a8dade502c2622866c9a8e43d9eec0df","signature":"aacb92033c33af740c57f2401e6dfeeb91a58caebfbada7a281f0a35dc889502"},{"version":"cd9161557f36681d2efeede8e9add1d42209a0b06e2a2d2e4c4c2bfd41ee8f55","signature":"39caafa8eca6e1b25a97d98e9c01d5e0e69bfca321b2f2b626c50e19cf3647e5"},{"version":"aa2fec611accd5f2f07c80242fa45c9d346913af82fd8eb51ded034e646d971d","signature":"31d088e55566a8f1dc252e25881bc40bf39f5bedfebde17b7115e1e33bd75bb9"},{"version":"d7e329d3d991b4a848b2221ab85bf3e53033286bf19dfb63e87de9fad4ea4cba","signature":"6c4f8c32d43889ff7637aae1c86c9bf9a536771fae04f64c7457f7513ebb6b38"},{"version":"882c6137bee85e18d91bb379232569acd2cc35a92b4fb2784b9b78a5e7628390","signature":"1e2a68009d58c548d10ab7c153bdac800e2b672528913c6cbe52433e842fada1"},{"version":"5e85f07d73f50a2ae62f677eb757a36feb2c03c68c48cd84baa5d771bc4a86dd","signature":"32103869294d142e19946ac146f494b3878f75191e6d9e96913d44868bc1b6cb"},{"version":"39d5dad3ba88d880a1f0d8d01aa120561dd4b339d0afc89690234488e29473a5","signature":"0f9c90ea6967cd79a28894ae18fe17056905874df5140adc61efae0119382c9e"},{"version":"3def28895bd131c7719c047b117e7184b70f80a59fe9ff15f597ab044f11bc2c","signature":"c588f8a9b3ef0640f1988a2ca6e0c9e7457e66cf2d3f6d14cbbcfadadd79e28e"},{"version":"5aa80cbc76ce9e5f86e92f4705965ce2f6f47c33fb0acf010a1d24c47e0ee7fe","signature":"512c049d8ca62546029fa1954b71e635f82de8d9113d0ddd3ebac023a2ffbeda"},{"version":"ba391e0f887720ede1e7924b344cdbf07aa587996ee0ec095e69f350830c9914","signature":"76c62981ddd9d1a7a9a4efe30ee470d2bfb263a81124aa96c42c1a3b93039503"},{"version":"4dc087782bde5811a9fdd4a9c1e88013945d13d74f8630261f95a26790f351ee","signature":"60cd780eb939eefa9454bef827ad9a03625e8b87b9f437249c41cd0da21d025e"},{"version":"17f70bbc70ef9f11a148aa230879669100be8f71c64f7fd2d62f2924dc966f5c","signature":"e230b4bc73aad5fdf400faa82ef55b8d5d4ecce9d475c287f825205df0099521"},{"version":"f79637c4f9291ef95d5ed87b4ca214da83d230fdc545cdb5371ddc743cffb94e","signature":"4a1a4058b983f315d2233dbd7d96f0a1074b03daeef7cd7acc3e5868d31f75bf"},{"version":"116af9b10e21818cad8445de9260d780cdc48b5b99184052c60967deba881202","signature":"b3b848602a730dbdead58405d1e72cc665af2cd29f6bb3e41485f203071e3293"},{"version":"911f0abc025446633f27443f5ea82d197fabb0521dd10b45a32deefda70e0257","signature":"35e988e8cfa7260c27807007eef1320ef79c9691bd5d7eece7bfdca2162b75c3"},{"version":"32f23b16eb841067abc67606786c67ba31125a00c791b77365825b6f7d210b7a","signature":"16c3c4a895b9388e841ef72d3550c779c8418ac18722c74261732a18725236ed"},{"version":"12adfc904619a46a2a3fa61961c740040c101868ed90b37d76fd290e5e87ba5d","signature":"8b4c83eda1d716da4ccc6503fa4fa6f45eb809880a5a7b6807bc7337d1756ebe"},{"version":"e7d44baf5d9577fd187c6f22085bf99e7c99433dccbab75a0934f2e3c89487d5","signature":"4b3d248e010ea74f465c2ba01fa4976b6665fcda0855384335d1c682c06c8298"},{"version":"46a86f07908d19dcf08c72a07a3b5d93228093b0524d3e160f09089ebdb072d6","signature":"72a60b2a892d4df0c7f030fe5f7ed293b67e00591b46435c18d79889ac070be3"},{"version":"25e5c8b73c6ad21f39e8e72f954090f30b431a993252bccea5bdad4a3d93c760","impliedFormat":1},{"version":"5bf595f68b7c1d46ae8385e3363c6e0d4695b6da58a84c6340489fc07ffc73f8","impliedFormat":1},{"version":"b87682ddc9e2c3714ca66991cdd86ff7e18cae6fd010742a93bd612a07d19697","impliedFormat":1},{"version":"87d3ab3f2edb68849714195c008bf9be6067b081ef5a199c9c32f743c6871522","impliedFormat":1},{"version":"86bf2bfe29d0bc3fbc68e64c25ea6eab9bcb3c518ae941012ed75b1e87d391ae","impliedFormat":1},{"version":"8d9c4957c4feed3de73c44eb472f5e44dfb0f0cb75db6ea00f38939bd77f6e84","impliedFormat":1},{"version":"00b4f8b82e78f658b7e269c95d07e55d391235ce34d432764687441177ae7f64","impliedFormat":1},{"version":"57880096566780d72e02a5b34d8577e78cdf072bfd624452a95d65bd8f07cbe0","impliedFormat":1},{"version":"10ac50eaf9eb62c048efe576592b14830a757f7ea7ed28ee8deafc19c9845297","impliedFormat":1},{"version":"e75af112e5487476f7c427945fbd76ca46b28285586ad349a25731d196222d56","impliedFormat":1},{"version":"e91adad3da69c366d57067fcf234030b8a05bcf98c25a759a7a5cd22398ac201","impliedFormat":1},{"version":"d7d6e1974124a2dad1a1b816ba2436a95f44feeda0573d6c9fb355f590cf9086","impliedFormat":1},{"version":"464413fcd7e7a3e1d3f2676dc5ef4ebe211c10e3107e126d4516d79439e4e808","impliedFormat":1},{"version":"18f912e4672327b3dd17d70e91da6fcd79d497ba01dde9053a23e7691f56908c","impliedFormat":1},{"version":"2974e2f06de97e1d6e61d1462b54d7da2c03b3e8458ee4b3dc36273bc6dda990","impliedFormat":1},{"version":"d8c1697db4bb3234ff3f8481545284992f1516bc712421b81ee3ef3f226ae112","impliedFormat":1},{"version":"59b6cce93747f7eb2c0405d9f32b77874e059d9881ec8f1b65ff6c068fcce6f2","impliedFormat":1},{"version":"e2c3c3ca3818d610599392a9431e60ec021c5d59262ecd616538484990f6e331","impliedFormat":1},{"version":"e3cd60be3c4f95c43420be67eaa21637585b7c1a8129f9b39983bbd294f9513c","impliedFormat":1},{"version":"ceab6f5ec17081f0fe22cc49fba800a6a0927bc5fda02a7540909bbb5ed6dbc5","signature":"19a102ed2b4c50a31b480c52f5fcf73301abffc816dc812539d58550d90c289e"},{"version":"b40885a4e39fb67eb251fb009bf990f3571ccf7279dccad26c2261b4e5c8ebcd","impliedFormat":1},{"version":"2d0e63718a9ab15554cca1ef458a269ff938aea2ad379990a018a49e27aadf40","impliedFormat":1},{"version":"530e5c7e4f74267b7800f1702cf0c576282296a960acbdb2960389b2b1d0875b","impliedFormat":1},{"version":"1c483cc60a58a0d4c9a068bdaa8d95933263e6017fbea33c9f99790cf870f0a8","impliedFormat":1},{"version":"07863eea4f350458f803714350e43947f7f73d1d67a9ddf747017065d36b073a","impliedFormat":1},{"version":"396c2c14fa408707235d761a965bd84ce3d4fc3117c3b9f1404d6987d98a30d6","impliedFormat":1},{"version":"0c46e15efeb2ff6db7c6830c801204e1048ccf0c8cc9ab1556b0b95832c9d1c9","impliedFormat":1},{"version":"c475aa6e8f0a20c76b5684658e0adaf7e1ba275a088ee6a5641e1f7fe9130b8a","impliedFormat":1},{"version":"a42db31dacd0fa00d7b13608396ca4c9a5494ae794ad142e9fb4aa6597e5ca54","impliedFormat":1},{"version":"4d2b263907b8c03c5b2df90e6c1f166e9da85bd87bf439683f150afc91fce7e7","impliedFormat":1},{"version":"db6eec0bf471520d5de8037e42a77349c920061fb0eb82d7dc8917262cbf0f17","impliedFormat":1},{"version":"13c83c04f3cbd2da8276c6290b75f295edf309b4f907f667f1b775d5f048f47e","impliedFormat":1},{"version":"ca70001e8ea975754a3994379faca469a99f81d00e1ff5b95cabac5e993359aa","impliedFormat":1},{"version":"d6fa1d345cf72ead5f57dd2561136b7d09b774891d81822087499b41b3f04913","impliedFormat":1},{"version":"3bdc578841f58bfd1087e14f81394ece5efd56b953362ef100bdd5bd179cd625","impliedFormat":1},{"version":"2bc15addade46dc6480df2817c6761d84794c67819b81e9880ab5ce82afb1289","impliedFormat":1},{"version":"247d6e003639b4106281694e58aa359613b4a102b02906c277e650269eaecede","impliedFormat":1},{"version":"fe37c7dc4acc6be457da7c271485fcd531f619d1e0bfb7df6a47d00fca76f19c","impliedFormat":1},{"version":"159af954f2633a12fdee68605009e7e5b150dbeb6d70c46672fd41059c154d53","impliedFormat":1},{"version":"a1b36a1f91a54daf2e89e12b834fa41fb7338bc044d1f08a80817efc93c99ee5","impliedFormat":1},{"version":"8bb4a5b632dd5a868f3271750895cb61b0e20cff82032d87e89288faee8dd6e2","impliedFormat":1},{"version":"2a3e6dfb299953d5c8ba2aca69d61021bd6da24acea3d301c5fa1d6492fcb0ec","impliedFormat":1},{"version":"017de6fdabea79015d493bf71e56cbbff092525253c1d76003b3d58280cd82a0","impliedFormat":1},{"version":"cf94e5027dd533d4ee448b6076be91bc4186d70f9dc27fac3f3db58f1285d0be","impliedFormat":1},{"version":"74293f7ca4a5ddf3dab767560f1ac03f500d43352b62953964bf73ee8e235d3d","impliedFormat":1},{"version":"06921a4f3da17bed5d4bc6316658ce0ea7532658a5fc575a24aa07034c1b0d3d","impliedFormat":1},{"version":"90ee466f5028251945ee737787ee5e920ee447122792ad3c68243f15efa08414","impliedFormat":1},{"version":"34c17533b08bd962570d7bdb838fcaf5bcf7b913c903bc9241b0696a635b8115","impliedFormat":1},{"version":"1d567a058fe33c75604d2f973f5f10010131ab2b46cf5dddd2f7f5ee64928f07","impliedFormat":1},{"version":"5af5ebe8c9b84f667cd047cfcf1942d53e3b369dbd63fbea2a189bbf381146c6","impliedFormat":1},{"version":"5e126f7796301203e1d1048c1e5709ff9251f872a19f5ac0ee1f375d8128ef9b","impliedFormat":1},{"version":"147734cfd0973548fb6ef75d1e7d2c0b56bb59aad72b280784e811d914dc47d6","impliedFormat":1},{"version":"d2594d95d465026ebbee361f4819dc7b3146f4a8b42091ffb5dd90f9ceb345ab","impliedFormat":1},{"version":"e399d54c1b272a400ed446ca35d5e43d6b820723c2e5727b188ebea261e7cc2e","impliedFormat":1},{"version":"123568587c36c9f2a75091d8cdf8f287193855ba5aa10797b4fc320c80920b7f","impliedFormat":1},{"version":"6deffa531bdb8817b363505e88d957653d0c454f42c69e31588d00102cd1a076","impliedFormat":1},{"version":"973551068756351486afe706b240eb4dc83678ab2d829a1c6b1a19871394fd5f","impliedFormat":1},{"version":"e647d13de80e1b6b4e1d94363ea6f5f8f77dfb95d562748b488a7248af25aabf","impliedFormat":1},{"version":"9b7b0209a8841f5ffa60ccdfae26f7dc70ea4e7e446a603ef4732e84f1bb1b4f","impliedFormat":1},{"version":"5edc4b81a61ea5e0319b32d8f581d9643cb747cf44477b16af048f62d358c433","impliedFormat":1},{"version":"d47c9f84b00def208cbfdd820f8d10425ead9dbf36350d77fb55d5ef6857dabc","impliedFormat":1},{"version":"3bc5f767d5e0cd548c92e4623e0a7f4486889a72d2ca9cbc81df760669270dcc","impliedFormat":1},{"version":"20cf19c8028a7b958e9c2000281d0f4c4cd12502fef7d63b088d44647cdd607b","impliedFormat":1},{"version":"799780c3726407eaa2e09e709c376ec459582f6f9c41d9643f863580cecf7ff8","impliedFormat":1},{"version":"37280465f8f9b2ea21d490979952b18b7f4d1f0d8fab2d627618fb2cfa1828e3","impliedFormat":1},{"version":"52e29afa525973fc7cff28c4b6b359d91ad030d4aa198f060f813d4abcadb099","affectsGlobalScope":true,"impliedFormat":1},{"version":"a890cccdc380629c6cd9e9d92fff4ca69b9adddde84cc503296ada99429b5a3b","impliedFormat":1},{"version":"168b6da36cf7b832173d7832e017bc6c6c7b4023bf6b2de293efb991b96bca44","impliedFormat":1},{"version":"05b39d7219bb2f55f865bca39a3772e1c0a396ea562967929d6b666560c85617","impliedFormat":1},{"version":"bcae62618c23047e36d373f0feac5b13f09689e4cd08e788af13271dbe73a139","impliedFormat":1},{"version":"2c49c6d7da43f6d21e2ca035721c31b642ebf12a1e5e64cbf25f9e2d54723c36","impliedFormat":1},{"version":"5ae003688265a1547bbcb344bf0e26cb994149ac2c032756718e9039302dfac8","impliedFormat":1},{"version":"e1744dbace6ba2051a32da3c6b40e0fc690810a87b9ad4a1925b59f8f7157a34","impliedFormat":1},{"version":"ba8a615335e3dfdf0773558357f15edfff0461db9aa0aef99c6b60ebd7c40344","impliedFormat":1},{"version":"6921769648e4b83bb10e8fcf7011ea2d8f7de5d056daacf661648935a407376e","impliedFormat":1},{"version":"dd21167f276d648aa8a6d0aacd796e205d822406a51420b7d7f5aa18a6d9d6d9","impliedFormat":1},{"version":"3dea56c1745af2c31af0c84ecc6082044dc14cfa4d7366251e5bf91693eecd8b","impliedFormat":1},{"version":"eb6360635bc14b96a243bd5134e471f3ad26b0ecaf52d9d28621e443edb56e5c","impliedFormat":1},{"version":"3ad73b5b1d43cb8ba3975b2999396e10bf3ed015b3337876079caf07f22501d9","impliedFormat":1},{"version":"62a64260ea1dada7d643377c1a0ef3495363f4cca36adf7345e8566e7d7f419b","impliedFormat":1},{"version":"8b15e8af2fc862870418d0a082a9da2c2511b962844874cf3c2bad6b2763ca10","impliedFormat":1},{"version":"3d399835c3b3626e8e00fefc37868efe23dbb660cce8742486347ad29d334edd","impliedFormat":1},{"version":"b262699ba3cc0cae81dae0d9ff1262accf9832b2b7ee6548c626d74076bff8fe","impliedFormat":1},{"version":"057cac07c7bc5abdcfba44325fcea4906dff7919a3d7d82d4ec40f8b4c90cf2f","impliedFormat":1},{"version":"d94034601782f828aa556791279c86c37f09f7034a2ab873eefe136f77a6046b","impliedFormat":1},{"version":"fd25b101370ee175be080544387c4f29c137d4e23cad4de6c40c044bed6ecf99","impliedFormat":1},{"version":"8175f51ec284200f7bd403cb353d578e49a719e80416c18e9a12ebf2c4021b2b","impliedFormat":1},{"version":"e3acb4eb63b7fc659d7c2ac476140f7c85842a516b98d0e8698ba81650a1abd4","impliedFormat":1},{"version":"04d4c47854061cc5cefc3089f38e006375ae283c559ab2ce00763bca2e49516b","impliedFormat":1},{"version":"6a2146116c2fa9ca4fefa5c1d3de821462fc22e5330cda1196be15d439728c51","impliedFormat":1},{"version":"3b10140aae26eca9f0619c299921e202351c891b34e7245762e0641469864ffd","impliedFormat":1},{"version":"c0c0b22cefd1896b92d805556fcabda18720d24981b8cb74e08ffea1f73f96c2","impliedFormat":1},{"version":"ceec94a0cd2b3a121166b6bfe968a069f33974b48d9c3b45f6158e342396e6b2","impliedFormat":1},{"version":"49e35a90f8bd2aa4533286d7013d9c9ff4f1d9f2547188752c4a88c040e42885","impliedFormat":1},{"version":"3261b6d56270a3d8535f34c2fdad217cfba860d0f74f154f0a6a2031d0c8daf9","impliedFormat":1},{"version":"7eca5b6e1cd1c28637103d2b6c44e8b89035a53e515ff31ae3babc82e6c8e1f9","impliedFormat":1},{"version":"49c9c8316d59f6175e6e0439b1d5ef1218f02ce622d1a599449de30645559eed","impliedFormat":1},{"version":"e4c48be0ffac936fb60b19394739847145674582cbc7e24000d9fd35ab037365","impliedFormat":1},{"version":"215de2c70639abaf351b8ff69041e44a767ecffc5e8d2ac13ca3f201853fa1fb","impliedFormat":1},{"version":"d228c7773484140fac7286c9ca4f0e04db4a62acb792a606a2dda24bef70dc21","impliedFormat":1},{"version":"8e464886b1ff36711539ffa15ec2482472220271100768c1d98acfdf355a23ba","impliedFormat":1},{"version":"fb0135c4906ff44d3064feebd84bae323ebb7b59b8ce7053d34e7283d27c9076","impliedFormat":1},{"version":"178c8707a575baddc8f529a6dbd5d574a090e3498b2d525753db7938c74227c3","impliedFormat":1},{"version":"ae81e464a7db70637d07b93582b051487c7d119ac7e1bab1b1582a96e631b3f7","impliedFormat":1},{"version":"148634fcee440c7bd8c1339b97455aaadc196b0229ffc8dc8b85965a7d65b380","impliedFormat":1},{"version":"d3c60c4cf88594f84f7f5ca5f87d59090787bfcf032e86d4f03d58394b826910","impliedFormat":1},{"version":"f3c3f17825c6a78681186da04c2f3a0f1c60cfa95f3d4b82bbbd6ebd57214a6a","impliedFormat":1},{"version":"eb45a1782ef50423c1ffac4d2a89c60004f4e2d25ed8e7dcb9e24e6cf984ccdb","impliedFormat":1},{"version":"e4c94ea55cac34117f423db62a62df7121006b0e69ab77d27f98cef27973f876","impliedFormat":1},{"version":"97e4a153ac1feffdc4cbdf8f2b8ecfe8a2c651c3ba949fd5fbcc1e29d8e7d283","impliedFormat":1},{"version":"0b801e25d3e163d246c70e059cd529e5edd280692c6b384c20314825b23511bb","impliedFormat":1},{"version":"97f2a8c88656c95104fe635a2c254b0ec431591d26838e1fbc2a6db6b822a1df","impliedFormat":1},{"version":"d120b9af0cd364387d676de7c662643006ea2d942f7d5af18ca3024fee73c149","impliedFormat":1},{"version":"274097c2dff5959c9a6242b87aa858e772f0d269dd2847e44e4f6c5c0922937f","impliedFormat":1},{"version":"9a5ce34f28948d1b84dea73942e10b8a36fa10dc958356589c8f03aea866c96d","impliedFormat":1},{"version":"75897ed38868fdd8ce289475d2812e4d3e5cb64bcf812e44ebbf3f5e21ae486d","impliedFormat":1},{"version":"d994fb6705faaae18b9d71ba2d89b4a7e5e77c2b801a3dae51c0821da4a90acb","impliedFormat":1},{"version":"b30cc18b84468d3fa20ac04ca5ba9bed5a03431fc8a22bcf2c266c132baa1d3f","impliedFormat":1},{"version":"fda3e93361b17be4f24a12ac951b3c87cbae8e81b351d849caec342c94f932fa","impliedFormat":1},{"version":"f0ce99f229795d96ffbcee89e1de6c6d9dc52b4c31bfe6fe002893e243c305b4","impliedFormat":1},{"version":"680bd67092c080fefa2045bba990378e0c0f2ad5ed56b7fe928b4287413d7842","impliedFormat":1},{"version":"0840cd0220ad530dbc6e8549161e9304dac9d520c3b78caeeeb6af194a9f0588","impliedFormat":1},{"version":"a54f60678f44415d01a810ca27244e04b4dde3d9b6d9492874262f1a95e56c7d","impliedFormat":1},{"version":"84058607d19ac1fdef225a04832d7480478808c094cbaedbceda150fa87c7e25","impliedFormat":1},{"version":"27abd2f2ed5aaac951b12b8332aac7970c9cf0cfd88c458f0f016228180b4293","impliedFormat":1},{"version":"901c640dced9243875645e850705362cb0a9a7f2eea1a82bb95ed53d162f38dd","impliedFormat":1},{"version":"ebb0d92294fe20f62a07925ce590a93012d6323a6c77ddce92b7743fa1e9dd20","impliedFormat":1},{"version":"b499f398b4405b9f073b99ad853e47a6394ae6e1b7397c5d2f19c23a4081f213","impliedFormat":1},{"version":"ef2cbb05dee40c0167de4e459b9da523844707ab4b3b32e40090c649ad5616e9","impliedFormat":1},{"version":"068a22b89ecc0bed7182e79724a3d4d3d05daacfe3b6e6d3fd2fa3d063d94f44","impliedFormat":1},{"version":"e70d18d1352550a028f48d74e126a919c830267b38c76ddae4dc1571476a462a","impliedFormat":1},{"version":"5624b09ca38ea604954f0422a9354e79ada3100305362a0da79555b3dd86f578","impliedFormat":1},{"version":"24830e279f5773a4108e0cbde02bdcb6c20b1d347ff1509f63eed031bf8b3190","impliedFormat":1},{"version":"8899fd9f8ab5ce2b3af7ba0e1a47eede6a2a30a269283cc4a934ab755d0aadaa","impliedFormat":1},{"version":"f10759ece76e17645f840c7136b99cf9a2159b3eabf58e3eac9904cadc22eee5","impliedFormat":1},{"version":"363dd28f6a218239fbd45bbcc37202ad6a9a40b533b3e208e030137fa8037b03","impliedFormat":1},{"version":"c6986e90cf95cf639f7f55d8ca49c7aaf0d561d47e6d70ab6879e40f73518c8d","impliedFormat":1},{"version":"224d293a02b7d22edb77b4ab89c0d4f63b95ecd7c0698776719f33863a77ffdc","impliedFormat":1},{"version":"1518707348d7bd6154e30d49487ba92d47b6bd9a32d320cd8e602b59700b5317","impliedFormat":1},{"version":"ede55f9bac348427d5b32a45ad7a24cc6297354289076d50c68f1692add61bce","impliedFormat":1},{"version":"d53a7e00791305f0bd04ea6e4d7ea9850ccc3538877f070f55308b3222f0a793","impliedFormat":1},{"version":"4ea5b45c6693288bb66b2007041a950a9d2fe765e376738377ba445950e927f6","impliedFormat":1},{"version":"7f25e826bfabe77a159a5fec52af069c13378d0a09d2712c6373ff904ba55d4b","impliedFormat":1},{"version":"7ffef1ed1c2bc7d9cf2fc134a7e8c68b10416cdbe8e70da8a4bd7ad5c8698d9c","impliedFormat":1},{"version":"63c0926fcd1c3d6d9456f73ab17a6affcdfc41f7a0fa5971428a57e9ea5cf9e0","impliedFormat":1},{"version":"eb524eabfa1809d54dd289374c0ce0ed4f145abb878687e4fd5e67f91d7d08a6","impliedFormat":1},{"version":"4ef0a17c5bcae3d68227136b562a4d54a4db18cfa058354e52a9ac167d275bbb","impliedFormat":1},{"version":"b748dd4ccc072a2b7194b898dc8996a2cb56bfa15ccdb60ac0d2f9eaa8e28e9d","impliedFormat":1},{"version":"64269ed536e2647e12239481e8287509f9ee029cbb11169793796519cc37ecd4","impliedFormat":1},{"version":"c06fd8688dd064796b41170733bba3dcacfaf7e711045859364f4f778263fc7b","impliedFormat":1},{"version":"b0a8bf71fea54a788588c181c0bffbdd2c49904075a7c9cb8c98a3106ad6aa6d","impliedFormat":1},{"version":"434c5a40f2d5defeede46ae03fb07ed8b8c1d65e10412abd700291b24953c578","impliedFormat":1},{"version":"c5a6184688526f9cf53e3c9f216beb2123165bfa1ffcbfc7b1c3a925d031abf7","impliedFormat":1},{"version":"cd548f9fcd3cebe99b5ba91ae0ec61c3eae50bed9bc3cfd29d42dcfc201b68b5","affectsGlobalScope":true,"impliedFormat":1},{"version":"14a8ec10f9faf6e0baff58391578250a51e19d2e14abcc6fc239edb0fb4df7c5","impliedFormat":1},{"version":"81b0cf8cd66ae6736fd5496c5bbb9e19759713e29c9ed414b00350bd13d89d70","impliedFormat":1},{"version":"4992afbc8b2cb81e0053d989514a87d1e6c68cc7dedfe71f4b6e1ba35e29b77a","impliedFormat":1},{"version":"1810b0b14614e53075d4d1b3e6be512bde19b1ed3a287925c0d24bae8585fa1b","impliedFormat":1},{"version":"1c390420d6e444195fd814cb9dc2d9ca65e86eb2df9c1e14ff328098e1dc48ae","impliedFormat":1},{"version":"ec8b45e83323be47c740f3b573760a6f444964d19bbe20d34e3bca4b0304b3ad","impliedFormat":1},{"version":"ab8b86168ceb965a16e6fc39989b601c0857e1fd3fd63ff8289230163b114171","impliedFormat":1},{"version":"62d2f0134c9b53d00823c0731128d446defe4f2434fb84557f4697de70a62789","impliedFormat":1},{"version":"12056fec06740770f1d14d47dad75ba54a9a8e0f40e766b38172a44b42112886","impliedFormat":1},{"version":"ec8db3ff5638196d3debc149d61fb4d604e73e12ffd28ea02a326ad40d2240bf","impliedFormat":1},{"version":"e8c0a909df5d29c77c9b5037b9a25ec6be0b70e0d3f5ce95ae0923f2582d6c1c","impliedFormat":1},{"version":"9fad83dff123fb56c6d11f8973ebc6cffec784f5d5c0cc2572344df246bfa0dc","impliedFormat":1},{"version":"c7865dbee3c1955c9c91b107ac95e51c27ee67dc2b85893ee91d4ce46f37ea99","impliedFormat":1},{"version":"5776c61de0f11da1c3cf8aafc3df524e8445201c96a7c5065a36dc74c2dc0ef6","impliedFormat":1},{"version":"94ffa91cb9f8eba1f468c5439994d4544ad24658e1192060f76267b767114445","impliedFormat":1},{"version":"7f0f90d0ffdd54875c464b940afaa0f711396f65392f20e9ffafc0af12ccbf14","impliedFormat":1},{"version":"483255952a9b6240575a67f7beb4768bd850999a32d44d2c6d0ae6dfcdafe35c","impliedFormat":1},{"version":"a1957cc53ce2402d4dc5c51b7ccc76b30581ab67bea12a030a76300be67c51d8","impliedFormat":1},{"version":"8149e534c91fc2bcb3bf59f7c1fab7584382abfc5348055e7f84d2552c3de987","impliedFormat":1},{"version":"c280ec77789efcf60ea1f6fd7159774422f588104dae9dfa438c9c921f5ab168","impliedFormat":1},{"version":"2826b3526af4f0e2c8f303e7a9a9a6bb8632e4a96fece2c787f2df286a696cea","impliedFormat":1},{"version":"77ced89806322a43991a88a9bd267d6dc9e03fd207a65e879804fa760292a03b","impliedFormat":1},{"version":"c8ff3a75cd1c990cbe56080b1d254695c989136c9521cb1252c739788fe55c83","impliedFormat":1},{"version":"485f7d76af9e2b5af78aac874b0ac5563c2ae8c0a7833f62b24d837df8561fb9","impliedFormat":1},{"version":"8bdf41d41ff195838a5f9e92e5cb3dfcdc4665bcca9882b8d2f82a370a52384e","impliedFormat":1},{"version":"4f42e9c177a3c68913560c086a889285cd51c20b4d54a44bae77c17dee7cfa46","impliedFormat":1},{"version":"fe8a3e5492c807cc5cfc8dda4e6464aff0f991dc54db09be5d620fb4968ba101","impliedFormat":1},{"version":"31f931c21c7a121317c5b4d0223e6c13f616312ddb481409e54393340bd61ed9","impliedFormat":1},{"version":"654bcc87bc095d6a2248a5889ec057b38cae6052744b48f4d2922a7efac4554f","impliedFormat":1},{"version":"cad0f26943006174f5e7508c0542873c87ef77fa71d265968e5aa1239ad4459c","impliedFormat":1},{"version":"0be66c79867b62eabb489870ba9661c60c32a5b7295cce269e07e88e7bee5bf3","impliedFormat":1},{"version":"eed82e8db4b66b1ea1746a64cd8699a7779138b8e45d495306016ce918b28440","impliedFormat":1},{"version":"3a19286bcc9303c9352c03d68bb4b63cecbf5c9b7848465847bb6c9ceafa1484","impliedFormat":1},{"version":"6cdf8f9ca64918a2f3c2679bc146d55f07490f7f5e91310b642bc1a587f2e17e","impliedFormat":1},{"version":"3b55c93b5d7a44834d9d0060ca8bad7166cf83e13ef0ed0e736da4c3dbe490a2","impliedFormat":1},{"version":"d1f8a829c5e90734bb47a1d1941b8819aeee6e81a2a772c3c0f70b30e3693fa9","impliedFormat":1},{"version":"3517c54fba6f0623919137ab4bdb3b3c16e64b8578f025b0372b99be48227ad7","impliedFormat":1},{"version":"19b3d0c212d241c237f79009b4cd0051e54971747fd89dc70a74f874d1192534","impliedFormat":1},{"version":"0d0202235a9c54a36e08502ddee21ad24013eff328831630b438ca927ff15b11","impliedFormat":1},{"version":"f17963b9935dd2142c08b006da53afeeaca2c9a600485f6eb9c018b96687275b","impliedFormat":1},{"version":"1afd33ee26eb73a31cca8870ef7da768d1f13827957847d68d0270ce4568b4c2","impliedFormat":1},{"version":"2eba01b5ecb62f6e3204c7a45918760a5b9047071dc98e2ba86ff777f8d605bf","impliedFormat":1},{"version":"a24adc6e8aa778f0b66578f842a3c4fb7bdca5f8bb183c5e90c9746676a33454","impliedFormat":1},{"version":"12a6a37d9676938a3a443a6bd9e8321d7221b6ad67b4485753322dc82a91e2a1","impliedFormat":1},{"version":"6c4833182ba7a753200bf30986d254653c1ac58855d784edd8dfe82f5db98954","impliedFormat":1},{"version":"69eeee4818209fdb59544d6f74bd6ff024944bdd4050a33577f62376d5cada8e","impliedFormat":1},{"version":"4cfbd2a7a4afee212bfb0c9c3cb6e4c7d48366e0565bf5b43a4cd96c91cf14bf","impliedFormat":1},{"version":"37c175e28375e157933b40ca98eeb608e05f2583821a0fae564dc04614d2d95e","impliedFormat":1},{"version":"3f20a041a051abfb2b47a66611cf4bcbf263605f5469ed7e8b51b3977892d83f","impliedFormat":1},{"version":"7de33f94f482eee2f6d1d8f24427b737e2c4006792ec4c2b87da0a426e741c4d","impliedFormat":1},{"version":"79134a050ccec1692c31f1dacccd05ce4fcdacdf98f0fa56546b98eb8bdefead","impliedFormat":1},{"version":"24f1b6865be734484de2baf99146122137654c5f5f28086c5cee97b998bfcd5c","impliedFormat":1},{"version":"398feb1537ae0409646b0489bac99a9f0d757a2048f0009255f8e35e9c0f9828","impliedFormat":1},{"version":"3da4432a9c24123f98f6f1ddc5cda9c9eedf0a8853d06321803dbc5a116e5270","impliedFormat":1},{"version":"afc60e07200c5eae65b702f95d83096de54d99fa6eb2e0154e83b5e11c520bda","impliedFormat":1},{"version":"f4651affee2900f19746d1bf0fb1c45e77f57576197561ddc90b7272835c3f37","impliedFormat":1},{"version":"19527fc5a08c68414a234b02ae9b9619cdb4b811435d12c0af528e5640236f6b","impliedFormat":1},{"version":"20a629bc3f82d238f596230637365b8aec8284c963d13dafdd4c8e2746be5e64","impliedFormat":1},{"version":"01c48e5bf524d3fc2a3fa5c08a2e18d113ad1985bc3caea0503a4ea3a9eee64a","impliedFormat":1},{"version":"68969a0efd9030866f60c027aedbd600f66ea09e1c9290853cc24c2dcc92000f","impliedFormat":1},{"version":"4dbfad496657abd078dc75749cd7853cdc0d58f5be6dfb39f3e28be4fe7e7af5","impliedFormat":1},{"version":"348d2fe7d7b187f09ea6488ead5eae9bfbdb86742a2bad53b03dff593a7d40d1","impliedFormat":1},{"version":"becdfb07610e16293af2937e5f315a760f90a40fec4ffd76eb46ebcb0b3d6e16","impliedFormat":1},{"version":"710926665f4ada6c854b47da86b727005cc0e0831097d43f8c30727a7499788c","impliedFormat":1},{"version":"3888f0e43cd987a0dfa4fc16dd2096459deea150be49a2d30d6cf29d47801c92","impliedFormat":1},{"version":"f4300c38f9809cf811d5a9196893e91639a9e2bb6edf9a4f7e640c3c4ce765ec","impliedFormat":1},{"version":"676c3327721e3410b7387b13af857f4be96f2be91b3813a724eedc06b9ce52d7","impliedFormat":1},{"version":"10716e50bcd2a25cecf2dd993f0aadf76f12a390d2f7e91dc2cac794831e865e","impliedFormat":1},{"version":"81a8f1f6218d0acc8cd2cf8b5089d21b45cf812bb5820affe3bab058b46cba7b","impliedFormat":1},{"version":"fa69921924cf112fa523a18215a3bfb352ac3f498b46e66b879e50ca46cc9203","impliedFormat":1},{"version":"9b82a268ba0a85015cb04cd558582c7949a1b91b6761292b9360d093c18e1dd1","impliedFormat":1},{"version":"ccfb77fcac04c34442ffca82ae90c8dd2a0ec1689ace547fab9a0ae337dd4752","impliedFormat":1},{"version":"7b464488950d74ca5037da375308fc0c94a539378fd0e9554556df45483aad02","impliedFormat":1},{"version":"beebde754323e430b4ecf5b9f837a05b1667b3df86bd924b52c4f80f20b3d660","impliedFormat":1},{"version":"40eda068f71d159edc51c273a01948282d6e3d38dd2430944595d526dc4b40b9","impliedFormat":1},{"version":"c790db6044ce1bbafc46f13bde46b9f0065de155b26a199f442fe064f6b05d63","impliedFormat":1},{"version":"f405e934163ed30905b4682eb542bb2d446e59c477871be9d29f92ab474d522a","impliedFormat":1},{"version":"8294ddd1c6ea4ed9ec190a2d41500539c1623e274d5a67786d6b09849cb98d45","impliedFormat":1},{"version":"666d6d6d9f2298f8d8d17ac7a34ac9ca9a59e09fc97b1ae505df6ab4934e2dbe","impliedFormat":1},{"version":"f3941ac359b8377c0ccce596a2bd3cde8986279f42d75290b0272f3ab1aa604d","impliedFormat":1},{"version":"de3d39262355af808ff74b3df62aaad0ad3cbde76c13fb4fa6fb6e4cc817e78e","impliedFormat":1},{"version":"8c38034476af70d7ad430f69cb960c5bd6efc9962f266b39ed54dd8e9cad566c","impliedFormat":1},{"version":"757f7967151a9b1f043aba090f09c1bdb0abe54f229efd3b7a656eb6da616bf4","impliedFormat":1},{"version":"786691c952fe3feac79aca8f0e7e580d95c19afc8a4c6f8765e99fb756d8d9d7","impliedFormat":1},{"version":"734614c9c05d178ceb1acf2808e1ca7c092cf39d435efc47417d8f744f3e4c0b","impliedFormat":1},{"version":"d65a7ea85e27f032d99e183e664a92f5be67c7bc7b31940957af6beaaf696844","impliedFormat":1},{"version":"5c26ad04f6048b6433f87556619fd2e50ba6601dcdf3276c826c65681197f79d","impliedFormat":1},{"version":"9c752e91fe237ce4857fbbef141bee357821e1e50c2f33a72c6df845703c87d5","impliedFormat":1},{"version":"f926160895757a498af7715653e2aedb952c2579a7cb5cc79d7b13538f9090bd","impliedFormat":1},{"version":"255be579a134ab321af2fefb52ace369a11ffb4df09d1fbfc1ed1a43c1e5eec5","impliedFormat":1},{"version":"fa05a4a765755e92c1dcab306ef3648fa4aa108494b6e10d2329db8b89e89908","impliedFormat":1},{"version":"ea385ec05b32ad43bbd1002a7c553bbc6935754504d60dc38ee64cc8b3c21768","impliedFormat":1},{"version":"d61821435a95c7a660d5850ce6fe9c4400787595009853d982343b8089724319","impliedFormat":1},{"version":"3e56d3093cd84dac94761258d6d98226671481d43e93321f9825692940fcc0c2","impliedFormat":1},{"version":"25091d25f74760301f1e094456e2e6af52ceb6ef1ece48910463528e499992d8","impliedFormat":1},{"version":"ed79978235b685e7e9d2ac149c6ddaf602ce7e3a30725c20023e57f011760593","impliedFormat":1},{"version":"3345fc785abb65f2263f91ba092bb77470d949eddb41fc208256b964c2ccd5cb","impliedFormat":1},{"version":"d899e749dab44c9c0c1cf2118324ca1e6651113268551aa7a1e320b161953505","impliedFormat":1},{"version":"8a49075f007383f24df5b52376e41198e341a7b715da34a90b2c54b8fc8d4bcc","impliedFormat":1},{"version":"83f5494ed714e7898414a220e56886e00ffb58c80cbf745ecc45d6aae55f051a","impliedFormat":1},{"version":"28a019a0ebc44efd6b9148bbaf688bed1778d6977950da3d277f09e17687a46a","impliedFormat":1},{"version":"a67eff604aaeeae1c93f410dd9f176ab75a3fac2d2761e0bd9d83f933ec3e657","impliedFormat":1},{"version":"76a0210cd218dd17365ac0b7a99936804ede147c0a899d6bd2a9ac29cc8d4fea","impliedFormat":1},{"version":"853d02f4f46ca9700fefd0d45062f5b82c9335ba2224ca4d7bd34d6ae4fc4a7f","impliedFormat":1},{"version":"5f9ab7ba179f92fa3c5dddafec778a621fe9f64e2ba8c264ddf76fe5cf9eaf93","impliedFormat":1},{"version":"a0bce0fb40b88d17305f113ed02c4014329be52e8168b01fe825c049e9a37028","impliedFormat":1},{"version":"364e53fe15122e9d37aa8ee2c8eb037cde59bf5890b46a8205f4516b529501c0","impliedFormat":1},{"version":"1a577fdc45901cf461d4edc7697860c63a60526f60b7b2ba8ff7c89a9e7a1932","impliedFormat":1},{"version":"a6da29e6495bf303eb5f0b65dca3f92711b9cd6729eb1bed3e29dbc8b0fc2604","impliedFormat":1},{"version":"c35bd33a53356146889d87a05b34fc5a130ba93bc1bb36d021c0a7c817c4cc8d","impliedFormat":1},{"version":"f6067be38e8661d7e68287db5a602501897e2fbb385dfa2ecd170dd9f4524e8a","impliedFormat":1},{"version":"506b96750c75c5e5af820ac8356f27b2bcd0b077a02d33d90e87be9f0604f59d","impliedFormat":1},{"version":"d9c87237f57a1ca503eebb89514f65c48800b10b5574f7cda978addab1474dfb","impliedFormat":1},{"version":"b3cc1bb7311f35569b531e781d4a42d2b91f8dfd8bc194cc310c8b61011d6e43","impliedFormat":1},{"version":"33cff836b608822d3b4a52f523d964472c5437d08d81dbd1171ccf743276a223","impliedFormat":1},{"version":"8d334e97bce2fca74d7f2a416e1bd0c434785bffbca381632e8676dd4dee5324","impliedFormat":1},{"version":"4b610a25fe9e65ebf0b80d83dd52d4b40044768016821f947de60ba1b59cda4d","impliedFormat":1},{"version":"a9452e81c28c642c2f095844c3473d979eba5ae89726ad52b15ea86b3e112ee2","impliedFormat":1},{"version":"dc4a2cf12254395c8ae3fb4c61e6fd9f7c16110be66483599f9641941416988f","impliedFormat":1},{"version":"82b4045609dc0918319f835de4f6cb6a931fd729602292921c443a732a6bb811","impliedFormat":1},{"version":"8a2c67c55dfab4ea1f6e378cfa4b5cb60d8e8931316500f5b59c201674f6105c","impliedFormat":1},{"version":"b7b45ff1345f8e6bd6109a5b6ef0394c2e3bcbe48830516d9e78e20592ce468a","impliedFormat":1},{"version":"e5eb4863b7fc8515078dc09cd2f98fd179ff1a55216ecdc57d2dec7ce13e36c1","impliedFormat":1},{"version":"81785a3ea03d6db981ddfcf8fb1bd1377f985564def845c55e49e16f171deec4","impliedFormat":1},{"version":"537a2b61594512c5e75fad7e29d25c23922e27e5a1506eb4fce74fe858472a6e","impliedFormat":1},{"version":"8f9a2a6ddbd11ecbbc430ae8ce25528e696206f799ef1f22528569caf6ce580c","impliedFormat":1},{"version":"e05e03e1687d7f80f1569fdae117bb7b97feef1e839a61e1b3c61ffca8cc67c9","impliedFormat":1},{"version":"b311d973a0028d6bc19dfbaae891ad3f7c5057684eb105cfbeec992ab71fbc13","impliedFormat":1},{"version":"8a49e533b98d5c18a8d515cd3ae3bab9d02b6d4a9ac916e1dba9092ca0ebff15","impliedFormat":1},{"version":"fcb26ad5a6c39ce71dfac5dc16b3ed0e1a06a6dc8b9ac69112c935ad95fcad69","impliedFormat":1},{"version":"6acdef608420511aa0c9e3290b37d671bab4f719ffc2a2992c2e63a24605a657","impliedFormat":1},{"version":"291df5da0d84d1452cd68abfbcca08a3f96af610bf0e748528ba8d25784ce2b1","impliedFormat":1},{"version":"176cda558a7f76813f463a46af4607a81f10de5330c0f7a43d55982163aa0493","impliedFormat":1},{"version":"6621af294bd4af8f3f9dd9bd99bd83ed8d2facd16faa6690a5b02d305abd98ab","impliedFormat":1},{"version":"5eada4495ab95470990b51f467c78d47aecfccc42365df4b1e7e88a2952af1a3","impliedFormat":1},{"version":"6b08ada439e3c7fba3e6d18c19f934e7bbea3f34979f2490074f0623b849e8e4","impliedFormat":1},{"version":"40e9c2028b34c6c1e3281818d062f7008705254ee992d9857d051c603391e0f4","impliedFormat":1},{"version":"07a9aa7f3facdfac577ed4aa0c166295d54637508942a2154566d87804a33ae2","impliedFormat":1},{"version":"4a34de405e3017bf9e153850386aacdf6d26bbcd623073d13ab3c42c2ae7314c","impliedFormat":1},{"version":"993bcd7e2dd9479781f33daab41ec297b8d6e6ccc4c8f9b629a60cc41e07e5c8","impliedFormat":1},{"version":"714a7869be4ff21fa7be0dc183569db5e6818ca22882a79d2bb3a7801f5bfab4","impliedFormat":1},{"version":"dfa99386b9a1c1803eb20df3f6d3adc9e44effc84fa7c2ab6537ed1cb5cc8cfb","impliedFormat":1},{"version":"4cb85ba4cf75f1b950bd228949ae508f229296de60cf999593e4dd776f7e84e8","impliedFormat":1},{"version":"e39730c031200579280cae4ea331ec4e0aa42f8f7ad19c3ec4b0b90414e40113","impliedFormat":1},{"version":"e90bd7922cb6d591efd7330d0ba8247ec3edf4c511b81346fd49fff5184e6935","impliedFormat":1},{"version":"1b581d7fcfacd6bbdabb2ceae32af31e59bf7ef61a2c78de1a69ca879b104168","impliedFormat":1},{"version":"20f7f9e30ac8cbf38189b3adafbd945a755a049b082f27d89d1d5d52f46818fe","impliedFormat":1},{"version":"c749b03596746c41abf1e8ed6b5a6a1bcd316c00dc39a337cc152780efc593bb","impliedFormat":1},{"version":"846953ab15b2bf3a06da6ec485be611455c5c83a02102138dd01102f3e6307de","impliedFormat":1},{"version":"218ed8ccd7078df39a26ccc59a094919d7ed1c0cd0b0182233deffda851ac3c6","impliedFormat":1},{"version":"8422f4ff58293a827a8bf401bb36f7eefbf981ae9aac48643d19c1e5439ee1bc","impliedFormat":1},{"version":"f70ab2e7bd23db437c2d5ed8690c401a921afbd5d3998a6dd2aab90d9efbaf35","impliedFormat":1},{"version":"fdda29d1f7eb83a912e34ae73f4e6e612350a7d1a496d5facc2f75487e2a1601","impliedFormat":1},{"version":"8ec6b7dc9062dd5c3c2fcc54bbf24e1e8a32b29ed902abe9192ddd0fd5f5f2a7","impliedFormat":1},{"version":"52e7386606a26e912bd39cad7752cc33009aefbb062d4a45e557c29095987095","impliedFormat":1},{"version":"3a6ce66cd39bc030697a52508cfda7c248167467848964cc40bd992bd9ce71e0","impliedFormat":1},{"version":"b4ec75c8a71c180e886ffccb4b5391a5217d7e7077038de966e2b79553850412","impliedFormat":1},{"version":"58c7522c1b1c94667777664bb9b26be14159220a28abf43e42807815e3102e14","impliedFormat":1},{"version":"d1666062675fe2f5408bfc458dec90de7279820eea20890b19484250c324b8ea","impliedFormat":1},{"version":"aed88228359e87a1b1a4d3d45f5b6555724c01ac81ecd34aa56d4a0a01ba6910","impliedFormat":1},{"version":"25307c3fd3840b5bd50bf95240a134854e80f736a4666711ea8ea40ba706eaa9","impliedFormat":1},{"version":"4fce1ce36a7f6fa69d3954cd685d27995123b683d31819218d204ca6bdcbfc53","impliedFormat":1},{"version":"a26bd8cdefaaf5a4cfe2c2f9e5b9114072f6d274ed4422eb7dcd1f72705a7eb2","impliedFormat":1},{"version":"5bbcd14f0138f4e65971ed5cb5606e8591ffefe3ac78ac310b164a975ea38f4f","impliedFormat":1},{"version":"0220b23c1c15820dcbb94eb74b8671020b53cd192a708e4d1828f290149e7e89","impliedFormat":1},{"version":"4adc1491e1338de6745d009222786747f50d67ac34d901420fbaefbf1b51b58c","impliedFormat":1},{"version":"ab0926fedbd1f97ec02ed906cf4b1cf74093ab7458a835c3617dba60f1950ba3","impliedFormat":1},{"version":"f1a661906cd0e7fa5b049b15bdef4b20a99abca08faac457eeb2b6407f30d12f","impliedFormat":1},{"version":"7f5a6eac3d3d334e2f2eba41f659e9618c06361958762869055e22219f341554","impliedFormat":1},{"version":"626291e7b45a4b6871649c908fbbc5ac98009a5182e2594fbfe80b860f513c77","impliedFormat":1},{"version":"4093c47f69ea7acf0931095d5e01bfe1a0fa78586dbf13f4ae1142f190d82cc4","impliedFormat":1},{"version":"4fc9939c86a7d80ab6a361264e5666336d37e080a00d831d9358ad83575267da","impliedFormat":1},{"version":"f4ba385eedea4d7be1feeeac05aaa05d6741d931251a85ab48e0610271d001ce","impliedFormat":1},{"version":"348d5347f700d1e6000cbdd1198730979e65bfb7d6c12cc1adedf19f0c7f7fca","impliedFormat":1},{"version":"6fa6ceb04be38c932343d6435eb6a4054c3170829993934b013b110273fe40af","impliedFormat":1},{"version":"396e7b817fc4f5461b92f9a03325c2ebb09711aebcee5c41c5fd3e738eb78526","impliedFormat":1},{"version":"4116c4d61baab4676b52f2558f26fe9c9b5ca02c2792f9c36a577e7813029551","impliedFormat":1},{"version":"a294d0b1a9b16f85768553fdbf1d47f360dbff03649a84015c83fd3a582ba527","impliedFormat":1},{"version":"8f2644578a3273f43fd700803b89b842d2cd09c1fba2421db45737357e50f5b1","impliedFormat":1},{"version":"639f94fe145a72ce520d3d7b9b3b6c9049624d90cbf85cff46fb47fb28d1d8fe","impliedFormat":1},{"version":"8327a51d574987a2b0f61ea40df4adddf959f67bc48c303d4b33d47ba3be114a","impliedFormat":1},{"version":"00e1da5fce4ae9975f7b3ca994dcb188cf4c21aee48643e1d6d4b44e72df21ee","impliedFormat":1},{"version":"b991d92a0c3a48764edd073a5d28b6b4591ec9b7d4b2381067a57f36293637d0","impliedFormat":1},{"version":"51b4ab145645785c8ced29238192f870dbb98f1968a7c7ef2580cd40663b2940","impliedFormat":1},{"version":"100802c3378b835a3ce31f5d108de149bd152b45b555f22f50c2cafb3a962ead","impliedFormat":1},{"version":"fd4fef81d1930b60c464872e311f4f2da3586a2a398a1bdf346ffc7b8863150f","impliedFormat":1},{"version":"354f47aa8d895d523ebc47aea561b5fedb44590ac2f0eae94b56839a0f08056a","impliedFormat":1},{"version":"b152c7b474d7e084e78fa5eb610261a0bfe0810e4fd7290e848fdc88812f4504","impliedFormat":1},{"version":"67f2cd6e208e68fdfa366967d1949575df6ccf90c104fc9747b3f1bdb69ad55a","impliedFormat":1},{"version":"603395070ec53375882d53b585430e8f2dc6f77f4b381b22680d26c0a9595edc","impliedFormat":1},{"version":"cef16d87ff9aed3c5b96b47e0ac4277916c1c530f10eedfce4acaeacefddd3bb","impliedFormat":1},{"version":"fab33f402019d670257c8c833ffd78a7c9a99b4f7c23271e656cdbea1e89571f","impliedFormat":1},{"version":"976d20bb5533077a2135f456a2b48b7adb7149e78832b182066930bad94f053a","impliedFormat":1},{"version":"589713fefe7282fd008a2672c5fbacc4a94f31138bae6a03db2c7b5453dc8788","impliedFormat":1},{"version":"26f7f55345682291a8280c99bb672e386722961063c890c77120aaca462ac2f9","impliedFormat":1},{"version":"bdc2312da906d4129217238545d7e01e1d00b191beea1a9529b660de8b78834f","impliedFormat":1},{"version":"62b753ed351fba7e0f6b57103529ce90f2e11b949b8fc69c39464fe958535c25","impliedFormat":1},{"version":"514321f6616d04f0c879ac9f06374ed9cb8eac63e57147ac954e8c0e7440ce00","impliedFormat":1},{"version":"3c583256798adf31ef79fd5e51cd28a6fc764db87c105b0270214642cf1988aa","impliedFormat":1},{"version":"abdb70e24d3b39bf89aa07e769b33667c2d6f4ddcb4724735d72a941de6d4631","impliedFormat":1},{"version":"151aa7caace0a8e58772bff6e3505d06191508692d8638cd93e7ca5ecfa8cd1b","impliedFormat":1},{"version":"82f75b2de1456b0be46945c6c68547f032f11238d07db45bbc9c93fca6abfe41","impliedFormat":1},{"version":"812e55580eb591f3c04245345be8c9dce378b26238fb59d704e54a61e6e37c83","impliedFormat":1},{"version":"1de7ee494c7ac185e6abf94428afe270e98a59f1bb4768e4bea7804645a0d57d","impliedFormat":1},{"version":"2c12f912bab4b1eb797b2fded3f295fee98736f8053a08d0032becbecb4b34b1","impliedFormat":1},{"version":"2b234fce994b272403881b675d6ae2e2afb2a8be8bdec71002ff8ff2d5b59bd0","impliedFormat":1},{"version":"97ba9ccb439e5269a46562c6201063fbf6310922012fd58172304670958c21f6","impliedFormat":1},{"version":"50edac457bdc21b0c2f56e539b62b768f81b36c6199a87fbb63a89865b2348f0","impliedFormat":1},{"version":"d090654a3a57a76b5988f15b7bb7edc2cdc9c056a00985c7edd1c47a13881680","impliedFormat":1},{"version":"e0f69e399a1c3b367058918f3a6f9e5f29a0db26330c28208173556a8c48a0a3","impliedFormat":1},{"version":"6fe7b261e7242f5fb09166aa561110a9e356faf0452cc1828e18f17396dd7eb4","impliedFormat":1},{"version":"c18f1b2b05af7add7728cb54f0fa186e09bd33ae9b9c2a57e75bd45458db1b43","impliedFormat":1},{"version":"97776c2abc3ad9cdf00f858980a12e4b8033de0f6f7df62caa32e930726d665d","impliedFormat":1},{"version":"73bf1badc6969cf5eba2018b20dce1cc0dc00b056e66870abdca1f7b1275a648","impliedFormat":1},{"version":"8c8ee967649fed1f3bf84b3eea1928a2148e7320e5e04bdaa7b0f73946131156","impliedFormat":1},{"version":"638969ce0b4ed7122f2891bd934bb76718e1b3e10663651dbc230669939a297f","impliedFormat":1},{"version":"bb832613b3a3246183c65561fdeb87b6b35324bf46ce407dfa376a49ad259545","impliedFormat":1},{"version":"10291d06c7f3242f3e773262522d2bbaff113109b897b5ef0e7f2cc596408978","impliedFormat":1},{"version":"03239379d84009ee413ab759d85c8a5e254b1a97b175527e8cf946cee7d84621","impliedFormat":1},{"version":"57044825e3199de7211f1cc94a1d8080d5a1b3556f08b1263402014234983d06","impliedFormat":1},{"version":"6b567540de9240ef5b14e86239dd821be6e90274df164ef61a11921344019410","impliedFormat":1},{"version":"c2f426df0b5346a824013c507c611656884f4681e23782ae5bc396cd6ad6fe00","impliedFormat":1},{"version":"f737d4c256faf88b0b84255da24ac95de364db1c5564d67856fc2be2571e477d","impliedFormat":1},{"version":"082be20017cd0774d12bf0a928d0834bcc83454e706dfb3ebf2c314b8065cdc8","impliedFormat":1},{"version":"7e0b933925fa33fca25738c4daabc959f315f1ca3b2431a724069db01ab4e53f","impliedFormat":1},{"version":"34dce1ce616534c11ce825f1acf55cbe45bf7f38b48ddf451fe0ebdb8b04924d","impliedFormat":1},{"version":"561b27749052f76b32ac63a1da5a3f2f87e60fce44189dee4f9a728d4390d144","impliedFormat":1},{"version":"677b3a027ec7df033eb7df33eecfe77e066debd14d983a22916bcc67214b222c","impliedFormat":1},{"version":"547fbf3353abf40a827b1b48b832f9139e41cc600dac954cd4dd1c8faf1df517","impliedFormat":1},{"version":"dc0a609a8ab9995e439c887be9327c272400a3aadd76dae41f6c4eae9af723b8","impliedFormat":1},{"version":"843d9c252981c2cabd6fb28b0e2fc730df962ed2b78d208ca60745d434a2801e","impliedFormat":1},{"version":"c0c5ebf4cc5ef8ed64733c3dea8de54fbb45bb0a40306d26d840e50a954327cd","impliedFormat":1},{"version":"408c753ef793f71dffa590cfc3a95e5246ab67238f71083b7a96056b986e8964","impliedFormat":1},{"version":"6650fd077aa1900d99c2a52af9dd79971322e2991e70e463331d0cd60fe9f914","impliedFormat":1},{"version":"e074ffc31f9e3c284b4220abf9c4082e549842c6af98ae2d80b17071b21a573b","impliedFormat":1},{"version":"e32ceea1db1cf16192a7fd97fdc689dc625ad1c57555a2dc18e7b5c019216482","impliedFormat":1},{"version":"0359c9ffe72d124c521a64dcb4f37a6391e9234e0496a38f62ba9b18be1f1d89","impliedFormat":1},{"version":"5eda29b34e4f3ebed27585c7afb3a23f6017ec83feff42518cb7d4b9eebe7688","impliedFormat":1},{"version":"b1cdc896fa6216d1ff898423395f5ca6bf3b1118411b7f5ccc732c8acbd37e19","impliedFormat":1},{"version":"8d986fc98555e4b017177816f51b86236a357ff9c468f4c86e5b4096775f1639","impliedFormat":1},{"version":"37441514e58c1412999e02a2931b179a4e9330a5b67dd8f75b3dbee19e1b33db","impliedFormat":1},{"version":"a1c9f45912f4ad052795e0ef5743d8e1a8400efb0393a13c47c2b6320bdd9406","impliedFormat":1},{"version":"6402c4b62a62f59d78ad558c0abaa7a8059e966f083c99b89fd4b24302353413","impliedFormat":1},{"version":"a8378c2611bbac166238618fb9bc0af95b77ac59447d9046af1c4188ad758c1b","impliedFormat":1},{"version":"c745295efc9d176836701a358fcab80c77ca7a63f4c8e7bfaf0da3a0910607be","impliedFormat":1},{"version":"9074405ae056ee3066a3cdba3ad29eed41ce32c74838b6ca09e14ba8ed9e3123","impliedFormat":1},{"version":"d486cfb79e7cb30710b2f3cc8c02909fe71b34e26938ba536100ff207e38749f","impliedFormat":1},{"version":"1ec802755650072584f58dd6fc680d1519e790528a4658c17eb12c9a27f16053","impliedFormat":1},{"version":"d668cba43ed8e4364f2625bfc373f51ca7a0ef4333aadee0b953c6a8ce0ad953","impliedFormat":1},{"version":"38acd0b49cda3d9393086f8bd92754eeebaa4bf52b9f3dd0031d464f15d4bddf","impliedFormat":1},{"version":"b80ab976d1968b44e78d7c6312fa7895989e30604712001e504ec74c95d50c0f","impliedFormat":1},{"version":"7cdbc72e654c4cc3a95cf44a0f050a5affa822d626501731ea688deb0662b027","impliedFormat":1},{"version":"af12968b48a6d0d45a4737c6a02cdadae2ace6738b65a1c3425928677154db38","impliedFormat":1},{"version":"8d6283ed4e0f2164daeabb7f26be39785c8366f5deae0f39c1246486ca3df24c","impliedFormat":1},{"version":"ab9ea3afaee16b10a6d7d2fdde5c3c66a154e1f93266832a439ea4ae8f92ed3c","impliedFormat":1},{"version":"94ef22b94d7cafa97366308aba95caee00b8c740567c518dae3dcfd09c58a4d1","impliedFormat":1},{"version":"66332888aba679ae44ff8405d814a794a8328730d008ab5ca2c3b0629a1acc38","impliedFormat":1},{"version":"de9d02955efdcff476bd95184b75e321d962b8fc22f51b2033da443ffa33c029","impliedFormat":1},{"version":"8134618a47361a523fc5daea803db42a8f0751944e8a314128cdb4d51a4d0359","impliedFormat":1},{"version":"3cac1d701152a399417ca60ec47f9ca83fe96a4c519bfc4b9ceb79578a894bc3","impliedFormat":1},{"version":"d0a951a1c211da46f226452a1b758b046ff797e5c1a4d285ccec361749aa937b","impliedFormat":1},{"version":"dba5745334b583e1b8f7ef055939cc750c546c250a74b95e0ede1ad5b863400a","impliedFormat":1},{"version":"801f5c06b1bdb1f520fa9d12ad1cb1f0f5a60233ede2d6313a1894f4c48208af","impliedFormat":1},{"version":"afe7e63907fb014390c8b2fc1bea86232a17431faccfbd8248e9ecf848c698e4","impliedFormat":1},{"version":"6511247b2b1388ebd4e4d7131c3d1f89633b41fe5bf44bca8f90ac7fcaf1a6b2","impliedFormat":1},{"version":"894b86b56bc595744893ed21374abd71aebd7a346b711dbf4c1a39f714edcda2","impliedFormat":1},{"version":"c92258027ef4f99109f802bd82af908afa731ccc379373b5ee69cf420b40f060","impliedFormat":1},{"version":"ba19ddf03bdbd63a4a44ebb761439016507911f153b1b618b7aaca18801b1e1d","impliedFormat":1},{"version":"73f6fd0ee5da5198efd378a07383da0337b8e73a30a53420a46a4a098ea34f26","impliedFormat":1},{"version":"1b5962b26e6698523cbd71a11281f59f6c4f57099fdebb11cf68fe77abac8c1c","impliedFormat":1},{"version":"0f55b6a44c5b1571426a2b4d261c8eb69452079aeefbbd43eacfdc202cdf415f","impliedFormat":1},{"version":"afda2e399d17b76e199f56029d053308fa3d93a90b9565b06a81be01fb86865c","impliedFormat":1},{"version":"dc995c080b26c63ddf09c5a0e949fbbbd562a081fbb3004d25db42a82fec3b58","impliedFormat":1},{"version":"1ac440b352b34743ec4ee150e2e4c4d3cbe20795c803651e9a976fd0cd124bb8","impliedFormat":1},{"version":"6141b4f6daf9d8770f32c174fb185714e97851a345b5734256b2f615f6bfcbb5","impliedFormat":1},{"version":"cd454337b92454283ac6fa6a271d8688a9a875b32386a33c747fd0cb67407843","impliedFormat":1},{"version":"bff7736d65b7f6d60604986ad2a213f19c7828fbcea2fb8a559c0933f3b9ad07","impliedFormat":1},{"version":"2bb117bc6d5073d2562fdc273ffd5264f7b0a92e7b82b55001095873a202ea81","impliedFormat":1},{"version":"1749042b52c8e5fe166dd24318d18ab56db865cca56829a69ac42e6bb55b001f","impliedFormat":1},{"version":"5f8eaa67e3d8b40ab42257d96a714fafc809ab526142230a09f58635c192cd22","impliedFormat":1},{"version":"c1a706fc9524ed51280a131c9b7a4123f9617d18f9b84cc278eb010e695e477d","impliedFormat":1},{"version":"e76a5c0be446202df4edfbc86834735c54010f5241709daac7096683fc18658e","impliedFormat":1},{"version":"a1a60e9c4aa722efc1c0402f7819c4fac46514a2004dabebacbfce89c524c5c3","impliedFormat":1},{"version":"71b7ae47fcf519a301fd2cd1067e3934380054e8fb3adf3d8d53ab74e080cd7a","impliedFormat":1},{"version":"59291388675b917b92ac856ccb5059201cbaa757a358c4a3c50b2dd63a4bee79","impliedFormat":1},{"version":"7ebf776831a1bcf30321f35361a30984036808cd74cf5ff8af65d9bad4c6ea0b","impliedFormat":1},{"version":"9810d5e27b657a2e13ba091c899460c92aeeb2230bd83bd68a57936d8004cb0a","impliedFormat":1},{"version":"77fd2f9105721dee1b0680be07e3dbd7080719f3b7d4fa34ee2f84b5fe301bd1","impliedFormat":1},{"version":"f43678d8b19945df13814b6838120b7c7b8dfdce7569296b1b1976eaf085c705","impliedFormat":1},{"version":"07c21b91f2117a8bd633177327a3b86856b36c387d2f2e045073072fb8a37714","impliedFormat":1},{"version":"b5cf7048adaa2cae79df7f1a0774e2894714c254f56feb89b8a4e07705191c87","impliedFormat":1},{"version":"13baee3bc1768b9923c49f08193495c4e6781262a64d59a2e63c141b771a9fcb","impliedFormat":1},{"version":"fdc91fa6795007d22e50ba7ce3dc88c2fe6499899ab3e89f09aa5cff98e0bbb6","impliedFormat":1},{"version":"0638bd7f74c6483ef226245204f32be39d0fc8e7c4c6c8a5b1577f8cacb70410","impliedFormat":1},{"version":"bef9d33e702185e457db9520d32df7e7e72bd33ed7b9596b211d9c8819e5402b","impliedFormat":1},{"version":"7ff5b052778be5d4b85df000ee9efe6c75e6e4f87861f2a1313987f5476d9ef5","impliedFormat":1},{"version":"b6914c5199969e2aa490bd347dfc5e289813b260dba1331d06c55b3cbcab9b6f","impliedFormat":1},{"version":"f65070ba38c06935f1ef31a318e5e9812363ec7791d556055cd025a882ee8e55","impliedFormat":1},{"version":"93e07e1a200d32977d8e2fce3de3db49fb9fb482b392611ca0cd3d66d0397444","impliedFormat":1},{"version":"21f2230f4af0aebfcfbadd985578bc55e5b8f43fd44bed0bf218823376033394","impliedFormat":1},{"version":"ac214369128f0b6a614bfa33ab522967abe8e970d2dfc7d523f5a3882d72c863","impliedFormat":1},{"version":"3caef1cb673fde4b700242696f2ae18aa9fe35bb93f2c3e6b5e6faf9fb34190a","impliedFormat":1},{"version":"16ae8a29108474734e2f5ec9806bf7a1d596e03265399bacb6cb0c6a122b7c9e","impliedFormat":1},{"version":"c20d73302303381a3b3f510e79576ca38d073294eb9ffcb3858d678b10e511aa","impliedFormat":1},{"version":"c94d37997cae72d328e2492bd572151ab72398daf771a3651672bbde53cede17","impliedFormat":1},{"version":"0a6c7438c05d0c1243030cb34884c927f3e393590242eef0b60f216df11248b9","impliedFormat":1},{"version":"2a198644dda57c757234a8d26b6e466f1fb610515a584478459a2d39c14900aa","impliedFormat":1},{"version":"2491ea15b97d8f9e6602d0dec94e57943be6150adaf6fccf8b77a5830c300743","impliedFormat":1},{"version":"2723a90d915173b65458fe6bad29131622704c0f1ab37b8913342461f61bd496","impliedFormat":1},{"version":"ebcfb573a78eedc5a034d975ef663e77eb2a81ec14bedcb55fc519d6699090f3","impliedFormat":1},{"version":"1fc17ce97c5ac21cb923f0f8eca6e3d6d8a514b6d66da902587cbb9d799f5c87","impliedFormat":1},{"version":"be93596d4f187f67d7845f895d10073ccbac95214f7ebf8671a438249b6e9787","impliedFormat":1},{"version":"13860f2e3e36443d23ad7241926a974077098f400da4cca6db3acf31d24dac81","impliedFormat":1},{"version":"fd0efac55de17fd65bbaa67b9586b6ff148cdda9b7161c77a0ac7b20b79c5645","impliedFormat":1},{"version":"9d74f18aad3b7c8a0336eacd427c659bd8bf57444426f6f0e41f8fca1ceee97a","impliedFormat":1},{"version":"f3c4bf50cd6721e827571cb942ebde7d12764b6f742a58b6c10e579127927ab1","impliedFormat":1},{"version":"297511a767d55924440a6014baf09e42509ce46c63c881380947b642d3d46a2d","impliedFormat":1},{"version":"33073bdc3cfd190a9b1c15a0abdcf611eeb924de4faa190e42227f631ca0ed70","impliedFormat":1},{"version":"83a6c79b5bc0c8b167ff323e2289c716346e4cb86008d2dd25b97c6454a47e35","impliedFormat":1},{"version":"4dd7bc31a2e62befc4dd7411c393bb6890d9fe38cc72fb813644936fca82aa0d","impliedFormat":1},{"version":"adc2bfa76a57b734c8a30695b2538c60e9352d4e354a68d51829ea43b768fed9","impliedFormat":1},{"version":"8c15f3c23f926d273dd648d74539d50e8e23400539116b77ea72d3d17fc8f294","impliedFormat":1},{"version":"980f3c0431cc8689000e35012e23a248dd781e3e5282057126624a7137a3bf42","impliedFormat":1},{"version":"9b8202cbccd1249e8640ea844c0dabbb99a604a6ba52feee963e557680ed285c","impliedFormat":1},{"version":"983f0fa9ecfa26398247e83ab68f60aa5cfe967905736047cad0864762d9eeea","impliedFormat":1},{"version":"473f53747832bc2588d9e9e0347d3fbcc8aa8e61124b4b4ed54185f930e4f80e","impliedFormat":1},{"version":"bf96e903108160a97d684bb1d0991faad9a0c9a209759a7338ea22fbd4510f75","impliedFormat":1},{"version":"ea99aa2e537966df22f8192e99929ee81719c1cf0b9d9d83d0c6fed53325ccc6","impliedFormat":1},{"version":"d8feeb2d114397bfaa224f022701560ede9ed33299d2ab26a5979b5be2bf6b49","impliedFormat":1},{"version":"ee0ff5abedfc526345058f7d0bdb94f038db1d2b85e527a72f8b1d50fd8585d1","impliedFormat":1},{"version":"61029abbe36480fb1b2ab39f2ae333b909e9e25ccf3e192b50909cf26140bcbe","impliedFormat":1},{"version":"d0060157c0e676a2da04b924645eba5892a702e9640c27b9d0aa84e8aa036421","impliedFormat":1},{"version":"2542bb610f66daadbbe7abd6cd6b61413e1b763aac9a599f765a0ec3bde1c370","impliedFormat":1},{"version":"7d0d9e7638a6800b06045dcaa53bae64dea4ec3027736ee26f7c1cff50d671bd","impliedFormat":1},{"version":"c8df12e4975f0e4bfe405a9a9319cd2fda5bd83ed3907c214b37ba77778ef2d2","impliedFormat":1},{"version":"1b601a6492f374b24723cc645a6731ca97209025edd688aed45fe6e6217c100a","impliedFormat":1},{"version":"74a907fa14655328575b29e4dbdf58440dd07c081d9d245f785c4143d10510c8","impliedFormat":1},{"version":"197da644ac572eae35c0e3cf933d24b120bdf09db3ea7e98965cc09eb1e9fb64","impliedFormat":1},{"version":"76d7adf42ea58b58559c7b85a37d84a3ea163c215f12c812df49b0ddec849ea6","impliedFormat":1},{"version":"ae375d815b60f98942bd1af5f865d4e9aa7977ac10e7fed8464b0dc42291b718","impliedFormat":1},{"version":"712b3998796d62fffe46f5cbb47453a141549c0cde81e04da8464243ec3e60e2","impliedFormat":1},{"version":"2d9f4d58554a246616eeaa090a2fb0dddccf412e88617975138389fb15770ca9","impliedFormat":1},{"version":"500561aaec426c1d47bcb10a78c275b7b59cb489c19eb72e1daf4d7d241e4f4a","impliedFormat":1},{"version":"c1bcf1e07567521e99973bdb3a2a7c6508c013356af71bdca9cc1be05560b0d6","impliedFormat":1},{"version":"1ad980595342aa0d8a6711e5c10c18107fcfd11d41cc72d8c1c2160ad11df9d3","impliedFormat":1},{"version":"7c3343bf83008efe46f7dbbbaace983b0f2b2e150edfed5d08493892d42633ff","signature":"1ce2876f803b25d752a531fc1cc48b963f73a19df3f8a6746e9a0f2f95938a53"},{"version":"90ee210adb95afdc326a461668b5fb2b406fec2b4500e5195cee3b2f316a840e","signature":"2fbe9b000c0db850fcee3aaee93b5b8418531bef194d7dfb9d5cd18e3fd593f8"},{"version":"68041d04acf55e24caf390ffd7ced8b1df493af318966d98fb08581405533f46","signature":"266aa5e69f4350a0b1ee18b210a7fdfcf5dd210cd61c2ede3804946753d7cd7e"},{"version":"1d879125d1ec570bf04bc1f362fdbe0cb538315c7ac4bcfcdf0c1e9670846aa6","impliedFormat":1},{"version":"4097bd34e56d6737381d5384a957f07dac545429fa78b458df548fbc320bc416","signature":"2503832ab9322399624f31de1dc8dad9acaac09c7d36c46afac04f8fba4c68fd"},{"version":"848a9da0519aa6167675843a024245b2c8166b18eab6cbbf54b21cb575a86183","signature":"6df0e11677f9c37f2253e752f8c4555a4d6958f7c02cdabf6bc65c0b31e3bbfe"},{"version":"03786eafe913c7887a474b3780f99996d2e8912090cffc67ee08a982cfc4bf02","signature":"92c90d66236656f1ceaff476f2006db24c20e8eee786035c3fd797106acebb91"},{"version":"bcae43f4d0bb77e1a4c3ca2882d5d247fc703af84673c369f3247944fcd43df0","signature":"dcdf71f69aaf20825ad3823e35ea6484e70dbbb78f9a9d739d1713ad03579b27"},{"version":"be7e3a7e9e914e50a526f24d01101b0eec10266bb5aab6ddb04b48f7e2542c18","signature":"4fea6f35ccfe5c583010905e3b68f69673c83a19f13c699434f6725d357f2678"},{"version":"6ada175c0c585e89569e8feb8ff6fc9fc443d7f9ca6340b456e0f94cbef559bf","impliedFormat":99},{"version":"e56e4d95fad615c97eb0ae39c329a4cda9c0af178273a9173676cc9b14b58520","impliedFormat":99},{"version":"bd821b87e2c0fb5f509cedf47da465c447451835ce0fe2a752c4fc53a9f95a5b","impliedFormat":99},{"version":"f1d7352c0f7041abb43e1054abb14fb8c53a13dd54bcc1d67b97d2c02bb5028c","impliedFormat":99},{"version":"fc820b2f0c21501f51f79b58a21d3fa7ae5659fc1812784dbfbb72af147659ee","impliedFormat":99},{"version":"08f88f75fc2f516413477606a4122b6d3f6eb6680e8eb79f3fda5a5d2ed306df","impliedFormat":99},{"version":"6ab9821afd2a06879620eb4e041b9492a90f294e9b733ae5eb022edaa3964a45","impliedFormat":99},{"version":"003533cc3fa10cc457668d4256d21a65706a67a04251962cfe85d240502f8d67","impliedFormat":99},{"version":"6c00cb8a4b187505dfe21aff242b07f69f84f5c832e8ab4357af69daaee1b0df","impliedFormat":99},{"version":"de14ddf9d780367c6a117bd8a1718d491aff66094186523b3eea680ea7035a7c","impliedFormat":99},{"version":"ee06b94d0521cfaf91e4b003518eeefc45bbd594b0c22955fe35be282958252b","impliedFormat":99},{"version":"9e5f8fdaeb03f1699392b4724a58ca7b47c5cbb6762920d2bfc722c265495ede","impliedFormat":99},{"version":"d05bd4d28c12545827349b0ac3a79c50658d68147dad38d13e97e22353544496","impliedFormat":99},{"version":"a1597b0039f39e9f3eeaf120f02d0c94a826fad30b027a2abfdb8d580c89be70","impliedFormat":99},{"version":"04ace6bedd6f59c30ea6df1f0f8d432c728c8bc5c5fd0c5c1c80242d3ab51977","impliedFormat":99},{"version":"57a8a7772769c35ba7b4b1ba125f0812deec5c7102a0d04d9e15b1d22880c9e8","impliedFormat":99},{"version":"1de82ba3718b2b3bc5333c5bc35da5cfc46d1b654edc012de46bbce48126fcce","impliedFormat":99},{"version":"3000898d25f5aba9733e6f5d4b240095aeab76964aad622d4cbf2ddde6248878","signature":"7a8eaa222b6603000a00ac7f50383fdfd54ea7dba7e8f17effe6b8c4aab2bd1a"},{"version":"cdf16a5d26bcae919f0abd6a5f7689689d51020b0f6493fb411528beb3d42101","signature":"a5f6a0d196335b5b988e9015ccc91222e627abf09a9d8362270b767cb10001bc"},{"version":"cf92f36adc7ae022bfa193cfc2acc82050e2259a81b8c44351d9ee0b426a1f49","signature":"452ba2f1dc68d8ad26442c107e166bf6b103edde09eb0035d084414bdeb0ca7e"},{"version":"6e2098976ed47aad056ea99dd336e34a889973b722ce6e6eef4279f728565ddc","signature":"4a32770ff8bb22f7990d8509442cf5670822783f5cfb13e13d0379fc8fb7f883"},{"version":"64089edc83778914d8a71dc3033792254d0c6b290d724070cef8aa90f1abf4d2","impliedFormat":1},{"version":"6c2369391695514bdfdd07208e423e41262561859f808b6f2793c53fb61805fe","signature":"fbaa27f02d82133afef915c639b79b5ab47d721bab1ae99f4fcd1704ca6f2c72"},{"version":"ccd0ea06d01f768847d7b943939ba49d29b42717f65d925214b86c40cb47a745","signature":"7511a389505669f877f4c4a5eb7715fba76b8d1fc7a8c9b1bcdedebd0c5db123"},{"version":"ed072d6a7f16dcd385437fb3ad5df5c5d15ceb216086e01c2e21223ff6f9a7ea","signature":"ffdfa1ce3aa203730a305ef6c72b488f5289e2d302e4a5685e6d02c291a5866a"},{"version":"62f99b4071470d7a5032398c18511cc388a4b21b81b3bd5be16c465e5fb4df37","signature":"950df39df3b15296df4f3f9e9bcb9afb3b671088237027b0a1b2e7d93c9cccf8"},{"version":"f5bc52ae804646d550a6910d77d2606244b7a5c0108d1294e08630546a543257","signature":"431823fd0889d3eb363763b841dc5d3c2947f713aa87c33086a83f083e784f33"},{"version":"8d7096beb6f7b7956417433562bcec8fee7bcf06901e15fbd0aca38b289be9ae","signature":"0d69c1b4d42146e395542713855e012d0eefb2b650b741957bf683f87f242fcf"},{"version":"8fe6499702d903a542361e0a025a90773b2ecc6dd1a1d525d8e0e8e2de86096f","signature":"0c1a01a6299a44cf6d16146d6038d987f59457af356499f21a578448ca110abd"},{"version":"07f666e86729a398a8179a74b8949dc33ba946ba3d063a80a73f11fa721e8abb","signature":"b1b05ef68433a4285c0de2e1e885902c15a285354f7e6469b4dc6e24baae79f9"},{"version":"cb5eaaa2a079305b1c5344af739b29c479746f7a7aefffc7175d23d8b7c8dbb0","impliedFormat":1},{"version":"bd324dccada40f2c94aaa1ebc82b11ce3927b7a2fe74a5ab92b431d495a86e6f","impliedFormat":1},{"version":"56749bf8b557c4c76181b2fd87e41bde2b67843303ae2eabb299623897d704d6","impliedFormat":1},{"version":"5a6fbec8c8e62c37e9685a91a6ef0f6ecaddb1ee90f7b2c2b71b454b40a0d9a6","impliedFormat":1},{"version":"e7435f2f56c50688250f3b6ef99d8f3a1443f4e3d65b4526dfb31dfd4ba532f8","impliedFormat":1},{"version":"6fc56a681a637069675b2e11b4aa105efe146f7a88876f23537e9ea139297cf9","impliedFormat":1},{"version":"33b7f4106cf45ae7ccbb95acd551e9a5cd3c27f598d48216bda84213b8ae0c7e","impliedFormat":1},{"version":"176d6f604b228f727afb8e96fd6ff78c7ca38102e07acfb86a0034d8f8a2064a","impliedFormat":1},{"version":"1b1a02c54361b8c222392054648a2137fc5983ad5680134a653b1d9f655fe43d","impliedFormat":1},{"version":"8bcb884d06860a129dbffa3500d51116d9d1040bb3bf1c9762eb2f1e7fd5c85c","impliedFormat":1},{"version":"e55c0f31407e1e4eee10994001a4f570e1817897a707655f0bbe4d4a66920e9e","impliedFormat":1},{"version":"a37c2194c586faa8979f50a5c5ca165b0903d31ee62a9fe65e4494aa099712c0","impliedFormat":1},{"version":"6602339ddc9cd7e54261bda0e70fb356d9cdc10e3ec7feb5fa28982f8a4d9e34","impliedFormat":1},{"version":"7ffaa736b8a04b0b8af66092da536f71ef13a5ef0428c7711f32b94b68f7c8c8","impliedFormat":1},{"version":"7b4930d666bbe5d10a19fcc8f60cfa392d3ad3383b7f61e979881d2c251bc895","impliedFormat":1},{"version":"46342f04405a2be3fbfb5e38fe3411325769f14482b8cd48077f2d14b64abcfb","impliedFormat":1},{"version":"8fa675c4f44e6020328cf85fdf25419300f35d591b4f56f56e00f9d52b6fbb3b","impliedFormat":1},{"version":"ba98f23160cfa6b47ee8072b8f54201f21a1ee9addc2ef461ebadf559fe5c43a","impliedFormat":1},{"version":"45a4591b53459e21217dc9803367a651e5a1c30358a015f27de0b3e719db816b","impliedFormat":1},{"version":"9ef22bee37885193b9fae7f4cad9502542c12c7fe16afe61e826cdd822643d84","impliedFormat":1},{"version":"b0451895b894c102eed19d50bd5fcb3afd116097f77a7d83625624fafcca8939","impliedFormat":1},{"version":"bce17120b679ff4f1be70f5fe5c56044e07ed45f1e555db6486c6ded8e1da1c8","impliedFormat":1},{"version":"7590477bfa2e309e677ff7f31cb466f377fcd0e10a72950439c3203175309958","impliedFormat":1},{"version":"3f9ebd554335d2c4c4e7dc67af342d37dc8f2938afa64605d8a93236022cc8a5","impliedFormat":1},{"version":"1c077c9f6c0bc02a36207994a6e92a8fbf72d017c4567f640b52bf32984d2392","impliedFormat":1},{"version":"600b42323925b32902b17563654405968aa12ee39e665f83987b7759224cc317","impliedFormat":1},{"version":"32c8f85f6b4e145537dfe61b94ddd98b47dbdd1d37dc4b7042a8d969cd63a1aa","impliedFormat":1},{"version":"2426ed0e9982c3d734a6896b697adf5ae93d634b73eb15b48da8106634f6d911","impliedFormat":1},{"version":"057431f69d565fb44c246f9f64eac09cf309a9af7afb97e588ebef19cc33c779","impliedFormat":1},{"version":"960d026ca8bf27a8f7a3920ee50438b50ec913d635aa92542ca07558f9c59eca","impliedFormat":1},{"version":"71f5d895cc1a8a935c40c070d3d0fade53ae7e303fd76f443b8b541dee19a90c","impliedFormat":1},{"version":"252eb4750d0439d1674ad0dc30d2a2a3e4655e08ad9e58a7e236b21e78d1d540","impliedFormat":1},{"version":"e344b4a389bb2dfa98f144f3f195387a02b6bdb69deed4a96d16cc283c567778","impliedFormat":1},{"version":"c6cdcd12d577032b84eed1de4d2de2ae343463701a25961b202cff93989439fb","impliedFormat":1},{"version":"3dc633586d48fcd04a4f8acdbf7631b8e4a334632f252d5707e04b299069721e","impliedFormat":1},{"version":"3322858f01c0349ee7968a5ce93a1ca0c154c4692aa8f1721dc5192a9191a168","impliedFormat":1},{"version":"6dde0a77adad4173a49e6de4edd6ef70f5598cbebb5c80d76c111943854636ca","impliedFormat":1},{"version":"09acacae732e3cc67a6415026cfae979ebe900905500147a629837b790a366b3","impliedFormat":1},{"version":"f7b622759e094a3c2e19640e0cb233b21810d2762b3e894ef7f415334125eb22","impliedFormat":1},{"version":"99236ea5c4c583082975823fd19bcce6a44963c5c894e20384bc72e7eccf9b03","impliedFormat":1},{"version":"f6688a02946a3f7490aa9e26d76d1c97a388e42e77388cbab010b69982c86e9e","impliedFormat":1},{"version":"9f642953aba68babd23de41de85d4e97f0c39ef074cb8ab8aa7d55237f62aff6","impliedFormat":1},{"version":"159d95163a0ed369175ae7838fa21a9e9e703de5fdb0f978721293dd403d9f4a","impliedFormat":1},{"version":"2d2ec3235e01474f45a68f28cf826c2f5228b79f7d474d12ca3604cdcfdac80c","impliedFormat":1},{"version":"6dd249868034c0434e170ba6e0451d67a0c98e5a74fd57a7999174ee22a0fa7b","impliedFormat":1},{"version":"9716553c72caf4ff992be810e650707924ec6962f6812bd3fbdb9ac3544fd38f","impliedFormat":1},{"version":"506bc8f4d2d639bebb120e18d3752ddeee11321fd1070ad2ce05612753c628d6","impliedFormat":1},{"version":"053c51bbc32db54be396654ab5ecd03a66118d64102ac9e22e950059bc862a5e","impliedFormat":1},{"version":"1977f62a560f3b0fc824281fd027a97ce06c4b2d47b408f3a439c29f1e9f7e10","impliedFormat":1},{"version":"627570f2487bd8d899dd4f36ecb20fe0eb2f8c379eff297e24caba0c985a6c43","impliedFormat":1},{"version":"0f6e0b1a1deb1ab297103955c8cd3797d18f0f7f7d30048ae73ba7c9fb5a1d89","impliedFormat":1},{"version":"0a051f254f9a16cdde942571baab358018386830fed9bdfff42478e38ba641ce","impliedFormat":1},{"version":"17269f8dfc30c4846ab7d8b5d3c97ac76f50f33de96f996b9bf974d817ed025b","impliedFormat":1},{"version":"9e82194af3a7d314ccbc64bb94bfb62f4bfea047db3422a7f6c5caf2d06540a9","impliedFormat":1},{"version":"083d6f3547ccbf25dfa37b950c50bee6691ed5c42107f038cc324dbca1e173ae","impliedFormat":1},{"version":"952a9eab21103b79b7a6cca8ad970c3872883aa71273f540285cad360c35da40","impliedFormat":1},{"version":"8ba48776335db39e0329018c04486907069f3d7ee06ce8b1a6134b7d745271cc","impliedFormat":1},{"version":"e6d5809e52ed7ef1860d1c483e005d1f71bab36772ef0fd80d5df6db1da0e815","impliedFormat":1},{"version":"893e5cfbae9ed690b75b8b2118b140665e08d182ed8531e1363ec050905e6cb2","impliedFormat":1},{"version":"6ae7c7ada66314a0c3acfbf6f6edf379a12106d8d6a1a15bd35bd803908f2c31","impliedFormat":1},{"version":"e4b1e912737472765e6d2264b8721995f86a463a1225f5e2a27f783ecc013a7b","impliedFormat":1},{"version":"97146bbe9e6b1aab070510a45976faaf37724c747a42d08563aeae7ba0334b4f","impliedFormat":1},{"version":"c40d552bd2a4644b0617ec2f0f1c58618a25d098d2d4aa7c65fb446f3c305b54","impliedFormat":1},{"version":"09e64dea2925f3a0ef972d7c11e7fa75fec4c0824e9383db23eacf17b368532f","impliedFormat":1},{"version":"424ddba00938bb9ae68138f1d03c669f43556fc3e9448ed676866c864ca3f1d6","impliedFormat":1},{"version":"a0fe12181346c8404aab9d9a938360133b770a0c08b75a2fce967d77ca4b543f","impliedFormat":1},{"version":"3cc6eb7935ff45d7628b93bb6aaf1a32e8cb3b24287f9e75694b607484b377b3","impliedFormat":1},{"version":"ced02e78a2e10f89f4d70440d0a8de952a5946623519c54747bc84214d644bac","impliedFormat":1},{"version":"efd463021ccc91579ed8ae62584176baab2cd407c555c69214152480531a2072","impliedFormat":1},{"version":"29647c3b79320cfeecb5862e1f79220e059b26db2be52ea256df9cf9203fb401","impliedFormat":1},{"version":"e8cdefd2dc293cb4866ee8f04368e7001884650bb0f43357c4fe044cc2e1674f","impliedFormat":1},{"version":"582a3578ebba9238eb0c5d30b4d231356d3e8116fea497119920208fb48ccf85","impliedFormat":1},{"version":"185eae4a1e8a54e38f36cd6681cfa54c975a2fc3bc2ba6a39bf8163fac85188d","impliedFormat":1},{"version":"0c0a02625cf59a0c7be595ccc270904042bea523518299b754c705f76d2a6919","impliedFormat":1},{"version":"c44fc1bbdb5d1c8025073cb7c5eab553aa02c069235a1fc4613cd096d578ab80","impliedFormat":1},{"version":"cee72255e129896f0240ceb58c22e207b83d2cc81d8446190d1b4ef9b507ccd6","impliedFormat":1},{"version":"3b54670e11a8d3512f87e46645aa9c83ae93afead4a302299a192ac5458aa586","impliedFormat":1},{"version":"c2fc4d3a130e9dc0e40f7e7d192ef2494a39c37da88b5454c8adf143623e5979","impliedFormat":1},{"version":"2e693158fc1eedba3a5766e032d3620c0e9c8ad0418e4769be8a0f103fdb52cd","impliedFormat":1},{"version":"516275ccf3e66dc391533afd4d326c44dd750345b68bb573fc592e4e4b74545f","impliedFormat":1},{"version":"07c342622568693847f6cb898679402dd19740f815fd43bec996daf24a1e2b85","impliedFormat":1},{"version":"fa40d705f9813843d47f19321591499f14d1a18fa5e8ca9beaee5aac633c3d0d","impliedFormat":1},{"version":"a7a6330fb015f72d821e23004e63a3827e0c632b614ef3a310b3c81b66de61fd","impliedFormat":1},{"version":"89968316b7069339433bd42d53fe56df98b6990783dfe00c9513fb4bd01c2a1c","impliedFormat":1},{"version":"a4096686f982f6977433ee9759ecbef49da29d7e6a5d8278f0fbc7b9f70fce12","impliedFormat":1},{"version":"62e62a477c56cda719013606616dd856cfdc37c60448d0feb53654860d3113bb","impliedFormat":1},{"version":"207c107dd2bd23fa9febac2fe05c7c72cdac02c3f57003ab2e1c6794a6db0c05","impliedFormat":1},{"version":"55133e906c4ddabecdfcbc6a2efd4536a3ac47a8fa0a3fe6d0b918cac882e0d4","impliedFormat":1},{"version":"2147f8d114cf58c05106c3dccea9924d069c69508b5980ed4011d2b648af2ffe","impliedFormat":1},{"version":"2eb4012a758b9a7ba9121951d7c4b9f103fe2fc626f13bec3e29037bb9420dc6","impliedFormat":1},{"version":"fe61f001bd4bd0a374daa75a2ba6d1bb12c849060a607593a3d9a44e6b1df590","impliedFormat":1},{"version":"cfe8221c909ad721b3da6080570553dea2f0e729afbdbcf2c141252cf22f39b5","impliedFormat":1},{"version":"34e89249b6d840032b9acdec61d136877f84f2cd3e3980355b8a18f119809956","impliedFormat":1},{"version":"6f36ff8f8a898184277e7c6e3bf6126f91c7a8b6a841f5b5e6cb415cfc34820e","impliedFormat":1},{"version":"4b6378c9b1b3a2521316c96f5c777e32a1b14d05b034ccd223499e26de8a379c","impliedFormat":1},{"version":"07be5ae9bf5a51f3d98ffcfacf7de2fe4842a7e5016f741e9fad165bb929be93","impliedFormat":1},{"version":"cb1b37eda1afc730d2909a0f62cac4a256276d5e62fea36db1473981a5a65ab1","impliedFormat":1},{"version":"195f855b39c8a6e50eb1f37d8f794fbd98e41199dffbc98bf629506b6def73d7","impliedFormat":1},{"version":"471386a0a7e4eb88c260bdde4c627e634a772bf22f830c4ec1dad823154fd6f5","impliedFormat":1},{"version":"108314a60f3cb2454f2d889c1fb8b3826795399e5d92e87b2918f14d70c01e69","impliedFormat":1},{"version":"d75cc838286d6b1260f0968557cd5f28495d7341c02ac93989fb5096deddfb47","impliedFormat":1},{"version":"d531dc11bb3a8a577bd9ff83e12638098bfc9e0856b25852b91aac70b0887f2a","impliedFormat":1},{"version":"19968b998a2ab7dfd39de0c942fc738b2b610895843fec25477bc393687babd8","impliedFormat":1},{"version":"c0e6319f0839d76beed6e37b45ec4bb80b394d836db308ae9db4dea0fe8a9297","impliedFormat":1},{"version":"1a7b11be5c442dab3f4af9faf20402798fddf1d3c904f7b310f05d91423ba870","impliedFormat":1},{"version":"079d3f1ddcaf6c0ff28cfc7851b0ce79fcd694b3590afa6b8efa6d1656216924","impliedFormat":1},{"version":"2c817fa37b3d2aa72f01ce4d3f93413a7fbdecafe1b9fb7bd7baaa1bbd46eb08","impliedFormat":1},{"version":"682203aed293a0986cc2fccc6321d862742b48d7359118ac8f36b290d28920d2","impliedFormat":1},{"version":"7406d75a4761b34ce126f099eafe6643b929522e9696e5db5043f4e5c74a9e40","impliedFormat":1},{"version":"7e9c4e62351e3af1e5e49e88ebb1384467c9cd7a03c132a3b96842ccdc8045c4","impliedFormat":1},{"version":"ea1f9c60a912065c08e0876bd9500e8fa194738855effb4c7962f1bfb9b1da86","impliedFormat":1},{"version":"903f34c920e699dacbc483780b45d1f1edcb1ebf4b585a999ece78e403bb2db3","impliedFormat":1},{"version":"100ebfd0470433805c43be5ae377b7a15f56b5d7181c314c21789c4fe9789595","impliedFormat":1},{"version":"12533f60d36d03d3cf48d91dc0b1d585f530e4c9818a4d695f672f2901a74a86","impliedFormat":1},{"version":"21d9968dad7a7f021080167d874b718197a60535418e240389d0b651dd8110e7","impliedFormat":1},{"version":"2ef7349b243bce723d67901991d5ad0dfc534da994af61c7c172a99ff599e135","impliedFormat":1},{"version":"fa103f65225a4b42576ae02d17604b02330aea35b8aaf889a8423d38c18fa253","impliedFormat":1},{"version":"1b9173f64a1eaee88fa0c66ab4af8474e3c9741e0b0bd1d83bfca6f0574b6025","impliedFormat":1},{"version":"1b212f0159d984162b3e567678e377f522d7bee4d02ada1cc770549c51087170","impliedFormat":1},{"version":"46bd71615bdf9bfa8499b9cfce52da03507f7140c93866805d04155fa19caa1b","impliedFormat":1},{"version":"86cb49eb242fe19c5572f58624354ffb8743ff0f4522428ebcabc9d54a837c73","impliedFormat":1},{"version":"fc2fb9f11e930479d03430ee5b6588c3788695372b0ab42599f3ec7e78c0f6d5","impliedFormat":1},{"version":"bb1e5cf70d99c277c9f1fe7a216b527dd6bd2f26b307a8ab65d24248fb3319f5","impliedFormat":1},{"version":"817547eacf93922e22570ba411f23e9164544dead83e379c7ae9c1cfc700c2cf","impliedFormat":1},{"version":"a728478cb11ab09a46e664c0782610d7dd5c9db3f9a249f002c92918ca0308f7","impliedFormat":1},{"version":"9e91ef9c3e057d6d9df8bcbfbba0207e83ef9ab98aa302cf9223e81e32fdfe8d","impliedFormat":1},{"version":"66d30ef7f307f95b3f9c4f97e6c1a5e4c462703de03f2f81aca8a1a2f8739dbd","impliedFormat":1},{"version":"293ca178fd6c23ed33050052c6544c9d630f9d3b11d42c36aa86218472129243","impliedFormat":1},{"version":"90a4be0e17ba5824558c38c93894e7f480b3adf5edd1fe04877ab56c56111595","impliedFormat":1},{"version":"fadd55cddab059940934df39ce2689d37110cfe37cc6775f06b0e8decf3092d7","impliedFormat":1},{"version":"91324fe0902334523537221b6c0bef83901761cfd3bd1f140c9036fa6710fa2b","impliedFormat":1},{"version":"b4f3b4e20e2193179481ab325b8bd0871b986e1e8a8ed2961ce020c2dba7c02d","impliedFormat":1},{"version":"41744c67366a0482db029a21f0df4b52cd6f1c85cbc426b981b83b378ccb6e65","impliedFormat":1},{"version":"c3f3cf7561dd31867635c22f3c47c8491af4cfa3758c53e822a136828fc24e5d","impliedFormat":1},{"version":"a88ddea30fae38aa071a43b43205312dc5ff86f9e21d85ba26b14690dc19d95e","impliedFormat":1},{"version":"b5b2d0510e5455234016bbbaba3839ca21adbc715d1b9c3d6dede7d411a28545","impliedFormat":1},{"version":"5515f17f45c6aafe6459afa3318bba040cb466a8d91617041566808a5fd77a44","impliedFormat":1},{"version":"4df1f0c17953b0450aa988c9930061f8861b114e1649e1a16cfd70c5cbdf8d83","impliedFormat":1},{"version":"441104b363d80fe57eb79a50d495e0b7e3ebeb45a5f0d1a4067d71ef75e8fbfa","impliedFormat":1},{"version":"6166402b3448bb8e8908ef0debe182d3d3aa458dc904faf3fb6c8c8e24c75ba5","signature":"b9b7909a64afb56326e22b22dc33a141e54d62440af2817e0a5e00e828e9a5f7"},"96d797d342bdf52d922c24d8bb3b6371e403190c8a013c360fd5f4996c6a1021",{"version":"c1a356c4d1f84001e7a8b2ac0e84a37b9e15ab64fa13da2c26460f853585db55","signature":"2e1ff43e4192d9c2aa6ce2674fe220c124134d08805696504d222b0b761c8440"},{"version":"c7074b8eae9a4471112877efb9fccc4ff50d400637b54034f4c697369b8e0c2f","impliedFormat":1},{"version":"000862b3dbaa3ba8112440086eebcf916f578f48f918882f545596c687272177","impliedFormat":1},{"version":"20c1111aa9534c7b7df1ddee02011a3d7067a791c4d5901dced9830fd8d486f3","impliedFormat":1},{"version":"7d3ecc412ca6fb2140ee8718a3d0f15c2cf2990b5891e6e1a40205ef7a4ec96a","impliedFormat":1},{"version":"07ca9984d5e03a45cffbb4c6a06178180667216450d8348a307323d0998d4bf4","impliedFormat":1},{"version":"1910301ed7c3c39bc34fc9715414b5355e4fee0401ae65e3e43ebf012bb1a63b","impliedFormat":1},{"version":"d477e2b38d400c14b60c07c0994601e129c5f17c7f958117601b63efa8e8017b","impliedFormat":1},{"version":"4182411ed303d53edbfe1bb80781f90af30c666fd957989037141bea45431668","impliedFormat":1},{"version":"9995b9787d6034dfe7d1bc6f271dd8bb21fd1d1903a08a62434f161682dcb457","impliedFormat":1},{"version":"66971fbc40a80f916b8ca2e94d5d6c3e35994d11065d0eb03ec41b370adfed97","impliedFormat":1},{"version":"3bc0e49ff8fdad5206ee15a435a7adc73bd293e2097ecc676fd4e0f5eafeee81","impliedFormat":1},{"version":"72f8a4bdb822b85e3cb6397971c39f277c70132b161bb12bebd94b163e2f73ae","impliedFormat":1},{"version":"ab75746ac80c9a08def860982af72a6f1038351901ea8603e71b3e8707668ad1","impliedFormat":1},{"version":"fec0d866732b6c774a60394b5a6d33dac1af6daa3fea1e53ae5975d599180a85","impliedFormat":1},{"version":"b9c5772b895e937641d51d4c2110751a60f12b760f67bcf4f36a8a2477aa833a","impliedFormat":1},{"version":"01799fc79b10be8bafc36e1c7b15f4bae98bb2da8100c040e83e1471ef37fa2e","impliedFormat":1},{"version":"ff49baf4336cacc5ea8646e723c8e4a3bdfbb9ed2f34c20e8c2de1d05b670b20","impliedFormat":1},{"version":"bcabb3091911455814cece1c2817e30f89cf4946f2da15b43472b9a885a56d55","impliedFormat":1},{"version":"4151431363ca6cca001810e3d416b4bee0a562330d26499f802764b2b51aca9e","impliedFormat":1},{"version":"e1ea7b24ce79efe8d19c880a1b4e004a06d6292f459ac153654e24d1ae87d9fc","impliedFormat":1},{"version":"825689d3a15da56a7f985aee87810c3392fb598646ba45cfdb0cf5cb58e9d130","impliedFormat":1},{"version":"49c0281e4c32f99df95722a8c10ba2703631c696c3b3330fc5feb9a9bd0f4eaa","impliedFormat":1},{"version":"b1f64c6aad9676f268b9114c4cf4a5cf4bb29abe936ab73f1c43f5f7dda28f2c","impliedFormat":1},{"version":"ad40c811757b8870312ce3da99ca4b06c6604c0d8139a3189f68c5314aff5fc4","impliedFormat":1},{"version":"e5c3362ea58b105f4468e63f9b3f746fcdb5714024931231ce9b1d59b74c6d67","impliedFormat":1},{"version":"2775b4acaa4ff841a96eeb3e477cb82c84b4c7e3c157acc18901cbe4edd06b77","impliedFormat":1},{"version":"bc0b40ef267631af97290073624c27a6722124d50d9699e29c1f8507ecb6a282","impliedFormat":1},{"version":"1ae6757ae8f0331b54b66381fe7e7cba2837f3681afb9278bf9f163bd5039466","impliedFormat":1},{"version":"b1229442187bea202ab606cd64cf583692e0c57581784dcb21667c7b1122eea1","impliedFormat":1},{"version":"092042bc75b55f727ed62a4e2579c1f010e9d404d6fcf1954473a01d9e94484b","impliedFormat":1},{"version":"38c77a69740b7655457c36f6d115182693f3ab677949682de677fa2dfb2599bb","impliedFormat":1},{"version":"f48592a06d340ee95784a3242b75710000d5e4a467958ed3b2ee8f2ece278b44","impliedFormat":1},{"version":"6ed0949ed569a5b26fe0b988267a6b93be0e525a2ab6098ce8fee2436806c918","impliedFormat":1},{"version":"91305a029dfab85b854e6d31697dc43b0657eb7536aacaa65c6d74da5b437857","impliedFormat":1},{"version":"cfa2d15f468650ea455799c4ebb4d0698fa8fe4665bc73a46d0b0c443e5bc5e3","impliedFormat":1},{"version":"64dec872f5228ebc07812a65af8b1269a145aa6d1fa6f47546abce9a3a15fd3c","impliedFormat":1},{"version":"940c0de32ddda6212e5543177fd063498f6cf422911c6bff0a2086cbb11f5ecf","impliedFormat":1},{"version":"8f13981832313611bfcf90fd96de1218744f759a0175914eaf406ffd4919cbaa","impliedFormat":1},{"version":"dcbc04cc196ef0ed45d4fb8925e867cb463d4db787d9039ecabb737fcf12ab79","impliedFormat":1},{"version":"7e61f20558c56cf82160c3edacc548312a35b1c7f11704f766746ef64f0e4c74","impliedFormat":1},{"version":"c3256abdaaf40e7e73c7c24ec2fba77569a8f03a096b2f9582fe7520d7fff1d1","impliedFormat":1},{"version":"728d7b16a89022b6a86d1ff1e4791c39a4401acc869d22a87baeb37b323036c6","impliedFormat":1},{"version":"f1f6923ec959a00397a2dd72864d9065d823d7aa078d9840af6b78c84b500146","impliedFormat":1},{"version":"f957b495c6b8f1f14a15661e849c02a8def9dc35840e8d2ddc2046bbe14bd547","impliedFormat":1},{"version":"21f98579035b0a0bfbb28996ad331c5aa79302624779c01bc077257c9d91c114","impliedFormat":1},{"version":"7a89725f1817ff1e6cb6b167ad84df1fc851193c66719ce79bc076043eddf118","impliedFormat":1},{"version":"81bb7e34312099a7d367187d7732afb3b21d5210c44a79db857e3920d5e148b6","impliedFormat":1},{"version":"1fe171ffb872e40a0a9a642ba529e87b597b1d67931878c33ac4d1bbb1df58be","impliedFormat":1},{"version":"5b1186921f801ae8e283eb2249ef6e27ae9f8a1834e57f2e16949cd3991846d6","impliedFormat":1},{"version":"b41fd4b7e7414a10054c8a498b0f452c429e63a5cfc621606f58752dccbe85fd","impliedFormat":1},{"version":"37ce26a13df5aede9c6b795ac7a26505ea91874a30b165307e2f264efdfb6511","impliedFormat":1},{"version":"d998bca69f80fc1f04757aa1cfc79ab2e1db7bba0ef4a30869156e3357386fc3","impliedFormat":1},{"version":"1c812a7d6565b034096fa5b5630f3e7159c0c289f525e01b3718f337a2cdeca1","impliedFormat":1},{"version":"c6d2376be1c9805e0ef5de09250553dabbabd67e508d5e922ac5d0902c98aec5","impliedFormat":1},{"version":"e2f44b8f481c2298220d49a59f91a8f33cf0c7075006b401010eb5b73c560453","impliedFormat":1},{"version":"377bb18a79fd85c00b731bb60b33e22a8984b4f97ba6825d5a4564a801038e7a","impliedFormat":1},{"version":"db382b318bc333eb7548cc1f0dcd149794d7e1d4969d54f9586f2533b287b5ac","impliedFormat":1},{"version":"ace39b5aee44e612eadcdf34d3d1f99ae5ff0a175a3fa8903fabb0953570947e","impliedFormat":1},{"version":"570801b10794486f9535c6cb55658aade8a2fa34d6e320062f8c704d49042818","impliedFormat":1},{"version":"0d6fee47fa4c30b31233fd5c6a69b238ee9944a680c69997c7c4093fe3bd6367","impliedFormat":1},{"version":"8bba3c4a509f012d471e1c4973be2b5ddbf81ae6347c99ca0b067fa6e3ba8b17","impliedFormat":1},{"version":"bda138ca76c8291b66ffc74c10f74a8152488da3a61e577bc1d0de329d0502cf","impliedFormat":1},{"version":"de6352a242e56c4df1829349aa0ae1f6e5fafb1326581f7bd1fd9eb3cf5772c2","impliedFormat":1},{"version":"48ff1dd366aef7a6d48201d4e830e99177cf115fa22f5576f0c899fe75280f5a","impliedFormat":1},{"version":"263b4d2898ad1b86569ca63585374d6c42685ae1048c52ffe964d1f62eb9a6f1","impliedFormat":1},{"version":"e21f31c0d0e10034621862f5d37868fc9c1d8e05619b16971f7cd3110f9b1be6","impliedFormat":1},{"version":"025ab7b7c805c3e4a5e4bd3f76cb8c092bec146a3240d7004f78a8b25c467e52","impliedFormat":1},{"version":"41aa1ef57640c8f90093237fff863a702317ef22e962cf2047460d59d3acadd8","impliedFormat":1},{"version":"c1c8a6cde0253f15cfa13429e43f10cc3944428150a308bb40cdb75c9565f784","impliedFormat":1},{"version":"7a1e098bf3c979054761df1cb1e3d09463c734977629576df5ae2be4e4aadb99","impliedFormat":1},{"version":"ebd61ac5f4d954aeb24e1157027a8dd7252cc2f03f733e864f8e38ac5bfac306","impliedFormat":1},{"version":"092c372db6eca705f0c4a3a265e8370e1cb5992c96e4bc8a42e8a71d13ac5d8a","impliedFormat":1},{"version":"e69ef73426bca8394b75d76b7bb582666fbf8fafe2f37d0ee8f95cddbcebca68","impliedFormat":1},{"version":"4a347a72f5f2de8132f13ff85c4207022898783be27e4823c8d811840ae4afe8","impliedFormat":1},{"version":"292c9cd91e57d749d8b83b0002b88fc6e47ae5c706659383ebdd9181ff302abc","impliedFormat":1},{"version":"707e26f766b6f7403f244e7311923a928dde53750a535bdd9b4ea7a9c43cd13b","impliedFormat":1},{"version":"7fe6758ebcb7aa7c040ada95a40149f4f934b5c43b3334058ca7b098b053af04","impliedFormat":1},{"version":"d2170a91c800517e2e768a8142e5bcfb145e281fe442c59b0378ea70ae2ac873","signature":"b10f6818e1533f4fdf3330b321877cf140f371d180496dfd092c241169f0e8a2"},{"version":"3390390f7e2004f5b17eaf402cd25d9725b9046245b2c5db4a7ecb032a177f0b","signature":"8d12a79e2bd55062f4fc71aa02e3a4ffb7da08fea04471091db74657de8833e3"},{"version":"d0f7bf6091eb170565a694c933b982c8feee9b3e0f0f6daa9b74fc55853171ac","signature":"e0fb7c7d3ff4db0511c1f6de083c3190c349033406b23b2c50972ea4ebadb558"},{"version":"3cc2ce8a22745ea3098e0afa83adbcd69fe55a664090faf9b0f8c5b7d0286700","signature":"b51a7aefe9864043882e7d2a4f64ab843083e9d4f09f5c0535ce00e7fe61aa32"},{"version":"7a0a0df6d27275a54b43252767f0eafde9a42ba6bced5ccde55566b1ad35bdcb","signature":"07f75d64a48f8d298da7a80c8d2d4d0bd7b9140b7498e9149746634438781ca4"},{"version":"f0ee6c44f15f0eea733fc3884d69586a20c385fcd37f5d95cb6123596f81f86d","signature":"42dc1dcc5b1f926450d6f8cc760ead9fb2fb5516a3b5cc04e8ee3ee2632bb63a"},{"version":"01fa15530f4248f35df05ef245abab27819c8d57f12f9bd8627926b90c8b8695","signature":"2a66f568377421e826fe7369bd31c6c4f19fbfe0e0d74a6ea0c637e0af73fdc9"},{"version":"689a8e4611f8cd55f62f7c815cf71c73cbb9a993d456ce31ed33566ea3503d8a","signature":"4f0ddd40c3d99eceb92dc9564035be9b6d75429ead9a11fc021ecb5f107c9fc4"},{"version":"ce7fb8867cd1fe888d90d81beb4ef350dbbce3617dbcbd9937f1c44ccc8c7056","signature":"bed9ff33e9577b6a11a3f519dc44430885ebc206b737e0378cd9a4fa0d767294"},{"version":"bee525e6e98ad59176798002e90c3be9665ae375ccf307840eba8dedb0da1e75","signature":"4e7ce323033347cc4daf6d2d471ae628e15bbf162e4eecc32fcc79a59b551de9"},{"version":"200ecb580e5cbfb414a5c87bca1258c7755fa4781fb18cca74c35b5675dda5c8","signature":"9418b1357b1bf8a884c423ba50671bbc80aca4adef147240d880a114ecba68ae"},{"version":"8b8d00903f1f9ca0ec93ecd5de1f6401ddaca394c3dcdd7efbe871d4258314c9","signature":"ab02bc38697b1b2df6adc45d777f93f6230cae1e8ff57cfd7eec0c86e603ee91"},{"version":"fa58002945d63bb92e23391245a3be28cbbe036b41b0f701652641409fe72dfa","signature":"7df5e90c41177a35aefaf2a6d1cedbee88a0068d68993303f558bf4c704b208e"},{"version":"c12609f25a35eda6db8e8a2a81a1d8b6d1b902c4c2666f9b363c8ca5e19d349f","signature":"d42e16ef6ec54d23465eb7fa515ae912a7bf1aa7ed5860398ef631e31f9c96de"},{"version":"6e5f4d921fff02c48b132b01df811805f3d4c99cb0d0420d35015fa32ca17d1a","signature":"2230f19c4871a5ef67ed59cff4891f286a3cfc45dc46d0d00f965d13be378fc6"},{"version":"ce7d79da26b858bb29a8c4f2cad8fa64ce123c28a561bd7356bb391dc135b753","impliedFormat":1},{"version":"6aac2c5ca00378e4d1421a03f614643dc1e9fd02279257cbf2e8e2a713b00907","impliedFormat":1},{"version":"254510b0a3c2e04f55e98ae89a6aa42f67852c192c3502b3b8488e578b21c9d6","impliedFormat":1},{"version":"b75be7355591118207e7f24143b27a860da4043a1950c746e034313d9ded4137","impliedFormat":1},{"version":"da15f699f56ab6a37b4eca73eb14a356f5d175d979f0c8197d325d5f23c91bd6","impliedFormat":1},{"version":"9b2fe69aa18037c1f47191b263016cc3edee57afb6660b1ae37c0b491aa4190c","impliedFormat":1},{"version":"d8c3b3c16a4a8656dcdd394df0df07d3149816cb96a89935d62cafe4dd84009a","impliedFormat":1},{"version":"e982879e6ea8ddf8899f637e639bc225996a729e07f068afb120d32fb4feebf2","impliedFormat":1},{"version":"94616e40e31224cb261a78c5cb96fd3f65f9ead7052eac20fc6c975714f3840c","impliedFormat":1},{"version":"931574e125523649902eee2db57c221a1b36417db4f2c4665bf38ce2170ea06e","impliedFormat":1},{"version":"cd0c8c8b5002ec4cac9e8a5e26d853549c5c446a670fb375b9c052b345fb5da1","impliedFormat":1},{"version":"7d27796c034612b6016db97555b84f1005dc3d55e2286379d48ec8db475b6430","impliedFormat":1},{"version":"0d59de214eefc455e13a7f747c011729ee76f1554fdef55554ecf4bfeb20568b","impliedFormat":1},{"version":"e16ecf37f6f2ca79ff19ba2e4c3697ecd9d38b8d01bf6682bc4003d0d5719651","impliedFormat":1},{"version":"845154327584247966f7dea7a3e4960906b7038cbe23ab43fb198539ca12204f","impliedFormat":1},{"version":"cce34c68dd760a55d002eaa02390985f4aeaa39786679f54ade28be6229792e9","impliedFormat":1},{"version":"877388f59a044fc4c4689637425d4f8762662b4c6dc86d55864ca8816382b69e","impliedFormat":1},{"version":"162ffbed80dad8ce0cf81c330c88dccaae85425fb457a6afcae0110419bdedfb","impliedFormat":1},{"version":"a85d6e7924c263fdb7a9e28a578401f2f96950ff9fd0e250c76f25de5ce3b9f2","impliedFormat":1},{"version":"b42b1904665d7ead2610cbadddb80163a637ad2d202ff446af2382bf75b38353","impliedFormat":1},{"version":"57947d16b34a3811f854965fe668e81ccea9dd6321e412ea1a2c75d4fd2619c1","impliedFormat":1},{"version":"e9d4bfe42849ba995ab572beba5f30bd484e88f9441a4eb223a54ddec0c4d490","impliedFormat":1},{"version":"6c36e6a90dd3e1f5e488c56aede7168ea429589d0095285701b7b56f47262bbd","impliedFormat":1},{"version":"4bc4c7612f5cc6298b01f76f7a21674181ae6e199a0b07c518107c15bde32344","impliedFormat":1},{"version":"c073f994576c9c9cfcf7a787384c51a81aa69ebfe61dd9a62d87d020707ea45c","signature":"cdd28eb0387c77b7c89e1a5da0b5d6ad503e419b0114b9c50e86a824cb04a583"},{"version":"569e762cf47aafdad508360a443c6c757e56c61db3b652b65458a7d168d139c4","impliedFormat":1},{"version":"02ed2766d79a00719ac3cc77851d54bd7197c1b12085ea12126bc2a65068223e","impliedFormat":1},{"version":"4b84373e192b7e0f8569b65eb16857098a6ee279b75d49223db2a751fdd7efde","impliedFormat":1},{"version":"5aeea312cd1d3cc5d72fc8a9c964439d771bdf41d9cce46667471b896b997473","impliedFormat":1},{"version":"cfa7bf135cafc5aad7cf544bc1cebf65a1fdb4373223cc85ea7d7196e18be151","impliedFormat":1},{"version":"1ba59c8bbeed2cb75b239bb12041582fa3e8ef32f8d0bd0ec802e38442d3f317","impliedFormat":1},{"version":"f2c4a36eb216aadb0d9c79862a31b922ccfa1eaaa38d2124cc9192d40eda4779","impliedFormat":1},{"version":"cb5bb1db16ff4b534f56f7741e7ffd0a007ce36d387a377d4c196036e0932423","impliedFormat":1},{"version":"25be1eb939c9c63242c7a45446edb20c40541da967f43f1aa6a00ed53c0552db","impliedFormat":1},{"version":"08c2bb524b8ed271f194e1c7cc6ad0bcc773f596c41f68a207d0ec02c9727060","impliedFormat":1},{"version":"012b69bc8a16a21aa0863502339c49258c579723f9e7a54faa5f0d5c2b1ae1b7","impliedFormat":1},{"version":"29ad73d9e365d7b046f3168c6a510477bfe30d84a71cd7eb2f0e555b1d63f5f6","impliedFormat":1},{"version":"d99e9f5aa43397599fe824e38c33d13d3a9e19198806a4363114bd7ac58b29cc","impliedFormat":1},{"version":"440099416057789b14f85af057d4924915f27043399c10d4ca67409d94b963cf","impliedFormat":1},{"version":"ac44995fc7d0781d77927bae7dd41a31f0309e695fd2694b175c0ce3bc4b3b50","impliedFormat":1},{"version":"0c1f802f7a60ca8084e5188ac7952accdfc00f39ded3ebbbd3cdcc9da51b9a7b","impliedFormat":1},{"version":"a32e3fc530d8d1a18bf54678d8d55714827a50c9fabdd4ede7155a56be7d1dcb","impliedFormat":1},{"version":"14ecfc29e0c44ad4c5e50f9b597492cd8f45a2a635db8b5fe911a5da83e26cf8","impliedFormat":1},{"version":"ba0badf6398010ef0ba77947d4b52078fc84cb4f209f6b70194b3b2cfb1823d8","impliedFormat":1},{"version":"c2f041fe0e7ae2d5a19c477d19e8ec13de3d65ef45e442fa081cf6098cdcbe2d","impliedFormat":1},{"version":"0cef678147928ef223ff7f2aae3442cc9f4e9996956e9ac92434e626d0e147f8","impliedFormat":1},{"version":"198ae766bb698feb66d3188cfce59fb33696c951b10f901aa3fc3db4847ce76a","impliedFormat":1},{"version":"6dc488fd3d01e4269f0492b3e0ee7961eec79f4fc3ae997c7d28cde0572dbd91","impliedFormat":1},{"version":"a09b706f16bda9372761bd70cf59814b6f0a0c2970d62a5b2976e2fd157b920f","impliedFormat":1},{"version":"70da4bfde55d1ec74e3aa7635eae741f81ced44d3c344e2d299e677404570ca9","impliedFormat":1},{"version":"bf4f6b0d2ae8d11dc940c20891f9a4a558be906a530b9d9a8ff1032afa1962cd","impliedFormat":1},{"version":"9975431639f84750a914333bd3bfa9af47f86f54edbaa975617f196482cfee31","impliedFormat":1},{"version":"70a5cb56f988602271e772c65cb6735039148d5e90a4c270e5806f59fc51d3a0","impliedFormat":1},{"version":"e083384623f90cfa7e8d2aa7efe78c51095a04ad51d1f82c3e4052689666895d","impliedFormat":1},{"version":"1c0d854224f56d51f620c83193d91af7ebd326f710e5059bbfb156e908a74d94","signature":"e1c5c93bd7208db924d21f30378c98ca4027dc034706481843a201628186f46c"},{"version":"63af2c5e3b2fd9173c6a6d00593c5d010f8fab101a69213e9b843fbf76aae2b3","signature":"227319ac3b78de9cc11a6ef41bce8ab1d61d163845ed910b111e3797c7a2396d"},{"version":"2780b432b738c48860733ea660829559deb08600a001544a417df9e5d23bc4d8","signature":"d66855eb19987f9646998b09b84aa36bea9a13abbfc89554077b4a42eec9b30a"},{"version":"2f126b860286d4bcd799954d29d74214c6ab0d24b26971b0a96b33d6034569f3","signature":"31e7242c81c4e74d965d5d25d0db3f3b68fac21c4d74c66315ee9e8560a08114"},{"version":"661d4fe8d21a58cb22f0e6719da3f0ffcefd3e3355ff5f1a0a94e120a73f9f26","signature":"9289ce714eda5db5e3b76d3c8f89239793f4345754ca9741fa37180ab666db2b"},{"version":"ca15e4ebeb9d10974a0549e44cf1d6705e0d47a2c84954576c9b82692765bae6","signature":"09a438dda9993997009703ed00847af26c0eb1b09dffd17ea661c508f91e2b5a"},{"version":"9975e468c4a9ab18ee53a291d3b0cf1f5f69bb0d240e4c6e03eacc141c4a07e8","signature":"ee9b50bef0cdbd027497faca438c404542e9007fa7ee7a5973b49d72afd40b49"},{"version":"2f85a38b365753094225876d5985c5a99470c157a06e918e27913e3d2fa44bd9","signature":"4feb4b7aa8b6183ac1e350bcc28c9105a450b3cff928522ff5a6ed66888fc043"},{"version":"08c7a9ec52ff9a2a870617ca3292e2df51387e890c634feb0b8ee447a1ca4092","signature":"2ceb2bd76fe1961a5713fce82c31820aaad04e75dee63272cd62fe858e1ccfb7"},{"version":"5f189ac4d93d5bb73395adbd17fe107fa0316ac895dcf61fd0e9a81a11a24b61","signature":"caed67a82a8fadd14b201f02d12b59da59586dd55c667e771f69a5de29f5bdfe"},{"version":"117c9aee8e71ce1bce05b0b171e7e039b3a0c87704419d055f46790daf2e841b","signature":"d4968b4fbc46998f7c1a56dff0885c6103a62abc6332fabbaa8afc1adb92faf3"},{"version":"50daf5fe57739af35b761a7976092ebddc5a8f278f1f4f8173aa8392a4162eab","signature":"92292896582943434e53231f9529cb9f3b01b01732a4b618b82226f11226c5e4"},{"version":"2783ae75618ea94829e0c4f1827206c8a024bc1c85fc7aca21bbb23ccc544b39","signature":"d2f5d436caa18b265f2a065aebb0bebca71be7651f98e155249173b6dc3eb464"},{"version":"b6f84e82e3b6e063b0b92bd1de306e6fcbfbd14295b4fb19f1a54ede740f4761","signature":"99b32e4a74812424e54bf7fbc908c4a72f1d9cd192d400f5b8a1a4340515f15a"},{"version":"a4690b1fa33c34e87ff58e475ef1ce26140085527b420245b290dec278fb012b","signature":"d788700590156040d9de08b08f71fddc1ab97d7a27c8b003fd088848aec6f290"},{"version":"85c2fda93c8b89a59b0c94c3ec30032d6673f8766a4af04f0ea180f49d9547f9","signature":"c56e063539bd4c3f504a35b9623e5b2ebeb76cd0b3d93ade7d5f5f4cb37dec83"},{"version":"753b24837b64c5616b68bc9ace0ed7e3e5fc84a80603e2d9db50bf05ba5c769c","signature":"83db9b5a0b56d29bbd1d827dd500c0a3b648584f7d3cf88157ee1ac82f913ee8"},{"version":"fe4d6200da0d487c4567fc6dedcce1bb6efd24ff06d8d187e32f58743f7d32e8","signature":"efcd3c8714843ae23210c8a478fc67f67e877fb664ba2c69ff9a6aee64e00bc1"},{"version":"9586a0162d509fff5543dd3f6838e32fe98db3c67d3fe019670713404a6a017d","signature":"eb31e078586175214834c5cdd39a59a049278cb60c7b3912d22fd360012d9c61"},{"version":"673835fa53727f3e0ae0446b621a7eba754269d4e3d87823b13b6682aa1ab3cd","signature":"f672778829e0ba769cf73a0f6bcb84a1f6d0e80f0c51f376099bd13313ebc194"},"8c8b571592dd67ab31fd6dfbb388775873ff9ac8f2e0f001bc33a7824eb89756",{"version":"dd09d58f4e9b93f7cb06d90e7b827ef7060fc2f29c2c8bc9b715134327f5c103","signature":"86b555cc5ed263bfacfa9433336c3cd2714d1e71a000d9d6b0277e3844a9fabe"},{"version":"b6bbf5944c432ebb2e6a42c5206c172ed812465ce944eec2be6967d3878b0496","signature":"928d5c4be7aded223f282b5f7b6f232aad3444869fbc86e98ffbd1cea787056f"},{"version":"332c48b2e891f9a626bf76cf86c0238c8d297b71ede32e0bf812647a0e696410","signature":"caed67a82a8fadd14b201f02d12b59da59586dd55c667e771f69a5de29f5bdfe"},{"version":"ad30950e3739dcf4198b1471109971309cbc84c5f28a60f964c40978a4d3ce59","signature":"5d4f617942b9a20b543592e12e93b78ddff4dd5bf005d98074e0364784349b0d"},{"version":"e50dfd97b222c8c2303fa196dd3e994062d93d843187add88091bae74efbc2cf","signature":"cd72e04377277d979898bb041b35238fa4aba442d5761892584f2fc59dfc74ef"},{"version":"8634d291c7060d893de8f070465d71277680b89eab97304015f1532a7832acfc","signature":"cc84606838ce878e96cf4148cd5e6cf1da9055bac79f8418d7e7f8c3ef04f338"},{"version":"b6e995b5ef6661f5636ff738e67e4ec90150768ef119ad74b473c404304408a1","impliedFormat":1},{"version":"5d470930bf6142d7cbda81c157869024527dc7911ba55d90b8387ef6e1585aa1","impliedFormat":1},{"version":"074483fdbf20b30bd450e54e6892e96ea093430c313e61be5fdfe51588baa2d6","impliedFormat":1},{"version":"b7e6a6a3495301360edb9e1474702db73d18be7803b3f5c6c05571212acccd16","impliedFormat":1},{"version":"aa7527285c94043f21baf6e337bc60a92c20b6efaa90859473f6476954ac5f79","impliedFormat":1},{"version":"dd3be6d9dcd79e46d192175a756546630f2dc89dab28073823c936557b977f26","impliedFormat":1},{"version":"8d0566152618a1da6536c75a5659c139522d67c63a9ae27e8228d76ab0420584","impliedFormat":1},{"version":"ba06bf784edafe0db0e2bd1f6ecf3465b81f6b1819871bf190a0e0137b5b7f18","impliedFormat":1},{"version":"a0500233cb989bcb78f5f1a81f51eabc06b5c39e3042c560a7489f022f1f55a3","impliedFormat":1},{"version":"220508b3fb6b773f49d8fb0765b04f90ef15caacf0f3d260e3412ed38f71ef09","impliedFormat":1},{"version":"1ad113089ad5c188fec4c9a339cb53d1bcbb65682407d6937557bb23a6e1d4e5","impliedFormat":1},{"version":"e56427c055602078cbf0e58e815960541136388f4fc62554813575508def98b6","impliedFormat":1},{"version":"1f58b0676a80db38df1ce19d15360c20ce9e983b35298a5d0b4aa4eb4fb67e0f","impliedFormat":1},{"version":"3d67e7eb73c6955ee27f1d845cae88923f75c8b0830d4b5440eea2339958e8ec","impliedFormat":1},{"version":"11fec302d58b56033ab07290a3abc29e9908e29d504db9468544b15c4cd7670d","impliedFormat":1},{"version":"c66d6817c931633650edf19a8644eea61aeeb84190c7219911cefa8ddea8bd9a","impliedFormat":1},{"version":"ab1359707e4fc610c5f37f1488063af65cda3badca6b692d44b95e8380e0f6c2","impliedFormat":1},{"version":"37deda160549729287645b3769cf126b0a17e7e2218737352676705a01d5957e","impliedFormat":1},{"version":"d80ffdd55e7f4bc69cde66933582b8592d3736d3b0d1d8cc63995a7b2bcca579","impliedFormat":1},{"version":"c9b71952b2178e8737b63079dba30e1b29872240b122905cbaba756cb60b32f5","impliedFormat":1},{"version":"b596585338b0d870f0e19e6b6bcbf024f76328f2c4f4e59745714e38ee9b0582","impliedFormat":1},{"version":"e6717fc103dfa1635947bf2b41161b5e4f2fabbcaf555754cc1b4340ec4ca587","impliedFormat":1},{"version":"c36186d7bdf1f525b7685ee5bf639e4b157b1e803a70c25f234d4762496f771f","impliedFormat":1},{"version":"026726932a4964341ab8544f12b912c8dfaa388d2936b71cc3eca0cffb49cc1d","impliedFormat":1},{"version":"83188d037c81bd27076218934ba9e1742ddb69cd8cc64cdb8a554078de38eb12","impliedFormat":1},{"version":"7d82f2d6a89f07c46c7e3e9071ab890124f95931d9c999ba8f865fa6ef6cbf72","impliedFormat":1},{"version":"4fc523037d14d9bb6ddb586621a93dd05b6c6d8d59919a40c436ca3ac29d9716","impliedFormat":1},{"version":"f543114a5a6c890be15de0080d5ee0ef3697303fa4cc8b91010d30235c6ae4ae","signature":"bbb74701f3b0b790049e875dc0a7ea6e3327428f53fd4b9ad82da7d287332b8b"},{"version":"8fd19af367a738b6765d0a05516cf75db4f37c377956ba467d53ebfdbf195ef9","signature":"d35398f3fbbd4371477bffcff1a535da04f8bccc4445e85ea4f63fd642f538e0"},{"version":"81771c895eef0af8a51220cd1943abf6ce76887164809941a82b3029734cacea","signature":"8879d9b58a1119292d5056255f2e749cd06d0712290e4d4180d7dd10af2ee8ab"},{"version":"ec583fe004266e4249694febd44b0f69fff41c17f9ff815a97dd07127f682d52","signature":"53b3b36c0f65690f7588cba027d7a9c25bb2e57415e7f799ae2068f791ee27c8"},{"version":"f486fba867595c66699a055209677a64ced9c6d400df6ca2d297478ff5eed5f6","signature":"6cf3e45422ac36ab5681670af193997e39568ca211fcc6781e45c725531cac97"},{"version":"6755f929b0187afc1fce8780349859b3ec134a43fac2b19353e2a8ebaeafc2b1","signature":"4966999c95fe3894ddc26f2d0df91fe3e91a8030059680e7ebc49ef53c678ed8"},{"version":"9b50ee091a0f3aa69aa51ab09233b0e834772a5e537dd1c85fc9963256d82d17","impliedFormat":1},{"version":"46acc28f4b194c3cc7d1a7d79310ea91925c449cb272aa820628a8609dd0a447","impliedFormat":1},{"version":"78647004e18e4c16b8a2e8345fca9267573d1c5a29e11ddfee71858fd077ef6e","impliedFormat":1},{"version":"0804044cd0488cb7212ddbc1d0f8e1a5bd32970335dbfc613052304a1b0318f9","impliedFormat":1},{"version":"b725acb041d2a18fde8f46c48a1408418489c4aa222f559b1ef47bf267cb4be0","impliedFormat":1},{"version":"85084ae98c1d319e38ef99b1216d3372a9afd7a368022c01c3351b339d52cb58","impliedFormat":1},{"version":"898ec2410fae172e0a9416448b0838bed286322a5c0c8959e8e39400cd4c5697","impliedFormat":1},{"version":"692345a43bac37c507fa7065c554258435ab821bbe4fb44b513a70063e932b45","impliedFormat":1},{"version":"7283aaea55499e5a9760db0ef199e3c12a35ff5391733fe1495b2d3add0c58ab","impliedFormat":1},{"version":"c1c8ccb14c76efb31ff84038ec7833a5715ba23e681b158b3c83cc012b8c3cfa","impliedFormat":1},{"version":"abb0a41fa0432cdec9595dbe84bd3efc5b59f04aa85dcd5cec7b453908fc85c3","impliedFormat":1},{"version":"522edc786ed48304671b935cf7d3ed63acc6636ab9888c6e130b97a6aea92b46","impliedFormat":1},{"version":"a9607a8f1ce7582dbeebc0816897925bf9b307cc05235e582b272a48364f8aa0","impliedFormat":1},{"version":"de21641eb8edcbc08dd0db4ee70eea907cd07fe72267340b5571c92647f10a77","impliedFormat":1},{"version":"48af3609dc95fa62c22c8ec047530daf1776504524d284d2c3f9c163725bdbd4","impliedFormat":1},{"version":"6758f7b72fa4d38f4f4b865516d3d031795c947a45cc24f2cfba43c91446d678","impliedFormat":1},{"version":"1fefab6dc739d33b7cb3fd08cd9d35dd279fcd7746965e200500b1a44d32db9e","impliedFormat":1},{"version":"997b94a03707d35abe497e427bc26b403a538838c3a82f2be71d85109b88e32b","impliedFormat":1},{"version":"bdf7abbd7df4f29b3e0728684c790e80590b69d92ed8d3bf8e66d4bd713941fe","impliedFormat":1},{"version":"8decb32fc5d44b403b46c3bb4741188df4fbc3c66d6c65669000c5c9cd506523","impliedFormat":1},{"version":"4beaf337ee755b8c6115ff8a17e22ceab986b588722a52c776b8834af64e0f38","impliedFormat":1},{"version":"c26dd198f2793bbdcc55103823a2767d6223a7fdb92486c18b86deaf63208354","impliedFormat":1},{"version":"93551b302a808f226f0846ad8012354f2d53d6dedc33b540d6ca69836781a574","impliedFormat":1},{"version":"040cb635dff5fc934413fa211d3a982122bf0e46acae9f7a369c61811f277047","impliedFormat":1},{"version":"778b684ebc6b006fcffeab77d25b34bf6e400100e0ec0c76056e165c6399ab05","impliedFormat":1},{"version":"285d50a08440314f7aea3246a5e15acbc38e2867ff07d21ef457ae8cb4e8a31f","impliedFormat":1},{"version":"672701f824ff6b9dab50514c6c4db711fb7ec5e7c2f0f6bc27b2dbe2e445a52a","impliedFormat":1},{"version":"be8f369f8d7e887eab87a3e4e41f1afcf61bf06056801383152aa83bda1f6a72","impliedFormat":1},{"version":"352bfb5f3a9d8a9c2464ad2dc0b2dc56a8212650a541fb550739c286dd341de1","impliedFormat":1},{"version":"ebef680e3597d7b3c8a9fc9e5581eb078461fa1406ded8d9859353dd6286eff2","impliedFormat":1},{"version":"dc923d45c03261241cefb03ea952883784e28fbc1d089af1a5a553a7b0f5e2b5","impliedFormat":1},{"version":"764150c107451d2fd5b6de305cff0a9dcecf799e08e6f14b5a6748724db46d8a","impliedFormat":1},{"version":"b04cf223c338c09285010f5308b980ee6d8bfa203824ed2537516f15e92e8c43","impliedFormat":1},{"version":"4b387f208d1e468193a45a51005b1ed5b666010fc22a15dc1baf4234078b636e","impliedFormat":1},{"version":"70441eda704feffd132be0c1541f2c7f6bbaafce25cb9b54b181e26af3068e79","impliedFormat":1},{"version":"d1addb12403afea87a1603121396261a45190886c486c88e1a5d456be17c2049","impliedFormat":1},{"version":"1e50bda67542964dbb2cfb21809f9976be97b2f79a4b6f8124463d42c95a704c","impliedFormat":1},{"version":"ea4b5d319625203a5a96897b057fddf6017d0f9a902c16060466fe69cc007243","impliedFormat":1},{"version":"a186fde3b1dde9642dda936e23a21cb73428340eb817e62f4442bb0fca6fa351","impliedFormat":1},{"version":"985ac70f005fb77a2bc0ed4f2c80d55919ded6a9b03d00d94aab75205b0778ec","impliedFormat":1},{"version":"ab01d8fcb89fae8eda22075153053fefac69f7d9571a389632099e7a53f1922d","impliedFormat":1},{"version":"bac0ec1f4c61abc7c54ccebb0f739acb0cdbc22b1b19c91854dc142019492961","impliedFormat":1},{"version":"566b0806f9016fa067b7fecf3951fcc295c30127e5141223393bde16ad04aa4a","impliedFormat":1},{"version":"8e801abfeda45b1b93e599750a0a8d25074d30d4cc01e3563e56c0ff70edeb68","impliedFormat":1},{"version":"902997f91b09620835afd88e292eb217fbd55d01706b82b9a014ff408f357559","impliedFormat":1},{"version":"a3727a926e697919fb59407938bd8573964b3bf543413b685996a47df5645863","impliedFormat":1},{"version":"83f36c0792d352f641a213ee547d21ea02084a148355aa26b6ef82c4f61c1280","impliedFormat":1},{"version":"dce7d69c17a438554c11bbf930dec2bee5b62184c0494d74da336daee088ab69","impliedFormat":1},{"version":"1e8f2cda9735002728017933c54ccea7ebee94b9c68a59a4aac1c9a58aa7da7d","impliedFormat":1},{"version":"e327a2b222cf9e5c93d7c1ed6468ece2e7b9d738e5da04897f1a99f49d42cca1","impliedFormat":1},{"version":"65165246b59654ec4e1501dd87927a0ef95d57359709e00e95d1154ad8443bc7","impliedFormat":1},{"version":"f1bacba19e2fa2eb26c499e36b5ab93d6764f2dba44be3816f12d2bc9ac9a35b","impliedFormat":1},{"version":"bce38da5fd851520d0cb4d1e6c3c04968cec2faa674ed321c118e97e59872edc","impliedFormat":1},{"version":"3398f46037f21fb6c33560ceca257259bd6d2ea03737179b61ea9e17cbe07455","impliedFormat":1},{"version":"6e14fc6c27cb2cb203fe1727bb3a923588f0be8c2604673ad9f879182548daca","impliedFormat":1},{"version":"12b9bcf8395d33837f301a8e6d545a24dfff80db9e32f8e8e6cf4b11671bb442","impliedFormat":1},{"version":"04295cc38689e32a4ea194c954ea6604e6afb6f1c102104f74737cb8cf744422","impliedFormat":1},{"version":"7418f434c136734b23f634e711cf44613ca4c74e63a5ae7429acaee46c7024c8","impliedFormat":1},{"version":"27d40290b7caba1c04468f2b53cf7112f247f8acdd7c20589cd7decf9f762ad0","impliedFormat":1},{"version":"2608b8b83639baf3f07316df29202eead703102f1a7e32f74a1b18cf1eee54b5","impliedFormat":1},{"version":"c93657567a39bd589effe89e863aaadbc339675fca6805ae4d97eafbcce0a05d","impliedFormat":1},{"version":"909d5db5b3b19f03dfb4a8f1d00cf41d2f679857c28775faf1f10794cbbe9db9","impliedFormat":1},{"version":"e4504bffce13574bab83ab900b843590d85a0fd38faab7eff83d84ec55de4aff","impliedFormat":1},{"version":"8ab707f3c833fc1e8a51106b8746c8bc0ce125083ea6200ad881625ae35ce11e","impliedFormat":1},{"version":"730ddc2386276ac66312edbcc60853fedbb1608a99cb0b1ff82ebf26911dba1f","impliedFormat":1},{"version":"c1b3fa201aa037110c43c05ea97800eb66fea3f2ecc5f07c6fd47f2b6b5b21d2","impliedFormat":1},{"version":"636b44188dc6eb326fd566085e6c1c6035b71f839d62c343c299a35888c6f0a9","impliedFormat":1},{"version":"3b2105bf9823b53c269cabb38011c5a71360c8daabc618fec03102c9514d230c","impliedFormat":1},{"version":"f96e63eb56e736304c3aef6c745b9fe93db235ddd1fec10b45319c479de1a432","impliedFormat":1},{"version":"acb4f3cee79f38ceba975e7ee3114eb5cd96ccc02742b0a4c7478b4619f87cd6","impliedFormat":1},{"version":"cfc85d17c1493b6217bad9052a8edc332d1fde81a919228edab33c14aa762939","impliedFormat":1},{"version":"eebda441c4486c26de7a8a7343ebbc361d2b0109abff34c2471e45e34a93020a","impliedFormat":1},{"version":"727b4b8eb62dd98fa4e3a0937172c1a0041eb715b9071c3de96dad597deddcab","impliedFormat":1},{"version":"708e2a347a1b9868ccdb48f3e43647c6eccec47b8591b220afcafc9e7eeb3784","impliedFormat":1},{"version":"6bb598e2d45a170f302f113a5b68e518c8d7661ae3b59baf076be9120afa4813","impliedFormat":1},{"version":"c28e058db8fed2c81d324546f53d2a7aaefff380cbe70f924276dbad89acd7d1","impliedFormat":1},{"version":"89d029475445d677c18cf9a8c75751325616d353925681385da49aeef9260ab7","impliedFormat":1},{"version":"826a98cb79deab45ccc4e5a8b90fa64510b2169781a7cbb83c4a0a8867f4cc58","impliedFormat":1},{"version":"618189f94a473b7fdc5cb5ba8b94d146a0d58834cd77cd24d56995f41643ccd5","impliedFormat":1},{"version":"1645dc6f3dd9a3af97eb5a6a4c794f5b1404cab015832eba67e3882a8198ec27","impliedFormat":1},{"version":"b5267af8d0a1e00092cceed845f69f5c44264cb770befc57d48dcf6a098cb731","impliedFormat":1},{"version":"91b0965538a5eaafa8c09cf9f62b46d6125aa1b3c0e0629dce871f5f41413f90","impliedFormat":1},{"version":"2978e33a00b4b5fb98337c5e473ab7337030b2f69d1480eccef0290814af0d51","impliedFormat":1},{"version":"ba71e9777cb5460e3278f0934fd6354041cb25853feca542312807ce1f18e611","impliedFormat":1},{"version":"608dbaf8c8bb64f4024013e73d7107c16dba4664999a8c6e58f3e71545e48f66","impliedFormat":1},{"version":"61937cefd7f4d6fa76013d33d5a3c5f9b0fc382e90da34790764a0d17d6277fb","impliedFormat":1},{"version":"af7db74826f455bfef6a55a188eb6659fd85fdc16f720a89a515c48724ee4c42","impliedFormat":1},{"version":"d6ce98a960f1b99a72de771fb0ba773cb202c656b8483f22d47d01d68f59ea86","impliedFormat":1},{"version":"2a47dc4a362214f31689870f809c7d62024afb4297a37b22cb86f679c4d04088","impliedFormat":1},{"version":"42d907ac511459d7c4828ee4f3f81cc331a08dc98d7b3cb98e3ff5797c095d2e","impliedFormat":1},{"version":"63d010bff70619e0cdf7900e954a7e188d3175461182f887b869c312a77ecfbd","impliedFormat":1},{"version":"1452816d619e636de512ca98546aafb9a48382d570af1473f0432a9178c4b1ff","impliedFormat":1},{"version":"9e3e3932fe16b9288ec8c948048aef4edf1295b09a5412630d63f4a42265370e","impliedFormat":1},{"version":"8bdba132259883bac06056f7bacd29a4dcf07e3f14ce89edb022fe9b78dcf9b3","impliedFormat":1},{"version":"5a5406107d9949d83e1225273bcee1f559bb5588942907d923165d83251a0e37","impliedFormat":1},{"version":"ca0ca4ca5ad4772161ee2a99741d616fea780d777549ba9f05f4a24493ab44e1","impliedFormat":1},{"version":"e7ee7be996db0d7cce41a85e4cae3a5fc86cf26501ad94e0a20f8b6c1c55b2d4","impliedFormat":1},{"version":"72263ae386d6a49392a03bde2f88660625da1eca5df8d95120d8ccf507483d20","impliedFormat":1},{"version":"b498375d015f01585269588b6221008aae6f0c0dc53ead8796ace64bdfcf62ea","impliedFormat":1},{"version":"c37aa3657fa4d1e7d22565ae609b1370c6b92bafb8c92b914403d45f0e610ddc","impliedFormat":1},{"version":"34534c0ead52cc753bdfdd486430ef67f615ace54a4c0e5a3652b4116af84d6d","impliedFormat":1},{"version":"a848339c272ab23e686b5d0b81297e3a7116ba7d27589c66ca1f4ebcd65e7f19","impliedFormat":1},{"version":"566315d39e476ca9e7fd0b1908074cb2a5ff9246cc3ed7da64cde5ad30f7e0b1","impliedFormat":1},{"version":"5f7ed82e82aa5017fe830ac40105321c32e3bf45d3cfcd557601f2ed58ff6ca4","impliedFormat":1},{"version":"9c8125fc43f5fc74a40240438d849d56ec7e5eb68961ce8af70a930ffb0580b3","impliedFormat":1},{"version":"d8d07d4c2cb69335afe919f64e519bd3972d8265ba1a073e4e7a2f1a0ddbe2af","impliedFormat":1},{"version":"fd3d0e2bc2829d94b6ea717f0217cc1fbe7f7e5c3e6dc20554d8682d3850ad72","impliedFormat":1},{"version":"e71863e8db54c3584405caa0331efbf08ab6db455b192e95ceb44a2905eb9124","impliedFormat":1},{"version":"a229c67e3306551dbd0310b21712247ffed4e881c7a834a19d62a149c8cbd3d1","impliedFormat":1},{"version":"83b5f5f5bdbf7f37b8ffc003abf6afee35a318871c990ad4d69d822f38d77840","impliedFormat":1},{"version":"634e56b085407249a5c67e6520fd7de77060f28c61c901e2e4d23784c204596f","impliedFormat":1},{"version":"a9caafa49a3cff37544feff78ea8a6d3bb49f7adce053a6eef7c8ed4ea87928d","impliedFormat":1},{"version":"b86ef9f4a38b5f28352d7b2a9f9a64eb0097cb01dadb9f6e57843b143c2e04fe","impliedFormat":1},{"version":"23ad184b6ec52e8c1eeee56ffb3ee922481330716025ab133fe1f0425bddcd78","impliedFormat":1},{"version":"d0bdbe18ff5e0ce26b0208dac7fb18f989547ae67661fab18eaf7d9c2a43eead","impliedFormat":1},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":1},{"version":"6ac5de574ea90ae410c8aa8528dd4e6ae38b6abd2161f4efd9b6571212c14d4d","impliedFormat":1},{"version":"da6a8af46de75441eb9eec84b82cb7a2a8fe7385ffdf29dff17a8f92e4e50b0b","impliedFormat":1},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":1},{"version":"50635666275413d74bdcec40ebff603c3e4b36cf62376c8781911951d55e1f01","impliedFormat":1},{"version":"fd77d9bad26c739ff2d8e9535f2bf2773bc340eb2e70c76a8d59e1b10d6543be","impliedFormat":1},{"version":"e6a42337746abfde6cf108ff93c5e5144f105fa8dc8f4bd6eb4b9f412c49d234","impliedFormat":1},{"version":"ab4dbe62366e31f41e3337c829f806c395a028aad37669ad60e45c58e34f8dc6","impliedFormat":1},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":1},{"version":"33942d18807cbf38c440d00701b9c83adf4bf5d66ba5dddbf01843384a24046f","impliedFormat":1},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":1},{"version":"267725112f5f9cf7235a8c1b1f995ad446e287dd50eeb088e6b6c4ae184aab92","impliedFormat":1},{"version":"110e37c80a94ae1f0bf01a12985d12fa9ea08e4f5226a5bfd7bb702e94da721f","impliedFormat":1},{"version":"103658c98e0af6a41556f845bb35886db9c95f9e40b6c345f476a85afc72add3","impliedFormat":1},{"version":"039f3db1416b09d58d86ffd54cbc08f975cbfc75e3857b5634ac461ca2123609","impliedFormat":1},{"version":"c7f6351ac45abfc84332fd2255e4fc9f40ab81be67418f95653c5b635f06489c","impliedFormat":1},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":1},{"version":"5db7bf2b215e4ad7db9356f7fefaf579ebc3669393c81addc94c815f7c56883a","impliedFormat":1},{"version":"1c465dcd7e295ca87621cfc722410abc34d2fb38133cc4d39a88182e7c1776f4","impliedFormat":1},{"version":"1ff6207c7c85da59a11b2a1ef4cfa88347b52f117faa4bdbd6e6bdb60d634719","impliedFormat":1},{"version":"89fff07e4f4a77d0973bde15c5652dcc6c2a5e67e3460a3ab14b520e8ee0dfec","impliedFormat":1},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":1},{"version":"ac2b859d346b9c79548810c0b5821b05a6a766db90bed7416f7ec0cc6bbbd3bc","impliedFormat":1},{"version":"68408a0a4000e2d3da6984c995252646d3ce12a0d593e97c12b7f4fd0ee22c86","impliedFormat":1},{"version":"4e0576acedaa905e811e3df0cc488d8e18c14925f14ab2a8e53935a4c0e4cd03","impliedFormat":1},{"version":"0459d2cac302ab51fa7f27504cc70bb0ecf36eef8cf23221effd4e16946ed89a","impliedFormat":1},{"version":"6d861e053bfac8f22f7612f6dbf0e1f4e8b2687fe93bda154b8dc34c643c26ad","impliedFormat":1},{"version":"eab879e68089c36bb373977a6e9338fa19a607f5581d30f2e5252d9333590922","impliedFormat":1},{"version":"cf06fe0d1db6452d4823c347e22edaf92d719d51d17faeb6a5c17f539b4a2dfa","impliedFormat":1},{"version":"ef0644180b71552749ee1bfa27286360d31a73de7c1fd9e43cbb70a04e2199db","impliedFormat":1},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":1},{"version":"74b321f970095426033b06dd2a8bf029b09688ec3747196d9455c526d0327de9","impliedFormat":1},{"version":"b8bbadecf2b1ca66f8ab691aed9910b37b3d3532ac3de360ea0141630d7701a2","impliedFormat":1},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":1},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":1},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":1},{"version":"faf4a3ee383cc6bb81207d4f8730e6d90ac38a010ded7583e4ba1bab1cf57b5e","impliedFormat":1},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":1},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":1},{"version":"4f52c5d04464feecaf4e55db0a0cc42d38b84a502afb54082ed6c2c8352c90d5","impliedFormat":1},{"version":"3a2a7e7343d20346af5b944a8d39d1756809c86f05bd95c4f62d53fb27a14a73","impliedFormat":1},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":1},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":1},{"version":"f22b102137264aa6bbe59e60da29993215477fbdf063109e36448c31c3ce369e","impliedFormat":1},{"version":"13137e2629cc5684fdb9dff20ef99674ac13609fb7bf0d52b6d1ee7bfb743e5a","impliedFormat":1},{"version":"f02a49f495484978dbc62b9dd7bc63904a4853754b789876f4f107aef19585ee","impliedFormat":1},{"version":"4e0b3f6636723fbfb0d0b2587aae5135b4d142aceaa185adbbf8260313e5526a","impliedFormat":1},{"version":"e9fceb52882eddb54d6010a966c1ff3b3056d3cc652ab020d32be7022030fbee","impliedFormat":1},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":1},{"version":"9acc62b08ad2fa64f71f557b849430ee8c96300af72448b86104ad5e3eca8bb0","impliedFormat":1},{"version":"56bac357cefcfd19e72e66bd6984bb39adeef3d513f6c5f396d97040b5a5dd4b","impliedFormat":1},{"version":"f81515e6391f020a4b8f346a2af47a48a85a12491feba9166ce71c692e45441f","impliedFormat":1},{"version":"702caf4b27b5454a3305f20cea27aaae0c5673b91df4936a8559f3112e4d68b2","impliedFormat":1},{"version":"0456f6abe968e44aa231527842e90fc493ccf0086c044685cb66fc9d307d5767","impliedFormat":1},{"version":"14b62d4b47d0c2c832ded6390e33303be8be08b21c1ae0d478a46f0f49854e78","impliedFormat":1},{"version":"5881743bcc06d8169dfbbb238422a9d75d3930b09b59e099d0aa4ebdfee7dc0c","impliedFormat":1},{"version":"d962ff332884aa5af93c4601189c35747b6724765a3cd697242b5ef1e02cef70","impliedFormat":1},{"version":"e5af1a573638d8532157d7c69bffe9aa3551ae84db39d02d255efbce2207b342","impliedFormat":1},{"version":"e3d196421e621fa84174dd79502e01d2e00d67e23362a8c530f7e05cd68e8ea1","impliedFormat":1},{"version":"f5e8dd756948f1c077b3ecccbdc1f95aa5a5edf4f58dd079667d4611814666e0","impliedFormat":1},{"version":"214cbcbd70d482acbe40ed45aaa8383e98c86a86706afa85cdddc9719ac548ab","impliedFormat":1},{"version":"73f84a43613929bfe3efdbc61d2dc1ae39e5a32c35795f7806cf0a60c83e60a0","impliedFormat":1},{"version":"6957a2d31554536d37e96402c117b2429f2e9baee89f26b87caace936ca2ac37","impliedFormat":1},{"version":"bad9e7bf88505357ad4e64ec0a87b7abfdc783fbba6d3c257d2eee2493703304","impliedFormat":1},{"version":"c90c20f613309279aa05bcb314e75d762538bdb1e5bb1ace75d1c1ef2a979637","impliedFormat":1},{"version":"923b19f9e0d134113ed5b15f48a046db1afbab4e34abad9993ba873b9e18dc7e","impliedFormat":1},{"version":"3c4ab379d2e80517f92e24479d0161f58fab9ec7b2b508d2f243ca765aca0050","impliedFormat":1},{"version":"e18dd77323af9b0e4f7b8a4de60688b08c157814a59383dda5b9dacce2230f46","impliedFormat":1},{"version":"ae76106be2fe3281cd7e96b9dc9e12b4583e61e31bae624656ec0feeaf75371f","impliedFormat":1},{"version":"afd70a57b376a4e926abd4c1c8e9310fe96c969d5a0197ffcb565d001676a9f6","impliedFormat":1},{"version":"d7538da5cadf8bd654a7725666b4382a9ae6f9aed039098a36ee878ca6a3bec8","impliedFormat":1},{"version":"8d48652a8cb3ab8370fb264ba855d9f5f232553a3d9f5bd25a88b290e3e23c10","impliedFormat":1},{"version":"6cd4ec90f2ed6e15bd4e940d5ca7aa38cd570d1cb1d0b9952624e5c8ddb4dba4","impliedFormat":1},{"version":"a7f1293a7400026dc420559629b54c8493343200ff36d92a8d78502a9282a35b","impliedFormat":1},{"version":"d2f3f85583a57ad1987ee9f6b8b174499e9c5d7115e37dc9a62a2dcb9b054d1e","impliedFormat":1},{"version":"7ff52e014e5f1f653a7c3d0f5f323e8e7daf5db30abfee1380b78dada4ec2da9","impliedFormat":1},{"version":"be6b8d983bd7d37162db454ea038196367a799156c3caa33927494bc99d885d0","impliedFormat":1},{"version":"b343cbbbeae17e5c0ac05bab9dd4e08c57a1559cd31659d7e152bd122ae646d7","impliedFormat":1},{"version":"42c67685ee5027789a51538b046b3a7a11a2b19705ebc63ce3f0404b8e9fb0f9","impliedFormat":1},{"version":"693104e41fb5dc31f325c518cce5eef5513d91a650148fcdecc064d137f8581d","impliedFormat":1},{"version":"c54d981103b6a51e2e7f52821795ea2f8a2e08093cbeeec3016613697df11d87","impliedFormat":1},{"version":"9f1a99a5145d55e9543b58d51eef81ed14318575355f554c76c97ec043d31131","impliedFormat":1},{"version":"2829c78953b67be242428630b35cbae50af7dd9c9b24ef5c467986a1f14d94cb","impliedFormat":1},{"version":"da3eab33856ccf1f35e8e9ded34994f2b4a23422f1e0e99f38805f66d4231a3b","impliedFormat":1},{"version":"43276dfec18eb7175615c6327a4ee01a116de68e37cebb56da1dd742225d3ac9","impliedFormat":1},{"version":"bc6e29688d6a2cae05887ddbb04aca69aff1e5102ed1074671445bcca1c881d3","impliedFormat":1},{"version":"51718633ad06a6d05d68a9ac009d49e97b84d980ed06b1cd04f0b40398310d43","impliedFormat":1},{"version":"ff90a6bc7812f609903f41b98c60f3edc2d593483fdeb9bed20cb93e6527a777","impliedFormat":1},{"version":"7da854941074e76cd1ed6f23c7ae615e931589f9cd3ef919ce832f47d666ab6d","impliedFormat":1},{"version":"2c6dafeffbb2afc2c681099fea24af5b76c43553d40867e25efe097ed4c78965","impliedFormat":1},{"version":"64135cfd2a693174828c8e842198f5e6854e6862df1ea685d62bc1a20fde9006","impliedFormat":1},{"version":"8c525341425df5d0863a79895b37ec46d306f98d856f6094b52c87c55a77fd31","impliedFormat":1},{"version":"dc8a332007798357766fb7131b180bbcd917b5cd98916d827d9a05edb9260e0b","impliedFormat":1},{"version":"831444604ca9fbb1f144fb399b92e3de5ce9d6c4c651f061fa5e34f72e29197f","impliedFormat":1},{"version":"f5831fcfbcf7f09591af1e5dec357cacf57b7e1259267a4ae5769b7f83f8a441","impliedFormat":1},{"version":"aae56a55145c92171dbe7280fd6c0ae4c286b2933b4b0ea56d398f6abd82f454","impliedFormat":1},{"version":"2b8d26d51897913d32cca6dfcbf2c509e35f77415e50a93466d560cf42ef703e","impliedFormat":1},{"version":"4fb248f0a9fed6d8658e6bdd6c1422b1a7fd9b27cf30bd3b1a5a26fe4d7d8963","impliedFormat":1},{"version":"d51176c3c6362a0f9a59184c71f3b8d8471b6a6a4060258c4264722fc5a11463","impliedFormat":1},{"version":"a2e88c1cb313192e2e5142e8898dd35b39a4f30d272cb07577787510df606bda","impliedFormat":1},{"version":"32b457a43b19f02c0fa6b92ed3171e2052cdd0eb2819fddb60b7324d4bc3b406","impliedFormat":1},{"version":"e172920ce3b5f5d4ba43ae4a4a2c6a61ea5960f267e5d25cc84dc12527005f6b","impliedFormat":1},{"version":"0e8785bc79cbfc14a2c4a001e272ff0ec909ec94564705e85664db9492265e1b","impliedFormat":1},{"version":"20c8eca485f3f73c9d5855a1c99029f2907846b88d0ec81dcc11d6abc20f5653","impliedFormat":1},{"version":"25c8897df13b2f74c1c3e68c3e8d1f22bd7adadbb0ffa6e48e14e09045694ff5","impliedFormat":1},{"version":"253db8a1162220c88e504d2e31af9a9afe802a498a8b4920ae5b8751bbbc7bbb","impliedFormat":1},{"version":"df35bc4ff5f2fa4cddd5d499477c595ea76644bd03150922e0c20184ce1f76ec","impliedFormat":1},{"version":"d8c33684d5af091b42e5e4fac2654ae0e4fb707ecd56d2b5ea954f1754dbff36","impliedFormat":1},{"version":"07d5c61850d955ff344ccacc4c35a1cc1b534ef92201da4d555e3cae26ca994c","impliedFormat":1},{"version":"19ba4067fc331691fc5af2aff7dfcf39a0b6d50b5bda255e3c6682b32983e5d5","impliedFormat":1},{"version":"a22fb21723983b4e2edf3d34893256c8b6075f77254f394048541f5a4eb25d15","impliedFormat":1},{"version":"969948f990cbb4f0b594d8b3e66bc37d04f4896314afb888e507ae0fb9aaac51","impliedFormat":1},{"version":"8b9782193fd21acd035ca67a18e607ca68e8345d5931962ff5862d89fae1965e","impliedFormat":1},{"version":"107c2243004cd47d8a63b15b42644343db310383b8008237f7563710116589e2","impliedFormat":1},{"version":"9a3a28ed970a073f6f87f9827839c2d06ecdd05f45e07ce30899f72ca968b46a","impliedFormat":1},{"version":"08107d403a7a4235fd239bd1185800d10f646ea07a71b119c2252713d466920e","impliedFormat":1},{"version":"175707c3c7618f8e3ea64636dc591ed6892328fa430149d3ad414018751da8f6","impliedFormat":1},{"version":"4b086cd2bf1f7fdac4fbbe9acb863b29040fd8ac4188c5d7a5b3c95bafa1b380","impliedFormat":1},{"version":"2a7ef8d34c40308dc2a2b05a78b8ee602f205e82e4eac3f44f1959e95bece679","impliedFormat":1},{"version":"022d05125afe3135d923892f13d1b176003560edd270900f52957a07e1efeab2","impliedFormat":1},{"version":"e0fa1fa96fdf10e88c8a23aa4eb2566232ac5f8d93961815158a7c6b22d7efaa","impliedFormat":1},{"version":"0a6a304a71bc56611b60ad013e583564b6056b8265961123d77fd65fd8b74061","impliedFormat":1},{"version":"63bba6da188f796caf21284a73dab06f85bd17042bd5ad49c0ec81451fdb0f5a","impliedFormat":1},{"version":"5f2a79b58c58371b68d6f3a3a225e0804c6ce517c423c8a1efec234765de7586","impliedFormat":1},{"version":"358a84f9e1f6680ffddb329a580be5f932c7ff10ced8d60f43904f66dddebaa3","impliedFormat":1},{"version":"b60efbac98231283107121b5b3327f56a6632c2d14d7616920bc309a4f6d4bc3","impliedFormat":1},{"version":"0e58e6f3fa554921c7950ff344d6c299caf9260e4c78bf7c699a6b5a6d02e6bc","impliedFormat":1},{"version":"3eb80f1addaca2a298abd6186a2cfe98567d88635f334a0f2415438ec5f2d3b7","impliedFormat":1},{"version":"8d5af927098c40f41f315552e7ff9ee9376b26fc03e83421a0cf2d93907ea346","impliedFormat":1},{"version":"c993b44ec48e09bf9e9b512941379656f9090ddf81f7ab61b4d8a6cdbe7ec409","impliedFormat":1},{"version":"54f323b0c25677fcd7dbb6541667f131e17bf49d48b6efdfed72ae71722fe8f5","impliedFormat":1},{"version":"7668c31fc1a0d6c5ee0ae1048d41f7232a56fbe18367929f78bd0c77303af11e","impliedFormat":1},{"version":"8b41773894ca3ba064857d72a6cbd669b299e47915c3b97cbc2a613735cbf35b","impliedFormat":1},{"version":"badddb55fb1a8186abb7d4b972820f9e5763916e59e9567a847d0237ba0f72d7","impliedFormat":1},{"version":"74689440172e6a26a40b93a21ca3f263e2d06bada97b270a884737f921e7818f","impliedFormat":1},{"version":"9c3ded425a22114083d56daa858d27b46bc1b059aeb023f504574312ab1439ac","impliedFormat":1},{"version":"08f50b290537a8bea3a96920b5d5664d4cd23472161af28c8bcdc5091250c3ce","impliedFormat":1},{"version":"c4d0d823f114af573cdd62f5724648cb9df7a7ca1f8473ebe65b7d7df1789422","impliedFormat":1},{"version":"e16aa5f3e598ad86a044934071f16729c0f95fd77794f0ada7a88faa2f66c185","impliedFormat":1},{"version":"098148c34c5cef91a12c622fadf8d19a7f513206d3dc61fc31af13fb361d99e9","impliedFormat":1},{"version":"4130eea8635f6d6bc82a8a9560b8064c163b1029d3efa39815fb53c4aa51c145","impliedFormat":1},{"version":"f1c957e436f37c6bd81fd6bc6a13eb1bf7a9ad5f297a167db0e96415f621ed66","impliedFormat":1},{"version":"98144631dc436418a7b927607618136353a32f4ccc420b76358a730310bbcc8a","impliedFormat":1},{"version":"026447d4bf29241ac992589ec620a86b13c76bdfcb1ff8dcc7e26f0eb2d0d210","impliedFormat":1},{"version":"12f79c131043198b4d0f789df3cc4b90d5cc00dc0c64afbe9e6965f4a55b3d61","impliedFormat":1},{"version":"fa890a742e523ead1ac2d8738c29c843d2a1acaa98da02a7667fe00d177aa196","impliedFormat":1},{"version":"b99faf232d2c47ddfdfa086a4bb0665bcb25e3a3989498d467caaa79200afb06","impliedFormat":1},{"version":"21e4a665ab9901d7a9f42aa585fc3bfba8ef4d090640a1e412669a0bb392edb2","impliedFormat":1},{"version":"e7d94b0ae7d41c1bd5f1aa4c2bd62676af83e1fe743316bf82bb32ec1be11421","impliedFormat":1},{"version":"aaf88ec377baa9cf35177eab96b5db57bcfdc5bbe34bf38b1805d883f6b2cfa4","impliedFormat":1},{"version":"d4b211bb230daef2a02fb8952c1b21730d4d14d70baba4f04c5efce000205ea7","impliedFormat":1},{"version":"8eeb941ef7939f9f0180fafe779c7fa9e1049b5716a654fc25463fbf472d3dc9","impliedFormat":1},{"version":"80e2b75b44778105663dee124d241ba133250df92d3b5760784ef9683c622c1f","impliedFormat":1},{"version":"e23514abb70d5803377e5367af5a9554b15529d97b658930335b195f9d5753b2","impliedFormat":1},{"version":"b5af0716932f268f2a4a41420d7ba9fdbc037e1bb406aa57caa7616b173422c6","impliedFormat":1},{"version":"af67cf7922d64c7e1cc0a0c327191d97ef6e1d54f7f1661a06e7225fa8b35e48","impliedFormat":1},{"version":"67ae5eaf9ef6ed32a30aced05943e9f83df215d62f80076f7cce3a55d08c8722","impliedFormat":1},{"version":"8bf4808d0cbdfee342649aaa6744ccdb7f3b98c127985024474f961e3a96d038","impliedFormat":1},{"version":"27e56c281e88ef3107c9ce67f02bdcfba297804d3d14006a3e3d59f45a3f1d9a","impliedFormat":1},{"version":"42d00c41e9cffbb3cfbad77417055030f952fe8d7dbd8f646fd0005153b6e821","impliedFormat":1},{"version":"ecebc4355edf1384d191afa1c0c4ccacadb199ab55c90c9c450720425e975fc5","impliedFormat":1},{"version":"77ff7b7d3bef88309b2c6b48e2fcdb7db8000b57f7f627b9481b014ef2db7581","impliedFormat":1},{"version":"b8d5fc4baf94f4aaf437c2505b751083c58983a126fa712d34ac5e4e7d064ee1","impliedFormat":1},{"version":"8f3a98972a1f230e69a9c11e2b78ead1761bcba0e6cd7ba029e1e57cb5f89eb8","impliedFormat":1},{"version":"f681b47b5e0275d8a2fe219e40d2c80fdac5c6f865af6fc61df0f15c59c6c9ee","impliedFormat":1},{"version":"17bec14562208b93665ecee566ecb99baf6ca82eeb92ab1eb9e3442aafb26a99","impliedFormat":1},{"version":"fb00be4532eaf1800d8a2a625a8843f5d8f122990d2bedd72ebeb90a555f8cd8","impliedFormat":1},{"version":"374ddae0cfbf334836cfbaf71ec0ab9c16e677306d31f6e843e428119a90dce7","impliedFormat":1},{"version":"688e6406967d02af975bd78a3015d9ea0d1d3bad93d62df0329bab69cd278f97","impliedFormat":1},{"version":"d8fd376b0555bd256ee497d88cfad88d6edce66b0136c57ac4e06c2c1226b48f","impliedFormat":1},{"version":"c9df649292e24f2cf451d81faaf1a5dedde2fa1bf043b3f74a647506cdc83ef4","impliedFormat":1},{"version":"0fe0ef68e8dd9341ac44174b965cad70c377d078c86085b92123ca8f1a6c5c56","impliedFormat":1},{"version":"75543c4a7a4755138a91f7260d663c393b700d3ed5fec65538851286f427c234","impliedFormat":1},{"version":"98d24457c0cc7837ec1e984430abb00a1b3556cb84134c00712cc08a6376b2a5","impliedFormat":1},{"version":"69fad132b41289d40e3d373ad0ef1065cbae26c78f11d2f1aa574a3eb9b7cb7e","impliedFormat":1},{"version":"eb2d716abdab98d94a6be94686ee0ed7506ba70dde267cbd6c24a1e94cfbda60","impliedFormat":1},{"version":"58d9aa65560fbd9932063c2be022c9346ab7c79ccd10522c61b60cc112ac4341","impliedFormat":1},{"version":"c6f19eb7c75210f37cbe83efae91b7df76a7999dc40fd7b79c7363b67a8790f5","impliedFormat":1},{"version":"68ad94cf9d4b8f9affd8682dce00daf93cd8c5e3072237c22dc6138b997e0bb6","impliedFormat":1},{"version":"42c93fb5d5b061f7f95a3ea59c2f2b31256939e97f5183c827234d32160901ea","impliedFormat":1},{"version":"0c27495545631e99af91cebdb55d81320cb7785e05e86a3745fac2949564a12f","impliedFormat":1},{"version":"248127a1de2be832dd6b9a4c2589ea241f3caa9bde05ee788dfa2128c6ee3218","impliedFormat":1},{"version":"b5929098539f514464ac3c468bc3c93ea40996de9eb5e770ecaefcfbb2677758","impliedFormat":1},{"version":"b9a6aadc8283344ec53b55c8774fd92ddb473b3702dc846afaaee80c49b50131","impliedFormat":1},{"version":"2f5685827f73fd674138b81ca48275458be188b01af8a00315231efe74397980","impliedFormat":1},{"version":"6097f1055816adfa4bfc72e65ee4b6abd46d1fbd65bff4c7c7180a031408321c","impliedFormat":1},{"version":"053b26604faa65cae78a2981e04dd917fb068fd367ac39203f02d64ed0ff498e","impliedFormat":1},{"version":"cd1876d05dc46e55c625362f53068c9d4e7307c7e74a590b2975c657a1d3b625","impliedFormat":1},{"version":"d0d95a2893c600f181619be0c35d30845d93ae669e0aee435f9c95f6237c6872","impliedFormat":1},{"version":"1d11106d40ce32966017ecc516704eef10e57f72e376b1ae3d66c9f9cc6dc14b","impliedFormat":1},{"version":"249470a524039cb8a01af509318599f57b0808a602617eb7a655e2ec584446a0","impliedFormat":1},{"version":"07c06586fbe383765f7a404c9ef9b96ce61aeb6849f375ba5fad13139051f326","impliedFormat":1},{"version":"a54442465e1152334910e6533d10e92fbb8b2d36ab0aeadcaeb3f3f151a8a825","impliedFormat":1},{"version":"49f3ec14c8cc3e911e40adc42a42f40c109d8a62c655cf905d15e235883b369c","impliedFormat":1},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":1},{"version":"6ac5de574ea90ae410c8aa8528dd4e6ae38b6abd2161f4efd9b6571212c14d4d","impliedFormat":1},{"version":"5e1dff61bf656dc543938e285ba67f84ba0416959a7c10792cdf19ae3b50672e","impliedFormat":1},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":1},{"version":"171c708e6354a6bea00590af5b2e4c67ac5b6db17f5c6610074aa5f89df16e48","impliedFormat":1},{"version":"e742691b51bb384ebe58d525ed1b2029521a52dc63e35d445208a1efffb4089b","impliedFormat":1},{"version":"e6a42337746abfde6cf108ff93c5e5144f105fa8dc8f4bd6eb4b9f412c49d234","impliedFormat":1},{"version":"ce8044e0d8a197824255a3738253d7045666f9ac7ede8ee9ce79da095e4aa99c","impliedFormat":1},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":1},{"version":"89c5aa237d783c2938e56bf33611ec6b54c1df76c62d474c2475b6120f185a7f","impliedFormat":1},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":1},{"version":"267725112f5f9cf7235a8c1b1f995ad446e287dd50eeb088e6b6c4ae184aab92","impliedFormat":1},{"version":"110e37c80a94ae1f0bf01a12985d12fa9ea08e4f5226a5bfd7bb702e94da721f","impliedFormat":1},{"version":"103658c98e0af6a41556f845bb35886db9c95f9e40b6c345f476a85afc72add3","impliedFormat":1},{"version":"62f0cde62b87e93aa0ca1a00e24eb7fb4196d2eb8d69fe27064c96ae92ae3527","impliedFormat":1},{"version":"2a527df5c4828328fa6b35cf8b8f5bf0640933a4602c517faace7a1c3af0d446","impliedFormat":1},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":1},{"version":"5db7bf2b215e4ad7db9356f7fefaf579ebc3669393c81addc94c815f7c56883a","impliedFormat":1},{"version":"f6694bea88421c6d7342b5381b1a49fc823ae746680ca9ee16d518c9c16118e8","impliedFormat":1},{"version":"b8a25d32e4a2a187e2169f0936416cfcac8926f56166f3895fb5f82942f3150e","impliedFormat":1},{"version":"89fff07e4f4a77d0973bde15c5652dcc6c2a5e67e3460a3ab14b520e8ee0dfec","impliedFormat":1},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":1},{"version":"dec52a42c912503c35463f974fb86cb1a772cab001c2c9ed413093845be2f677","impliedFormat":1},{"version":"d2a2a7be324ab271073676edb22f5de259d4baf5bad32bd2e5545f957f503ac4","impliedFormat":1},{"version":"4e0576acedaa905e811e3df0cc488d8e18c14925f14ab2a8e53935a4c0e4cd03","impliedFormat":1},{"version":"22f897e17f18b702f8aa1c6e6412fcd33d180f8ef61297fec6c395a2b18d9908","impliedFormat":1},{"version":"9b48fb7d6521c10569a09921fea776719fab153e4b24d6bf4290fe6fab9be6d3","impliedFormat":1},{"version":"1850f2a9f8d382ef6f4cfea284a6994a1a636095d24f610ddbcd0dd74eef387a","impliedFormat":1},{"version":"ef0644180b71552749ee1bfa27286360d31a73de7c1fd9e43cbb70a04e2199db","impliedFormat":1},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":1},{"version":"608389d7f7df2c187768c89d6c4edc5ea43b1720203b4d212b85233ddb951ae3","impliedFormat":1},{"version":"2873b8fe4083b54fb60dd1d03ee8b22496e41f96a4e536e06cd59a481aba01de","impliedFormat":1},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":1},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":1},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":1},{"version":"e0f1bf295d165e3e7fdb6bbd9910888e9c5645e19cb4ae4b86303ee5ba2c951d","impliedFormat":1},{"version":"59879794c2339dd47bec19378cbd9c53cf7f6f851ccba99eed6eb1e0bd3d9519","impliedFormat":1},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":1},{"version":"2d99e3afe124a3c40300762492a49425df4b8090f771cac8034e233eed31bdb8","impliedFormat":1},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":1},{"version":"88f162f40062f4d9db248fed81d8d9258b2d0846ab8640904e220d69e4a040b8","impliedFormat":1},{"version":"df35eb1e5ccd6b597d18655f69dbbe24e8cca39ffe13822158c9756c528faacd","impliedFormat":1},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":1},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":1},{"version":"73b9a46d4b4135cf3e6d2c298027c1df258e4cbdc6ab9774b957ca5e207e6271","impliedFormat":1},{"version":"13137e2629cc5684fdb9dff20ef99674ac13609fb7bf0d52b6d1ee7bfb743e5a","impliedFormat":1},{"version":"7f7ff540a5107b5ff4b47191cd6e16a1be6d1ece99fc18b40a791dac9e3fb369","impliedFormat":1},{"version":"4e0b3f6636723fbfb0d0b2587aae5135b4d142aceaa185adbbf8260313e5526a","impliedFormat":1},{"version":"13060c51147d2e618fd5def7eacffb38fe536f2fd5ce23cdd0642f7e1e29202d","impliedFormat":1},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":1},{"version":"503b2da756cf3ab36e9708b43379353ca1c6aeeadfde37d757923b81a97f93e3","impliedFormat":1},{"version":"91ec9831ad545c826a697fa50de41d6fff9d927b505aa66ed46d4a9263a9ce0b","impliedFormat":1},{"version":"ee76bd42d21021012bd1f11c216c7e710ff11ebf8bf7ad134ebe1ed484ba85df","impliedFormat":1},{"version":"bd9d4c79b48d0f7ed5d5d3f7d06c029599023546fbe37fa33a95b525bb6febd7","impliedFormat":1},{"version":"e3abd62334ac65b6ae754c0950e014c8869c10e280da768c9c66aeafa1fcd9fe","impliedFormat":1},{"version":"fbb2889b4f092db88dc16994888ca046365ad0cd2232d1bdab0ae23db74a8a97","impliedFormat":1},{"version":"b32293aada046eb0690590b723571aba0c34e57566f1067d837277ac5ea5a58f","impliedFormat":1},{"version":"9255c726528f439160ea4d10ed4a091652db99f259acc2d9edc40c467fe91070","impliedFormat":1},{"version":"f66e1aa906b1de4070ba4dc224201a81dd36a30f9ae27cd9a20f9fbce57d98a2","impliedFormat":1},{"version":"f83adc50f31f8c405bfcc1e537f511df92b683cf1c6daf2236ca1551ba576918","impliedFormat":1},{"version":"84ca3c4c8f8b4b3328afa2c57555cfb71deb39053a1b783b81a44c0d183bbe1c","impliedFormat":1},{"version":"cde540ab3aa5b98e2555a87f25316ffdaef51dc2fefa113ed96c47e17e06efee","impliedFormat":1},{"version":"6a0d02ea3868cbbecffc40e314a8cf76ae69b14102b14b007b6d43edd7d4b078","impliedFormat":1},{"version":"76ea95684460d48dbf18e249903aaca73e8b011a5a6bc6e3071e75b3cfda1dc3","impliedFormat":1},{"version":"8fe2df5f35b364703a41744b0263abab1d0c94ec4da0e97e3fb09e7f1f49d1ca","impliedFormat":1},{"version":"cd39f4dc66037ba2987de3a740b709cbb19370fd564d7e8a1082806e52fc7da1","impliedFormat":1},{"version":"f1bad28391c2827fffcd7210a91babccfe7a097ee5c33bb20566d777e4525aec","impliedFormat":1},{"version":"84827ed75fe6e06f34cff9752e36084d9a0d316372b3862d59c487f3608b9014","impliedFormat":1},{"version":"fdc17b998a4934f09616fafd7920df0c0d120e22a805c12e8170f9ac473d3ac3","impliedFormat":1},{"version":"4815c2f8effb9c688066003158a8cc0079ab50d4bfa3ce90ec43c1399ed40d51","impliedFormat":1},{"version":"6669c0c622aee9286e3b8f8f0bf70b329ee2a45cf7d98ad1b7a250a43e7e3374","impliedFormat":1},{"version":"0c27495545631e99af91cebdb55d81320cb7785e05e86a3745fac2949564a12f","impliedFormat":99},{"version":"b5929098539f514464ac3c468bc3c93ea40996de9eb5e770ecaefcfbb2677758","impliedFormat":99},{"version":"2f5685827f73fd674138b81ca48275458be188b01af8a00315231efe74397980","impliedFormat":99},{"version":"6097f1055816adfa4bfc72e65ee4b6abd46d1fbd65bff4c7c7180a031408321c","impliedFormat":99},{"version":"053b26604faa65cae78a2981e04dd917fb068fd367ac39203f02d64ed0ff498e","impliedFormat":99},{"version":"cd1876d05dc46e55c625362f53068c9d4e7307c7e74a590b2975c657a1d3b625","impliedFormat":99},{"version":"d0d95a2893c600f181619be0c35d30845d93ae669e0aee435f9c95f6237c6872","impliedFormat":99},{"version":"1d11106d40ce32966017ecc516704eef10e57f72e376b1ae3d66c9f9cc6dc14b","impliedFormat":99},{"version":"249470a524039cb8a01af509318599f57b0808a602617eb7a655e2ec584446a0","impliedFormat":99},{"version":"b0221dea05831243b8f2fc4da23079128a812a5a5a9908327725c92049c39bbc","impliedFormat":99},{"version":"a54442465e1152334910e6533d10e92fbb8b2d36ab0aeadcaeb3f3f151a8a825","impliedFormat":99},{"version":"49f3ec14c8cc3e911e40adc42a42f40c109d8a62c655cf905d15e235883b369c","impliedFormat":99},{"version":"6c468c7c33e01a672c907bb52fa16a29a930897b4883c895eaceb1774aa3e574","impliedFormat":99},{"version":"f753928cdc4391702905204cb54c5335545c786311c5f52ed9dade3f55040faf","impliedFormat":99},{"version":"d7bb71b8da554340046cb2986dea2f6802436149fb40fa949167756f00a51f18","impliedFormat":99},{"version":"dafdf0b0ccb55128f83fe0acaddadfb5d223887a7e8d59a0623860a68b1f59a7","impliedFormat":99},{"version":"ed54678c383848e12de6b9621648908d63703be33d3d542b125dd4cceafa99b1","impliedFormat":99},{"version":"e742691b51bb384ebe58d525ed1b2029521a52dc63e35d445208a1efffb4089b","impliedFormat":99},{"version":"37dfcf681f7dfa16001120800a5559d418c84bba05f74806169a953930ca1108","impliedFormat":99},{"version":"64415fcb1c664e0a60f10696d10027d96c9810e3412af9972e6a4dc2c2e726ae","impliedFormat":99},{"version":"bd02feceabd8455fae60013855ddfb8976adb97303d8d143b9fbecf8ba0844d4","impliedFormat":99},{"version":"a81510a532b797072381fc8b72287d54595f8b2d25691f793f5d114875282b23","impliedFormat":99},{"version":"8d071caad80707dc1853c718e6372349df8fdd4790ac57550cb243545ac91806","impliedFormat":99},{"version":"7b8f4bcf71399d7bbad22014a4eeb382841c61ad3aa079943ed287598e70485e","impliedFormat":99},{"version":"fc5115956fdfddcf86a30a1ba0cc02927cf7035a2bdc3adbc8766b79242e0eb4","impliedFormat":99},{"version":"6bc0e969085d2ad0696627de23af748de2afae059856a22fa0465036bcf2b6c9","impliedFormat":99},{"version":"dc147a0ab89bc4abf1913f699a9335e98a889f00cda6f07a5b133c5cc3112622","impliedFormat":99},{"version":"2a527df5c4828328fa6b35cf8b8f5bf0640933a4602c517faace7a1c3af0d446","impliedFormat":99},{"version":"ff1f7ea08241096cff8b3116afcc8babfaa1b9e319df043cb4a0c44af8e08034","impliedFormat":99},{"version":"b203573913f773b35d92a3a499a7873038149a35e0b23c7e189d7590b27f6da0","impliedFormat":99},{"version":"f6694bea88421c6d7342b5381b1a49fc823ae746680ca9ee16d518c9c16118e8","impliedFormat":99},{"version":"b8a25d32e4a2a187e2169f0936416cfcac8926f56166f3895fb5f82942f3150e","impliedFormat":99},{"version":"74f9f15dd600e9737bffdc26343d74b2d17adb91536fe4e29a9d110295136334","impliedFormat":99},{"version":"c3789c53874f2aba5a7c21e1ac1e467f95522ba5a0c8f9c8b8c519efa7aec51b","impliedFormat":99},{"version":"dec52a42c912503c35463f974fb86cb1a772cab001c2c9ed413093845be2f677","impliedFormat":99},{"version":"d2a2a7be324ab271073676edb22f5de259d4baf5bad32bd2e5545f957f503ac4","impliedFormat":99},{"version":"8da99e8ca9c8fced530f92f1f2faba413b961735ef92da80c577f981b767e9a6","impliedFormat":99},{"version":"22f897e17f18b702f8aa1c6e6412fcd33d180f8ef61297fec6c395a2b18d9908","impliedFormat":99},{"version":"9b48fb7d6521c10569a09921fea776719fab153e4b24d6bf4290fe6fab9be6d3","impliedFormat":99},{"version":"6908cf62ad2018d33473007b4f5f6c5f097aa0d28505e694aa7646291136dc67","impliedFormat":99},{"version":"2fac6a45f688a1be6081e321a9ca7886923ecfc3a9083959485567ffc38b1dea","impliedFormat":99},{"version":"2f5ff35a589b58b99c7d787c696155959a4057dd3c29db745ab2c0f88cc2e03a","impliedFormat":99},{"version":"19c7b443e13c14613f6cfe274d924597e3bea64375699b98603c40c4c4f3dfb8","impliedFormat":99},{"version":"2873b8fe4083b54fb60dd1d03ee8b22496e41f96a4e536e06cd59a481aba01de","impliedFormat":99},{"version":"5fc9e50135f4163989ce74b83b68a5ee44d151f04ec44078adbe913c8dad694e","impliedFormat":99},{"version":"321c7e382d36a823c6bf9ecb6cc8a4e5bf60265b4b37c86fdfcc85973ede2c1d","impliedFormat":99},{"version":"34a80ad568a06a539e43bde102bed1fcb8bec196811caa9abc3a0cf44a95fdde","impliedFormat":99},{"version":"e0f1bf295d165e3e7fdb6bbd9910888e9c5645e19cb4ae4b86303ee5ba2c951d","impliedFormat":99},{"version":"116418e8039e72fc6a67f90222c77ed8daa944be04eceb86bcf08e721e291ec8","impliedFormat":99},{"version":"2fc5b4b281cccfd2ed90d0384b2fc521dff07929703adc5d373c7ecfbe1d85e6","impliedFormat":99},{"version":"2d99e3afe124a3c40300762492a49425df4b8090f771cac8034e233eed31bdb8","impliedFormat":99},{"version":"85561bddf43096a73eb5f16e829bb4beee1906b56027dc4a9dfdc5356f36e864","impliedFormat":99},{"version":"88f162f40062f4d9db248fed81d8d9258b2d0846ab8640904e220d69e4a040b8","impliedFormat":99},{"version":"df35eb1e5ccd6b597d18655f69dbbe24e8cca39ffe13822158c9756c528faacd","impliedFormat":99},{"version":"30f861484a42eaa6830f91343556e401e0c9399b851f3a017cef5ffa233e8b98","impliedFormat":99},{"version":"af6cb3ec64660a2456997a8c5069e6e344aedd526418d727266807663f21df9f","impliedFormat":99},{"version":"d366ccfb8cb87789f1592b0be7430df7ce17fca8be422033bdf0a8d0e06b7336","impliedFormat":99},{"version":"e243dd83e46a4fd3614589b4589042576f86d4748866b9423c77dee1318847c0","impliedFormat":99},{"version":"6ffba5563c43d43c1ea6b051421d59af7f6d23cc1baa9fd18a99a39d060c1cdb","impliedFormat":99},{"version":"bceb3703983ccb7177c4f8f21ed775c0ae7672559c90059a7814b04065ae04bc","impliedFormat":99},{"version":"138b012318f035855153d24cfd0a266d0aa30cef0565d56b40cb6057324ff8c7","impliedFormat":99},{"version":"a0cf73046c0cbced0a194418eb5425fe8411221be669eda86673ea614f957fc5","impliedFormat":99},{"version":"7924c9999e6db6eb085f843000443a40efd7d30474fd038fff5fa0609994d766","impliedFormat":99},{"version":"91ec9831ad545c826a697fa50de41d6fff9d927b505aa66ed46d4a9263a9ce0b","impliedFormat":99},{"version":"ee76bd42d21021012bd1f11c216c7e710ff11ebf8bf7ad134ebe1ed484ba85df","impliedFormat":99},{"version":"bd9d4c79b48d0f7ed5d5d3f7d06c029599023546fbe37fa33a95b525bb6febd7","impliedFormat":99},{"version":"e3abd62334ac65b6ae754c0950e014c8869c10e280da768c9c66aeafa1fcd9fe","impliedFormat":99},{"version":"fbb2889b4f092db88dc16994888ca046365ad0cd2232d1bdab0ae23db74a8a97","impliedFormat":99},{"version":"b32293aada046eb0690590b723571aba0c34e57566f1067d837277ac5ea5a58f","impliedFormat":99},{"version":"0e3439dca47087f3f3685efd6b47efe55fba142bfba722f84dbd90101ba00f44","impliedFormat":99},{"version":"5482a5b0d560b5f1c22d894a1b30ca37b5a8b509668276d4f2699c097684c5d7","impliedFormat":99},{"version":"7163fb6c706557131bec0a259fa99ef690b9ea74e2c8e22448d53ee767dc0003","impliedFormat":99},{"version":"84ca3c4c8f8b4b3328afa2c57555cfb71deb39053a1b783b81a44c0d183bbe1c","impliedFormat":99},{"version":"cde540ab3aa5b98e2555a87f25316ffdaef51dc2fefa113ed96c47e17e06efee","impliedFormat":99},{"version":"6a0d02ea3868cbbecffc40e314a8cf76ae69b14102b14b007b6d43edd7d4b078","impliedFormat":99},{"version":"76ea95684460d48dbf18e249903aaca73e8b011a5a6bc6e3071e75b3cfda1dc3","impliedFormat":99},{"version":"8fe2df5f35b364703a41744b0263abab1d0c94ec4da0e97e3fb09e7f1f49d1ca","impliedFormat":99},{"version":"cd39f4dc66037ba2987de3a740b709cbb19370fd564d7e8a1082806e52fc7da1","impliedFormat":99},{"version":"f1bad28391c2827fffcd7210a91babccfe7a097ee5c33bb20566d777e4525aec","impliedFormat":99},{"version":"84827ed75fe6e06f34cff9752e36084d9a0d316372b3862d59c487f3608b9014","impliedFormat":99},{"version":"fdc17b998a4934f09616fafd7920df0c0d120e22a805c12e8170f9ac473d3ac3","impliedFormat":99},{"version":"a1b9afa158ba3d9e2ca524a899818a65f4b46c0b7aea5549ac161960d57e33e0","impliedFormat":99},{"version":"3525e2af342b254e263cec0aa825c7bc9fd77de4954d4cd32b0431b0e8fc4fb5","impliedFormat":1},{"version":"8a88236860c9107900f479f1f27927648652b67d6c8a3cf9d3b2e3d863e0130b","impliedFormat":1},{"version":"04123655088b1a0117eeee5a63e4c433c76bfdf157fd26d1a977323df2279ded","impliedFormat":1},{"version":"918fcf5a4ea36dc085da4f323c7ab62d73cf45f8ff472b8bea7db96234709db5","impliedFormat":1},{"version":"8d1a17b7fd8103f40cd859fa317d90182508957fa9eea299300ba8bcbef971fc","impliedFormat":1},{"version":"9f8fdc648eaf1a9e2db3e04e28a7c931f7fd4b343c9a24e35a1c90c083beb876","impliedFormat":1},{"version":"4bdc1052e65bae069341d76188efb10fe7fb5a4a740b5808ad2c5a1afa6227e6","impliedFormat":1},{"version":"02936427b136646c3576ad4a5edafe0246aa07af9fe7abf336fd530af54cb2bd","impliedFormat":1},{"version":"ce527d4ed3419efe88dc25bf76202ef4393424a8ada90fe84bed472bd94d8b1b","impliedFormat":1},{"version":"dbe33e437acf117b99ae63d7fb5ba9a0fd5ea42fd5521358f68a3c512a44492c","impliedFormat":1},{"version":"c26c342dac3d4a22334724962700cb4e12eb3030dfa6edff4f4df9ec5d645605","impliedFormat":1},{"version":"d364e530078696b3c5427f3d2e8dab96f148320c0f504566cd8e113738a0be28","impliedFormat":1},{"version":"a220c7653ec20a4b7706d467d277c16d71738974b5eaf4c78eb7dd056fa49f05","impliedFormat":1},{"version":"ee1495aad782d603e433e8b9dfaaa9b9ddd07851466df6ff6f21ca806071b2a0","impliedFormat":1},{"version":"789289e8b082a79a660dd568a95f91c33f27e89f9c25a9b7ea6bcf687a074006","impliedFormat":1},{"version":"181e3d1a29b3894478a22651d36b8b9e185f677168ddd30abc6bdd2f35f94789","impliedFormat":1},{"version":"a61c701659297fa42871310e54d4b3c081e362f498c97f64abe36190146b8885","impliedFormat":1},{"version":"2ca7817866541fe3f0c54ff923c1f7091d288880be9f5c07fb1d6edb49551f11","impliedFormat":1},{"version":"9464e8b0705443f5e1e2caa3a54a11b6f473c0836284f8cf8516ecb1ec697321","impliedFormat":1},{"version":"eb76861bb142f7c50a48d0dbac8de02f7dfbe2a0c70120eb9e4bfc323416587a","impliedFormat":1},{"version":"39d2b1cb54393286008cc6e0abfef5355a90db0c82699e0539940c3f8520a571","impliedFormat":1},{"version":"b3fa01d3b343a9030cb5db64e92ae8f7c080d112ab10df2fd74749f807860941","impliedFormat":1},{"version":"dcf1aeb81aa85d7a1d8e18258951f8932ca5054b8328a14d57468bb1dc1f5761","impliedFormat":1},{"version":"2705ae1316f3fc38cc78fe19804febc29cd0bdfcf9f2f8793f2e0b5027a03ab7","impliedFormat":1},{"version":"e72bbc7736c1191421ac004198b925090327ff28e361199876f24fda5855d646","signature":"c4530941b012c0d7a83b7276d628527775bb81e1849fac290daf14c4fe46089d"},{"version":"d97d4eeb57fcc4962ca112f0dfb9a92e27f538f1a9d21a7a3df91d910d08940c","signature":"b3b5cbcaebff7b51966e754ab06336e899047a86d26e08517356b7d631d24f03"},{"version":"8856b4d67d1ada2d7146cb998d984f0b11f03c1698b206dcb9b91c36aed2e294","signature":"e0d8a7fc43f9e341035ddce82cfcd39a0b776e89642cc638f44cf4b4e3fdd3c5"},{"version":"07fcc9be98e12bd2f0f71a501a9bfbe2e53d38c50e8a5e84223fdd05bd8749c5","impliedFormat":1},{"version":"8c724ec27deda8f7ecdca0f22d9bb8c97a4507500ec645f49dea9c73184c2512","impliedFormat":1},{"version":"cd29475b77289e166f1c40b7feb1b16a5f9378592b0a37dc8124176768339f32","signature":"c0b38f37240e4cd618ebd958ed1cb05493b5fc5b2f39711e788de5d005a575cc"},{"version":"11a658ba8bbe67ef6152ff43e913d599b4651d3e1dccf1c31c365fc120b983ba","signature":"20854999105f0005cddcc0f953ba5cfb569ac061688fa0b6d639b82e87954464"},{"version":"31e9ea2ee641f899958ed94ada3d2aa813bf3fcca07953a69261e278777f949b","signature":"74bcd45f35bee9ef202903c7a1b8763ba5b69a447b293d14fa127d0b94dad27d"},{"version":"ac5164b565224c5c31415040b2b19412fa6d9c6733c45838ffa3ce2ddbee6cc1","signature":"5eec3e9a7e64c886c613e53f643ae44cbda9f423470ed4ba639d190928343968"},{"version":"fa1692f13f4809ec85f4d01b1989ef0602c250d018c132eb7f605860a9638ea3","signature":"df66b1b249dea1d6c6e26f91bf1a18ddd0ae8dbc3c09b4c0bf6ee28ce669907d"},{"version":"387a04c1f18f4db7115dc80c534b3ca2d4a07e0506da3318c1a6459dd12a6c6c","signature":"f7f7a9cf48bc904ce1a718f5ee8b9e55205d95cdbb4dd46c518d6c37441a1065"},{"version":"21a8e1d3791c1fcce139de5a01be24ab493adb9675085add384aad28b3d8ffe1","signature":"415a8df5c29e85e1e5419b292a5f813754a9d469b5294f0b9698e3408228603f"},{"version":"179ada51f8c945065fc9c5e3e20dad9fb29c29a03dc5a04f0ac9032f4d38676a","signature":"7fe1f1272e840cbb48105b9105d7449376aef8c77f3e2d87060984662f502d93"},{"version":"5b0b9b8fffd1e1c2ab7ad02795083e8a72008f26bd8f6fe16dbf1430fd87a2f8","signature":"a7e86db62c48a2b39695894f73d694a7bd2443c61a253dac0cc2d77dd54184f3"},{"version":"b2f058cdecdbb4aa3ded60e628144b1440cec3500af030b4e6eccd2b34a533eb","signature":"d94e0fcded4ad578d70c0eb77080b70b7f8dae238a5ef1671b7bddaf91c68818"},{"version":"6dc1264aa5ff1b54e505e2decb512978a72d0f5f5a37fdc57f90385a50a8dcd3","signature":"3ee1121d5447b53a1c2eed62537d7c49008861c9ed202c6b8b9dd273f1a20117"},{"version":"99eda2f4740e7a643c32f4de06ed4bb725d5c23e44de1b49140f3faa223f60c9","signature":"5f60fa441decc15f96b74b7468eb63bbeffb2ba3f81df70cb877c2e820c808ca"},{"version":"748658041c16d4180275099d7d45245f545ac1832d50e7068771212452d0dbe0","signature":"aa1725e38b5a0b4009f6a74f296f173e7926c119a60454dbd2e523861735df69"},{"version":"2fc03e95076fed35d3db79314a4508724c8f6fc46a739071f01bc780d9f6e962","signature":"b60a01d239c3d9185ff26717746bd0c80f18d26b2fe62685c3459fc3c14ddbda"},{"version":"13520ffaa95cc1958faaf7087e21721efb7f3b5e3242b42549a59c41f565c9bb","signature":"cdd53006ec1732cc3116831e3323c5409c09172c5f8726c9f09824b60de9137a"},{"version":"c1c50aa2d9472e53e75c41ce155271387544f33ca568ebd9ea76cb7c382ea6e2","signature":"f2ab09d38bad6925e4f21039ce98c6d5d8f919fd98d63685189ee795b707d4ac"},{"version":"9260b5a5d97c00c205cbf1727b0770ae8269c68e2a6a8ee476c96af23eea3fd3","signature":"3c89805c308f4c17222fec206af788efbef66cb59320f35183c73aae71cd0e2c"},{"version":"606766fb696c92002835081bb7ede4d9855122ecb302045ccfcfe1cf5dc48be4","signature":"881665e6e71a7ba80ae766dec2230cf13a8d99dcb9869e560d4e3eb5b37a3ba0"},{"version":"ece3a6d082f09441fc260a5b08c35fed66b8a19195572ab67fb2dda97090319c","signature":"799cbfac346210cf9da62217bbfc293608166475ca139fe782a29862adc3b53d"},{"version":"a80528f52cee5f00dedcb61372bc9c6e5048b2df508b936940ebabb5ec21bc47","signature":"a5450f6bee08e27f93759f900dbef6c555e2cf07df1f92b91de49d257c6e6155"},{"version":"1c3960a022b772cb72484a6f420d3b6a7672b8e0be23052a15fcd3fd65e08f40","signature":"89984d79bd762f5c9bb67eedeb12323a0fe1ac7930005d0f92c2a8a251922118"},{"version":"dd1d50f905a287a8940a2ecf1b1a8001c16ed5d25ab6ad1b74eac75752947e0a","signature":"d322dfe2cee81d50d50953cd67571fa2f0f3e027f74943db55fbe5c276cd8f53"},{"version":"518434a5b2e2453d96bab382c5d9a7814418464104a37eb353bd23ebdc1a6c11","signature":"384403d3fdfe0e9b3cc5ea115e838bfd03fceedd4ab0e8bfb63a5578a01e6163"},{"version":"65c7767e98b78b2e4d37c40e8f58987afcfd58d740e70d15a67a51db20a34197","signature":"913f39bf36325ab9406b1236d001c13865633d849092f41db6058c864fe2989a"},{"version":"139bca1f72a559559aad86b5fcd96a5a1c7fedb3501b7441f4f1fe97d37c68e5","signature":"afc41ce4920988cd093e96f9b6942b98e2c4eb5f510c012070314acc3d1997f6"},{"version":"a2ddb20abe959ad018cbfd00c5e1e4a89e68ae66c39bc6f3808dcdd8ceb044ed","signature":"38a0f816a78633c667d3e014d78926b6fa5d4a00897bc12fc00c5eb4389956b7"},{"version":"cbc24f5033442c22627b4e051029224ac2a85744e854fc1d7a3c9c633c709a04","signature":"a9df2d67d817cca4278e54cf663827a674c30da76ae9cc6154a3149e10e711c1"},{"version":"3ebf65bcb7a8d80837126741d52b14f6ee77e9fd0ca1045208b111009eb53cd8","signature":"762c62245e9a8dae79add8f91400b2bc7e3e82b98442991e7270a906086dc7c1"},{"version":"6d83a3748d0a806ddefb2aba85c50fabb1ec5a9d8f451ca375828904cbe6150c","signature":"39ca282682a5a8f47636129b32a5bfc0b4a654c70114c78f3469b07c91a68ff2"},{"version":"9bf437ea6edd982863443e5aec5df6c6fa2cd330a1487f5dda2ad1042bbd5a2a","signature":"1afb316130cbe631c84942892c4a8358b0a47aaacc70c2144cc4f533489f0bfd"},{"version":"908d2db031e83498e0d46b7343f7a3279a9dc9fb2ced88b38a038618ee4f1b31","signature":"451f3479d02d43e0e79477ec9c46b93c5e12a9ccba228bbcf1831283f9ccb47d"},{"version":"c6ff2fa2c5287952edb6090c2b5709f40017d88f294d7854f8a476b195f01c08","signature":"d48128c34d07d89c49796487db7d009bf76d59e2f9a20ac1c7d1a728045ff000"},{"version":"ae6286c1b00a21dbc1ae28d5581d9f3e66e22b2cce3bffc394cf038d668e98a0","signature":"ca729b2464e676fad616818d10497f48accb4f90dc894126e03231c930c47eda"},{"version":"ce1571ade75f58f91bfbc8f38569e39395fec4ea5f0ac03445a2a0fe0966aa81","signature":"c55148a0fe948b74b87bed582de626fcd7a2cb4ed8ab606dad31782c9dbc39e4"},{"version":"97c4a6a478e39b86d6b3c08ef6dac6ba69d70ef394070c0014edf11776d48bd6","signature":"b4207dcc57c75798f3a0b9c393ba66605bc15e6abaeba09ff43885ca10280e10"},{"version":"da1e4a3691489cb1594532dbbd3826337a9b45b0d4e52dd7a0a1548e720f9ca7","signature":"425c29531872221eab92944fe11bcaf2e1be453a9500e4251530072ec0e57446"},{"version":"0cdd18e25e7a7fbcdcbd2e92a0fb422c0f0fd014ac84c1683f64b8735b9ddc68","signature":"31d3584078b834e1edf25d95f336923fcf9452b3700b312c1efa916d3b2cb7e4"},{"version":"a54b8566724e29a7ba5b63bf9e9b1491b0cfff6c9bc76c4c71dcde9f117450f6","signature":"3bbf7c13e6e7b0cd2eeb4a6f2ec948a5d8c278dac6098929d85d0d5d943fe1ac"},{"version":"fb893a0dfc3c9fb0f9ca93d0648694dd95f33cbad2c0f2c629f842981dfd4e2e","impliedFormat":1},{"version":"95da3c365e3d45709ad6e0b4daa5cdaf05e9076ba3c201e8f8081dd282c02f57","impliedFormat":1},{"version":"7245fa556a63e4c5997642422fe4250df16a56694cc1a1f50108dc80fe3018ba","impliedFormat":1},{"version":"7c14e702387296711c1a829bc95052ff02f533d4aa27d53cc0186c795094a3a9","impliedFormat":1},{"version":"4c72d080623b3dcd8ebd41f38f7ac7804475510449d074ca9044a1cbe95517ae","impliedFormat":1},{"version":"579f8828da42ae02db6915a0223d23b0da07157ff484fecdbf8a96fffa0fa4df","impliedFormat":1},{"version":"5f6beb8c43a52e9119cf31a7583ceca88dc910d7de29caad1f81e98497f808ba","impliedFormat":1},{"version":"3ae3b86c48ae3b092e5d5548acbf4416b427fed498730c227180b5b1a8aa86e3","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"3fe78931599e9a87799169632a3201498ee3140b6d2d49bc0c6f6732669f4fb5","signature":"b16eb023e2dcded7c29dbb213378919ed0e8e445cadf613fdb02596610c9c6a9"},{"version":"8e9fff65aca0168f12ed7fd6d885d26032dbb38cc7c7b06c4f76115703c51205","signature":"5af085ff0c4546a4be375f08f7ffb02c0db3c3deaf892a7776f52efee803c32f"},{"version":"f43da3b91f9dff78a1a6f1774f23e9cb27b1aa04cccac8c95c8b500115df15ec","signature":"ca4eb9507c795f47f224720f007c1b5286c851f66de6e891ba79a0ef517fabba"},{"version":"aa251c3463b1a85dfa60d59e7da1e6caefe8605fe7ca0ed269d0303559de353e","signature":"38ec0037c05ee1d3f2ad741c36d95e86437d3947959b63bddc91be7dfdc61497"},{"version":"dfd7469e00e6fe7e4636d1abce926289e5edea64da20646d1706e5b3e66b12ed","signature":"e046f68ce26e0fa3df0026ecec113350b6709b540bafebca1e8967ed98f0c540"},{"version":"4fc3365aaeefd0b401b7693150e2df8ca8df5c505c2a03efff2b290dfeaaa1fa","signature":"c97d22e4b6877a15b3cf4753987667c9924feebe8d7bca6ed0e80aaa425c12fc"},{"version":"699def7c90a86f184017675360cca68a613c42a27495427fdb4fd91b5d6e7d6e","signature":"9daa9d5ce742fe088f1f20d5879e283e5892c0cc18a9403f2c3c47d94a565689"},{"version":"1b0054bbe9fb2331a3bc4ccb85368e243bfc4066aea1d5faa5d68fb04d6a5c4b","signature":"73ce20f1c08a47f1a86c278b279e8ad7e4afb66aafbcfd6a69b6213dd2ed26c8"},{"version":"6825eb4d1c8beb77e9ed6681c830326a15ebf52b171f83ffbca1b1574c90a3b0","impliedFormat":1},{"version":"1741975791f9be7f803a826457273094096e8bba7a50f8fa960d5ed2328cdbcc","impliedFormat":1},{"version":"6ec0d1c15d14d63d08ccb10d09d839bf8a724f6b4b9ed134a3ab5042c54a7721","impliedFormat":1},{"version":"ac393d11e2c585763ce7a8b9118ba4a809cc19f9bf6d647657d38268ed5d3b56","impliedFormat":1},{"version":"b61028c5e29a0691e91a03fa2c4501ea7ed27f8fa536286dc2887a39a38b6c44","impliedFormat":1},{"version":"a4bf154e0f9d56112713c3a7d2d60c85d667cae17e69f7869a32578881b652a8","impliedFormat":1},{"version":"d5f65e3a5277cbd0b2c89da26703c5879cc428da7ca816d1d1fcdfd7c0a2500e","impliedFormat":1},{"version":"c784a9f75a6f27cf8c43cc9a12c66d68d3beb2e7376e1babfae5ae4998ffbc4a","impliedFormat":1},{"version":"feb4c51948d875fdbbaa402dad77ee40cf1752b179574094b613d8ad98921ce1","impliedFormat":1},{"version":"51d4fca2239d818a6254ba46be06e4def3be685ec034e9255cba403d3b27a07c","impliedFormat":1},{"version":"b457d606cabde6ea3b0bc32c23dc0de1c84bb5cb06d9e101f7076440fc244727","impliedFormat":1},{"version":"859cf43771b68e589bb12c6e5cde3edcde4b530c7d324f455af2b9e61d4f4768","impliedFormat":1},{"version":"9faa2661daa32d2369ec31e583df91fd556f74bcbd036dab54184303dee4f311","impliedFormat":1},{"version":"ba2e5b6da441b8cf9baddc30520c59dc3ab47ad3674f6cb51f64e7e1f662df12","impliedFormat":1},{"version":"8ae5f77a947f3af79012421d027e8ccfdfc4b004eb753e77c26845d4a1907de4","affectsGlobalScope":true,"impliedFormat":1},{"version":"be59d2459639699fecc0167ea82a9dc092e0c11407967cc429da602f4709650d","signature":"bba101a038992b84ab0b413481f8aae72ca13b1d597c86175b3baa094e2389f6"},{"version":"8a188784e101baec5d6b181416b13362d1613a58f26e51312c17d26c1966a72b","signature":"cbca3c6622695b5b7c1c5e4993b75abb45f167ce09f65e7fd6d47d084f7d712b"},{"version":"b9c5e870dd0a82dcae967d7989c8cb87002fd1239d19d35db4c5ef53ff735119","signature":"90ada2236f524c0adab35ff74331789cf617dab778c63a8ab7b80d6b8ea6bdc3"},{"version":"6b5df98ce7163b028594ef2fa651a009b510911d92020fedee22e3bc25053b06","signature":"41e6c0c32d16a64e605c808e80e5ebdbb68526838f972d44e790c141fab0bc3f"},{"version":"37e34ba0ef699bfbb9d213e9740e70c53e2b860773067bd545627ff91e278df0","signature":"f1d0d71c19bb8c2e4853abb23b8c8b19f9a3a4de3d7f95eb5f45ed4d34f45fc9"},{"version":"74db3ede6b5017bc8d850b3525f52c9a06915b8600560ceebbaf3cb54a0b067f","signature":"10024dae769a0dc666a7ee115de87ed6e4a4046404111f8bf1f5e730efc29b17"},{"version":"f2b1bb4ea0885c4a110fb3868afa434d999b28e8a350c20db81f94ddf61c81ad","signature":"91f6e507d45463a1b1e565888f74c0057d18a7ff86f99fc44dfc4c5fd04910d6"},{"version":"03c92769f389dbd9e45232f7eb01c3e0f482b62555aaf2029dcbf380d5cee9e4","impliedFormat":1},{"version":"32d7f70fd3498bc76a46dab8b03af4215f445f490f8e213c80cf06b636a4e413","impliedFormat":1},{"version":"e15a41e1ba2cc01e27daba99e95b243bd689708d477048350cd47ef561fc5da7","signature":"3b01781a9e34ae02723653388dada049e995fef80515348e82d46a50075daa23"},{"version":"01c95aad8b2a98ffb6a8c54e4aac0f1334a7a6131f5fa6abddce8d4a0fbd5cf7","signature":"e070d4dd9a6b10e7877b6d221ad76071f46d333683c7c18860d5ca152a0e2cce"},{"version":"27cb5c2735d297ad5f5de188aa9a78e0279804c1432aa25eb9ffd2687691da5a","signature":"2973ee5bed7ee5e2a57d4bf7d3d76945b1e1e50ab205239d37e12a9c8af98af2"},{"version":"c9f4bb3c2e43fb6702630a036325ddb30e92fbab0634a47fcc2916c478f88419","signature":"909e294833009d798301ca6a0626731ebfaf12ad02af753ba4ff4e8f3662a643"},{"version":"e5cbf9ff25224725ba6af3f191c3eb6cd0838286e07cbc8a7c68f0f793ac7a92","signature":"ae8878ec8062bd71ccb460666d98326fe08027eb2b5ced037974a6b6bf3d413e"},{"version":"19e694e106e36b452114a46e30a9bab2afcc316bc70e2a34bf6aaf0c5fa7efac","signature":"3956019096bc0ff2be1c900e5d69de15da5512f7267e02565843cafd682d891d"},{"version":"2b41c27f47f77e11da12b987497fa5550a4f3c4cb70de1148d0402db1a22fe02","signature":"be4f993bbe78fa9d29b45ecdc0cff6627e76b4637dfda546ecc293eb6ffa6730"},{"version":"56c8bbb465aca4cbac99984b713f48be67ed0246f7cc2b5353ab3e8975df4cc2","signature":"3f2e7e70b3ab593dfea203536f1c3e8e5a16bfa97f3ec17b689a5eafdef7039b"},{"version":"8d4650f9e7660866e50a2152eac99f46ea1769adcd624de9ca84e8fed9bd490a","signature":"a28b5c0c372fb375910b3fe3c3ce4331509bc18ccef7cc39c9ee9d8daf8225d1"},{"version":"6e26edb34a7287ad977079ede7100078b7592572d58b303a7646010e4896c2da","signature":"261e2cea40b6dd9a4b9624c6fc3a232859da20ba8646d4325e674472dc578f25"},{"version":"19ad10f2a123cb013b5836b4ae429bd8fb5d0a1a04edcd8dd097445914ba33f9","signature":"96f7d4d6bbc6173be7766d6648c8cd33d18e2fff000b79dbd4ea384f8c1ecd2f"},{"version":"7a8e4c42aab782d9d924be341c5c92507e0f87f3af52778913215a92188d71d3","signature":"9e570ff52fd9e656c7b237b7b838e889579c9c4bb388b2bfcb9c09450273007e"},{"version":"3c2c3d48637dbefb64982ee6f8a11744b7ad009bc2ae387455e0c4eb501d9796","signature":"e092af7fb6e4b96cd1137ddea63e6d9a6d608afd32a3187215a621746a9a633a"},{"version":"7ab4db806f38fb47aeb2c4cbe59c069f0649c30db2b8a3459682edaaf831024c","signature":"359a9834c6863f2a08f40ddcbb6f0958bccbf82de5c6bf53811dd6407ebd5419"},{"version":"2f2d624c39d3081d2f96225876529d5d622b5c6334592adb8aa1b981b5d0b194","signature":"7426b3600bc944afa9a51666c2b3a188a642a2330b0c373dc3a20d47cf0cf93d"},{"version":"6329ce71b3938abf748f3028ca7fb0e09e0cbb34d6f8348f48d152711a6fb242","signature":"6e0ad30b2d82eaf5888f9e89904c5013a296353257c31b5d761b9774544ab13a"},{"version":"cba38a62b45c34e633a637dd4e4ec62661782576927255ef1607809ebc9b8608","signature":"5075b66ac2a382c955c573fd01ed0ea3f511c9005699a8c5b4c2c3e5623c45eb"},{"version":"80f01cf889a8d05256332a6aae2f6037ddb35b905b6c65268f5df3f9101b58b3","signature":"1b14675f474a9928e316e35c16e4ab635310da112a018d915a0c640587eb5d64"},{"version":"d8f6d6a7b601e2d9959e6083568caf771dfd553471993130dfefd4091b142c96","signature":"1e8632545997bd9a477296ac87dabc2b24a32dbe6b4eb033295d932cfa766139"},{"version":"c66dccfd3c3b4a00bd0b1c05e4538e69c998af5ca05de8a22136e32aeb676a04","signature":"c78ffaa29f6f15a1b1a3b638410098449a33cedeab9cc2e43e64e1dc0813c7d6"},{"version":"57bd64cb1aa0156304ada6fa50662dca6f96ee256b3e4675826677241c34d531","signature":"908ae9ddde43b8cd8bebf4489fa223738936ea291a2c22acfd9d1f8e7f80c4c8"},{"version":"6f69c10b1f3f0ac3157f88719a16edd13d718b2c2aa44d59a9483e04b05e113d","impliedFormat":1},{"version":"480733a4c4383460d313b95b0c7101ecdeb57a3333a2569a4257e9e98393c4fd","signature":"bbaa65c0700c593ec4a3da26bade66e16ce922979ab85a4084f6ff7f3b520678"},{"version":"588bb65eb7d39a7d6a67c60d30616448806b0190396434fe1db9aee0d4f6b0d6","signature":"5d0d590f746af3dedbf313c78440fa46d822854c3fa67ef9c218f3c8827b5574"},{"version":"9c9627bde17c3c5ec1f536cbd0991e3cb971723cbe53236cf9d928596bc6da43","signature":"15be32f96d4a3629c4944349afe9c859ab61ddba7dac406682d9319d84f9b0ec"},{"version":"72ba0882777588a5ba2e2a5a3e336365cc7278226442ddd54f0f3b48fbfe6fae","signature":"1a2eb84c78e1d897a6f2d95cd9637229787935f853e56876859c0f0d1f0659ed"},{"version":"7b9bad3367eda5929b4705eec953583f947802ef229e0d2b6b8139b8dbe2835d","signature":"30b0f9c6d60d2428dd2526178858445f927a9edd7f425c74d11528d02a4f2ac1"},{"version":"4e1b21543e9243b273b0c7fd4976d172d10382b028c77bce776a56794ebfd750","signature":"2f055c81a3f7929eca0093f6207d35a082c350b91b2528d715aeed583cfbfb0e"},{"version":"7e0d54548cfd56010a36a4dc95c8927d66c73d941996cbb3d677559d10bacee9","signature":"ce22bbae73860a4ba092a237e0d9bc250cfb33bcabe8d9f23aa56aa6e310bfa9"},{"version":"1f5f008391a41c88087456ec44f614aa656ec0a38ddd08650219f67a8522fcbf","signature":"30440f7a149a54ee3021ae0f69eb182278fbffb63df7addb0555b7b4eff32f7d"},{"version":"09bfaa3c86e4740188e3d12ecec108147800142639eff3529b739b6d0150877f","signature":"748e5397bf2a4ea2ee29e4ec0f32c4337cf98a2e85ac8fda67f1a6d5f7dc474b"},{"version":"8fb4a36b2ed3fa8fa3fd468606332df2d71d62b965c491ef811d1864e43f4488","signature":"09d85a392165625c0aea7b0a56d34c0857d4ee853e7ee4da9a46b28da69d04a6"},{"version":"71df2c0931f79d550c8dd84ad5a06ebd359927975eac9d07af1e5dc278da09d3","signature":"976f33abed835b77161cb32835b299e80f6a2ef8bdfeb90dd4d854aabed86a9d"},{"version":"316c632e33b03960a39a36e5b9b4f31bb60155ca13265eff03edafee5d286610","signature":"3c6262a4d8eb650e807b4b553105c5f6d869063d0887bf721aeb3a6c3c619f7b"},{"version":"76cda1a8e2c56f8b4524c2e8f2c3d350eff94598f9a9e6d661d5ac2b29c3bc11","signature":"f6912e40e665895045a73b2e6b93a58bfa31b3323d5b737f92c269e02ca9689a"},{"version":"ebc99fc0a22065072427f9dc94277143d1cfc31b14a7d030b30534a8412d0d0a","signature":"edf83dff85d832e4902f4eab33d4424c527e9b4307dd44680882066eef783d0e"},{"version":"98862adc6a78a8f1bb6bf01ea3ed51b852bccfbd51bd6ae11d038e83d3641a85","signature":"70d1db022486250b03f79610c8f88fdfb0ffe98827f9fee4ae2be14d1ec877aa"},{"version":"d2baa4a1cdda89be766866ce20fea250e29e012b8018248165733a6bef24ebe2","signature":"c67ef206a9c4ec40e014f8aa048f54c727b8122b4366be7c4e44542a2fdc4e00"},{"version":"48db314911db3071cbb2a534fb66110d07ac8359ce411ef34059d39d29381758","signature":"bc4a6078a3dab96ee1d434bc34d60ca67f620e50ab09d3d4a092c73660d0c68d"},{"version":"323a932dbc484252c7391769a84ec88a6c0eb86c433b30c812f16f46f3aa27f6","signature":"06886c684bfd7effb79e1d592af0e0f2fc8131ff3211b3376fe95ac598cafeb6"},{"version":"48aa64f3a6d5d1d6ac68c20a914a3a665cecd66e0f33b00a5f072d3d81cc76fd","signature":"cd00f2e4b02edf0339216c6c93836e75991d1b8ce3686d428425e97fe26adad1"},{"version":"77502f5f496a08fa40118237faa562edee65cc82cb8ef7e1ac2cab119c74d18c","signature":"4e212c196a788b899aaa92f721a2665b2dd98c7f7a6b24cda26c2e3acdea781d"},{"version":"33bcfd78e8205a8dc3c7a14ab8fe248fc32a239f5b31ca7380a9b7fc0490f6dd","signature":"27a7419498d16d4e9928865a92b8a2ddc2b634a7494cf01cc6c4c0216d520c82"},{"version":"bfb6481673411532d0fe68fd6339d1c4a39a487d3ac76c78368c6ef4bc43761b","signature":"be5e1190302f00fdd1b55dc1fc669e4d373d1551c56b28298096eb5604cb24c4"},{"version":"a8c4f72b4796ad4dbe2d7a1520337bc9c38959cd69de74c5c895e1929b86e58c","signature":"4b077b5050e00450bda9dacc5cc837b5b92b3dfcafd3ef95b3551b68fb82c2f1"},{"version":"65cf50a184c30c8bb619109c9d46a6716385357f95c21a71450d8e22688cad4c","signature":"70a371fb97940a235ebfedb37c0d4e85abdaa66b0f1df90334e43a6b96695bed"},{"version":"911deba9657c167761018664b6ae140bad06e126986d5b0b8f142c2193d0f0c6","signature":"e02baa0b455824f2828ab7afdb31801171527c68a9fea42d3942d3357629fb81"},{"version":"82e87586160c9f2a031e7353b76aeb6cf6a8449758bf702de76eaf772664384b","signature":"6441938909d4b8cd83df3b84fcd33d4db20c4ba7c026ef9ec30a2dbc9b0bfac7"},{"version":"d9bff0783ea0a1cc126b2edf6d8dd2c583eba9e3bc2b6812ff09cf14a0794a5d","signature":"5e823d007cb5a060fda4839ff68bd613770f0fa07eab1c48ef587d4b5ad64274"},{"version":"3a5725716509c6458d3757377ec7aa77d5d8978e9e20083bf7b261859820d4c3","signature":"6b15dcbbf2c61d17518e66881e7366cdd5054f214cfb4495af422c1a67873c41"},{"version":"900cf8dc66f880a81e4bec2bf07d15727bd3e4e421fa936f04ddba124c76f172","signature":"87eacada476f4d66f7813a50cc52dbea297c5f64c8075737ad14b7ccc3dc56da"},{"version":"86be61c533ab46470c16bfcf57ae875cd718fb836201edaf427825bf18a51f7a","signature":"0cdefdfc80a2f3c6d476350defe3adcf8857f2bf9779e9b25f065457e0a0f4c4"},{"version":"754d535e9b08433f13d47b87114c5d33176c3f1a4680ebb542624d4aeb16af66","signature":"67e9071d216cc0ed783906aa5b47d7155db0d6bd62ecdb7b4e6353a5443cc2e7"},{"version":"49429a32099cc0ba1a4ab35b138f2f60d541fdd15ff9075fb56527a269a0f1c0","signature":"0fd0f4fd4dd737a4372d3c4b1f9fb444b85cec8ffd72121e38d7a045819d20ed"},{"version":"9ec88f150d9513b0c071c98c42768dc04424cfe92679bab8603f8fc4bfb644b2","signature":"5990b4f4aada8a094e437c81690c081b5495c163da4f8f407c78f2dc7e8d87f1"},{"version":"a4e9e0d92dcad2cb387a5f1bdffe621569052f2d80186e11973aa7080260d296","impliedFormat":1},{"version":"b2dac7c80e9f6c821024e635ffa39f2ab6def88b2d26072dd2915b29e5802585","impliedFormat":1},{"version":"d0fe3f291ed904d59025ef05bf98f1226b8814f924e6241512e21488b03d4cb7","impliedFormat":1},{"version":"8cb5dceed5b9fb8717b93ece4ea5b2adf3fe317d0f01b7278e5d55f19a7f7e1b","impliedFormat":1},{"version":"01329ca0b974c12cc8a198ee6e0e7c8cc9c305816dfdf0e7d5d08360abc34e13","impliedFormat":1},{"version":"f8b0e609ff71a048d86bf0c22a5852d80a762c1f3073fd87e4e24a748e645d5d","impliedFormat":1},{"version":"e24e97519fb282732d44ac765d18f90c1022927a952bc624d58fb7ab2ea11992","impliedFormat":1},{"version":"1edb00a6d353c9222891ff6cfce4dc70fe7cd5e1820456cc7e5c427507f39ac4","impliedFormat":1},{"version":"fa5d0a3ded577f413e3e1bd04e59e2b1c0acaa826bdeffe138b86b513a5d9de4","impliedFormat":1},{"version":"441bef2be624d8ab826cd8bc5dfa29d389b83bfd6a6f026a9d8cf4c7fc6cb20b","impliedFormat":1},{"version":"d877fe18289a4578fb34ea19013e3ef8fbf0d5c7c91cdef9bcd57e573874612f","impliedFormat":1},{"version":"6aa9c6f506b53c3a2e17006fd9b9d518f75506394daa687a3dd5e48f14fb8c0d","impliedFormat":1},{"version":"2e0b7b4d1db2d8031ef7e7f0bb8caebf0c3a5fb068dc1e79d28ff5c981185450","impliedFormat":1},{"version":"ec3005b118e0ccdd71203d774ac3557ba4692c6d2f4b7be802dfb8832054b743","impliedFormat":1},{"version":"b27065cfce873cd68383d574d64c347f473c25dc4619c5d57428db1758c17fef","impliedFormat":1},{"version":"4423f0146fb37cab9d4a722a3df77d195a81412bd9f2bef0f927469ad3d07e72","impliedFormat":1},{"version":"81656d14d40c277b25b02c7b68826b2978064e9eb7c5288e83d1c1941f118cc0","impliedFormat":1},{"version":"30b93c0cab303910b02829a4c6ec32bd09a622089d5f0cbc07782948ce8954d0","impliedFormat":1},{"version":"c77b7782fdfc83af4fbad71446874183d3a89be9a9d81b8114568b2a3c8e3ff6","impliedFormat":1},{"version":"2526683537bd5270fc8c828283530b971ef20fe613da53f50c7670c8102f8f7b","impliedFormat":1},{"version":"0dfcbfff816ec838d0610e07bd6cf004158cf5e41a5e046d99cfcb70b2bbc684","impliedFormat":1},{"version":"0d0c9f06f0efab0c13c2096aa9717df8a8260e6a5c804d301c605eefcfb193f6","impliedFormat":1},{"version":"11c0df3d2e349a7575427aa1a6e391a5284cad25292e4cc74109a1bfd02765fa","impliedFormat":1},{"version":"9dacf04c9c542d2969038334981d87468b87320e99e8cfc203a7e13fbea48231","impliedFormat":1},{"version":"a0c7e388df0028192f174dab2f074c1cd7c8a79b56205f0c0f91294915d52df9","impliedFormat":1},{"version":"a3bd20d0262d0d4da24b67a38629c92cdf1e0c67d610550fd0c40c23c2c5a331","impliedFormat":1},{"version":"6311c547c0230efd5ede4ece1f4ac5ecd983c3e01073bff7237324c47ce0c3f3","impliedFormat":1},{"version":"32e3f90d661c71be5aa6fac5a6e3b531b2bb49694c724e446694fd13980c9e66","impliedFormat":1},{"version":"986e6dcc124af0bea9d3437b3c59afded8f7a1bed66514e0bb7924464a0fb992","impliedFormat":1},{"version":"40e047c6d798ffbf2b769e01bdcc7a7b8fe8ae49b3260ed5151c623d77c2155d","impliedFormat":1},{"version":"b537f57f873438e6656c7f162939cf116a4fa3575d7a46fb4cb6c0e0fd563b5b","impliedFormat":1},{"version":"7af11204419c230069f9bed9b3388bfe50ac032a91ffa3501f8b96d0593bef8a","impliedFormat":1},{"version":"0f2fb2612fb867967bcdab5ea59bf874e19b3b7a8d31e95ae5c49b16b90ec825","impliedFormat":1},{"version":"4eac8a79f63e27930d509fbaf614cf0c779f3777d23c8a06023867997aac09b6","impliedFormat":1},{"version":"f3ff1cd0b656cf7b78c2c166c9bb7d4d2be0d6509691c64a7ad11cfb9fb65ac4","impliedFormat":1},{"version":"d6a87d509be5c033adcad19dbcabca9fd4ecd0114b3f44e45d8ec75e9f392334","impliedFormat":1},{"version":"6d9fabbf693d36d0281a389a13862ab2b20d2c596292fea0f884dffc8acacea1","impliedFormat":1},{"version":"479a80820456c48c5e7d1b917bffcda72efa4fc93b2157b72d01d53f6e524f8c","impliedFormat":1},{"version":"0ddd21a422292a1433c0acb0953b95dade7945db6ad10f47f66dbc3e9656ce77","impliedFormat":1},{"version":"259c2e20c17b8884c5854ca8e211abc423229dbe3ac4f03fa0c7c29bdd3c5f7f","impliedFormat":1},{"version":"ec47dba399069e35052531e38011dca52fe56de0ed8bd96f255e05b0e02da6d6","impliedFormat":1},{"version":"cf9c2ac8e0974422223b788cdf400e34c7d9b0b2ccbfcceda7084ff0b55e3048","impliedFormat":1},{"version":"86f5468cde4828a20f2dec5bdcb5679105bbbd08c3e49c9f9654af190ffaf32e","impliedFormat":1},{"version":"864a1d1df8b3d7882ec6f7050b26404ebc3e4bc9a4187f39a91aba83a13fee77","impliedFormat":1},{"version":"8fa762cbd08bb96cc522d04427ed1dc8af9d584cdab0eba21f33898bea6af91c","impliedFormat":1},{"version":"05d2e21a179498afa4a9e822884830a93e3e43f5289bc1528a5910f461b765d6","impliedFormat":1},{"version":"ca987d92730519fada583cab88c43f20798223b2bf97b37a687ec56c962c30e4","impliedFormat":1},{"version":"d2f787dc4cb4d048b9006a29a895f1d4d45c7777be533e61cdc56408a371f6a9","signature":"2dd5b2ed6ff6ee09e2641005c07b323c6b0ab977d0e9ad21a14b8db45954a736"},{"version":"fad7a7839c981951976e28e9c1489e9f03480c3303c69d5ac7e01fcaf8a616cd","signature":"1211d7b4176c86eb158ee81a0cbe2a4ed829c526067b15fc6ae9776debeb098d"},{"version":"2f94e54a3ac48073dc6f4573ee7c1caab3f2ad82f2e25d44d24e930836685944","signature":"198facc3492d79421db02e1ef6110eb2af98c7a8f2bc6bcf34c06f44ab26de88"},{"version":"b76da742601e9ab9ceac45c0a096701ae9722aaf4dd5c7ae55500a7ff5454c92","signature":"60f7ac4e18e23f5e8c4c90d7b4eae0437398edc2e2620c68d207c2163e9659b6"},{"version":"ee0b040acf320aa80ffdcf5e4b9a09b1c53cae381be359d16bec07695b29c3d8","signature":"afdf695780d8e346433aab7fcd8d19d36a0620717b50b6b88a15c1ed5a2f402b"},{"version":"45e76c13101818cb462d285280a045bba68e32cce15c9ada2d090cb1457c08cb","signature":"c12e20db120acd4fd2f63db15db82ce0d7cbc3cb14c8219c64305f354add1ad8"},{"version":"1579400471b3adf222ccd1235cf358d2210daaa6f2bb5b9e954c5db415151d87","signature":"efc1efd69347ec40895b25fed374e753b693aa08a533f8aee967d133fb528a4c"},{"version":"4d9a2008b44a8ee72088bab2f8c3d353e786e7e04a3d70eef3951db65af9f3ca","signature":"e14b07fd6651e5f4192d1aae0b5046461e2955482c2d18dd11ba1136028c8ba6"},{"version":"af1b2345b3a5f46d7ace10e2a42ee4d372034d92f7ed2206f11465a68d2ff7de","signature":"70ba1a5eeddf6630760c735f40c50a0f9be823f7ca8bab4e5e7eea7719f0eb26"},{"version":"748bbefbfd31d51aeb6af3ccf12b65a3453845992012d6008048d874e7136255","signature":"9c1f0c2fdfe9aaecd85f6e9ed65c8b30e58271c782d82066749144ce429bd128"},{"version":"d949a55cf175054b6157166e3c2b40f2fc3cdba794a00fc506430e6602c0fc96","signature":"f17839c2e99bd6af0b593d557d31e1aae8f25b0422fa6d4d24c6a93253c0110e"},{"version":"f2737b56ff2cc7847064ed7528da1a801a09b6044ef273ac07a973ebf3cf99ab","signature":"abddf183e3e424d8a1a2206534b7aa74f3668e3bd085a60c3847065bbb598cfb"},{"version":"84757db16f99df021d1c1afc2f23c05fe81c0acb433b6d8418a2704b939c43c9","signature":"3ce9d005858cf2d60386f3ed42ef88ce23916712ba78c7dd6ed4f97e538d76e6"},{"version":"631143d056c8bdcc60a0b4e487d1d1e20349b29a76d1a15d9e606ffe241be3a1","signature":"453386a654629f3b80e5b8ffbf9b1d0383a4608ac30e327109198fff47e228f3"},{"version":"e71d18f8bcf460f8a55a74262c23af797a3dfc68d84287e6354b90b98de6d73c","signature":"dab49fc848d2b70a4ed2e36fae4704e8dd96d9fe2b44f658cb7a8fc55ff88c69"},{"version":"bc977ebca0c96bf131dc18d78b5094b1af628f924819f69b4f2b5aa512c5525e","signature":"5f0250b0f68b75219218843b59320a2625179a54405cff10b07f034a759f6689"},{"version":"95a2252aba46e78fb6409f1965dce409366f06e2aad1b9122fe742eaf2619eac","signature":"5210586e840554711aa9e99ddb52a060dda9d91a1d8357d00a22880f6539e4ff"},{"version":"8151eb7bc3bed9604df1139be91d7cf68307f23e0c5bc438b407e9a769648768","signature":"303b47c7f772117c8b1365238518df88d2a0fff88afd306ff9835dc03ea5dd0b"},"3734bbf2c46475f0313d00db82e0b42c718f7562446e40d01e302346641facc8",{"version":"720ed112e29cd29a9c67ed6c88dd850169cbcd48d4cd2b214fa8d6a26583cbe1","signature":"b4bba6faec8291d33b8e504450b5ec4cc8b3246bc428a1528e3fbea78df2981d"},{"version":"5b89c7ff9869403501b6c408c02e1ac517b6d118fe21481831ebe749240d4ba6","signature":"5627a9dbb959953a12af3bce13ad3211b350db1430c5f7b844244684bc938b4e"},{"version":"ab4e5932133b8b848c4ba590dc6a721b1c34298099f4b09e63a5a91238734d6c","signature":"bd076080c93a258b398e9ea8ce2f78fe9448ec0490f3faf56d2c1cee8f2765b6"},{"version":"ba2b7fee881fb14763da8243b3fa24061121f874270fd9a3c5d296534d128097","signature":"38c513d5046a94b60d1a1ae83efe141c512d3890414ff50103c0645d9e8d1d64"},{"version":"10720de2cc62fb0750929df4dcda2a504dcbe844a902c0e1c1ff4353097c6987","signature":"e61a7e06ffe58b8b1df2cf0a3818c13bac6d5a767415c47994d64e6da6637672"},{"version":"776e47d4a7ac22ae8369d2697b21f6f0421fa2eac5e04aef0e92d07e173cad2a","signature":"3578ce5f560a72ebffb33afece7a97f7ab075894d2605be42082be5e98a2b75d"},{"version":"332717be8a6f579d116f24026966eefdf8e57023e179ae2e6c80c256a970d938","impliedFormat":1},{"version":"8c1c61728ff828d1643a656559d5466e8c9ef9234bce65a21c75d7f4d90b2b9e","impliedFormat":1},{"version":"4d03adbf48a9a0f36d3f9ce33b968ea8e0af07e32333bb5b1dc106b69ed9381a","impliedFormat":1},{"version":"351299cadad07cc40dddcd6bfd60681de6e5ecde9d84e4d2ba2303171f5b706b","impliedFormat":1},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"041fe7f6b53b6cc942dbc60aa9a0e0680bd52519b0a63c01583ef65f4068a3c4","impliedFormat":1},{"version":"a5579f898b9e47423f675bdadf4148222f28c08d2c906c42ecc8b8ef4c897dba","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"4a1d973d2c93128508df27876bcaeafb2d5b2e3cf6e84092b6fc8608939b1287","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"aa5524e0509c2168c9493604acf51ef97d2027f03f3b38da097802d3aa719dc8","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"d072cb61b086eeae042c45d85ea553a03b123d3e27dbac911faa1a695f5d6752","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"299fd0c281633d8dbfbe5f44c5f2850fe37392da6fd3b9cca3cb4e10cda16432","impliedFormat":1},{"version":"6c2af5c8d77956b1c82e11ac2386a3c15be42f758dfb597814d49dfdc446e8b2","impliedFormat":1},{"version":"a6e8cbf36e9d911856980c8efaa2187897919ffe897a7a4030693a2eba992279","impliedFormat":1},{"version":"7ed210605489d74ce93ef8b351a28aebd69409f1e9c3ba225d4fbf8ff0428932","impliedFormat":1},{"version":"fc9d689208e575600e837246841cdacf3812beaac77237475d7016422ba86bf4","impliedFormat":1},{"version":"537387829e8d47f812bac08196bc811c521ca53d28f53ead67c5673bebbf49c2","impliedFormat":1},{"version":"1762ed275a1eec5b7d30e479fd3825f88a27fa906a32ff16c64dc67b681780d6","impliedFormat":1},{"version":"a348f5ea72c33f6d2d7a98522858ed8f70981118000e926f915fa5c4aafbd7db","impliedFormat":1},{"version":"cb849466df885c46e229a616c9c8633537fcb44f2cfc39069d8dc0dfdc31d1bc","impliedFormat":1},{"version":"b04781b92ace25dcd4132687eac5d08c5264a87bea640ee77e89b210383e4193","impliedFormat":1},{"version":"ed3e9856772f055d63b460fbc89a04503def5ea71ee73cec0ba39e262a07ec4b","impliedFormat":1},{"version":"81600e99d5aad2774cb003e062357f2c05fe8cb0a370dee4fd48254c46c763bf","impliedFormat":1},{"version":"01c186e3788bc0bfd4d619555e2e15bddcc0eceb4cd256e476a04d091ba2abbb","impliedFormat":1},{"version":"48b020d8433eb29cc297ec5dab4e6eb62957ccbd6c1ee33d4ddb7f73fe50ec38","impliedFormat":1},{"version":"702a76f2b79cfb45d8a81237603017aa6c70558193325fe7cd6076023b6bdcc4","impliedFormat":1},{"version":"05adb45e3dde250b6ae4a50b9dd26457489cbe6bb5de36468aea83df2324e3b8","impliedFormat":1},{"version":"b71da9f22a46322f03f5900108c7bc23fe598e2dcd3955d94df0bf9b03adc9ef","impliedFormat":1},{"version":"98c95e6474d2924e976d896182bdee5635928e7e4a1d85fcd8722731286581fc","impliedFormat":1},{"version":"d2847d2d380d316031a863390efc06ba34ed055babc99a047c000d3a34e1908b","impliedFormat":1},{"version":"8cb85070594b644a3a07bb99b69a305dad09aa03d6325c50483ab19c86095bd6","impliedFormat":1},{"version":"c1bac9bbf0273a0c4a223f09db8d1e89f9a5223f86ebc8809429b140732110be","impliedFormat":1},{"version":"d7afb3a7abe0ea94ebb1afda11d80deed1a07959550dd48ee13897fb23d39a1b","impliedFormat":1},{"version":"cb31fbd38e527e01368c338daa15c472c9dacb72a0a6e483d7f338d2a2b88d92","impliedFormat":1},{"version":"9a056e9b9acc76b8320976d9fe6cd87c53bc1f4e2919613bcceebcff6b539cfa","impliedFormat":1},{"version":"a8f09ab4bfbaf32009c5ceb09f267c45d3e9fad63a9640c3dfc824480522eb1c","impliedFormat":1},{"version":"8d6da1d6d75e21fc322882a7a6cce39c4a85279582c0652fad76ae658f1fa4d8","impliedFormat":1},{"version":"2dcb4881af7d254e20cef0d36e33ef63025cea48ac9b739e310ac8dfb6a4d4d1","impliedFormat":1},{"version":"0e8c1b7ce40dab54106b02db1b529a9d1d34b0bec61bdd3af9c27dfc5041b8d5","impliedFormat":1},{"version":"fdd8e8b914f1d8c32363f45e37f9577be9f65e9171833a4c8c117afde983df3b","impliedFormat":1},{"version":"9fa2d338f2c6e4fb5a2cf20bc83f24102f177c9996a9550ab9cb295afc443322","impliedFormat":1},{"version":"b6b354bd57b44849015d119134a2adf6136dd459fb38f3625fbb35c844343362","impliedFormat":1},{"version":"831e08bc1e0e0fed9f34180a42bcffa15437283b3a90c453f98fd82f639784c0","impliedFormat":1},{"version":"576d3ac930288e7fe44873f24dd8ba11873ab8c536c5f67464e9acdbdbf5f0be","impliedFormat":1},{"version":"6210058f2ce3e9052681f3d2df475d6bda1cee4584dd3a5ef1ef0e60959522d7","impliedFormat":1},{"version":"7a04ce0e85d6db683f63ec9f2699fac3e2d9fdd6a9313dda42e25761a3c83a2c","impliedFormat":1},{"version":"2b9c4aed45c33a93dc6de1e5894094090363aaee045921a0e6ef245657c5315d","impliedFormat":1},{"version":"b9c7f144f9051934bba76cb051d806377a0676ed488ae5764daa9bf7a198fbad","impliedFormat":1},{"version":"dd36b72841bc2a5edbe39640abc5d0175f06b9de32d6b767615c62c460330382","impliedFormat":1},{"version":"de06c3051539ddd64044947bf5a804005e98b09613699b19de1c09ef8e8df95f","impliedFormat":1},{"version":"681c8a82369365bef1360957b467844e8bb3e9626df2162d904c8bbfc92863f8","impliedFormat":1},{"version":"8585b7a7cc1cb48171fd9e168ca0126a921189c6643cc4dd5dac43de6d3b61e4","impliedFormat":1},{"version":"7eb739af12059b8c368641641776937254b381ba298e43820b78696d4e12d3c9","impliedFormat":1},{"version":"f85ef2b6b6243d398b2146de3186f12c825a18295d3913aee1d7ad237856c6c3","impliedFormat":1},{"version":"e83218020bb0bc9a527cf10bca9f3afe489900c60dee03e8398fe135054c81ae","impliedFormat":1},{"version":"d30f3ae4c835c4006e1676170181461e7e97b4e1d2fa0c96a4d0a355cd97fd8f","impliedFormat":1},{"version":"989b02e98599537eccb0e89c3c737b75159fc64299bcee5ecf495535a4096efd","impliedFormat":1},{"version":"b0cfe92f5a41d98256df17702e2e30afbcbc5d33fcde330b20dcac2df26b393e","impliedFormat":1},{"version":"7de22e2447797056c5bbf57153d8e7d7106bab19b2bb8111cf9c9776935b81e9","impliedFormat":1},{"version":"74ecda5bfdd35b1b365b3069acb0224303c20b458e92dbacf6954eef4d9f30db","impliedFormat":1},{"version":"7e1862fcb5a27f449515e7ad569afb738896183889a3dfbf09f125d2ad1effaa","impliedFormat":1},{"version":"c3bc001ab25d65e716b576b0c607d413802122e85fedf847629227fdbf73298e","impliedFormat":1},{"version":"e0644b2e6e5f5709fd5b0377e801ae12fbd1a9d612312ed86b823159421e29fd","impliedFormat":1},{"version":"1dfa53faf0395d8e6714329b001659009171d5268f7231ad05fefeb19b7dd0a2","impliedFormat":1},{"version":"baf281afcc69097878a9f76190ec1139cdcb6d39adb1e0685644789fe99769ed","impliedFormat":1},{"version":"6d0e0c26cd26694ef8f4776443fdd846add0a327244424b5be1eb3320a770486","impliedFormat":1},{"version":"77df8e8553d35e13728f6d0a37ca982520046645694ec1edab9df2df4e905dc2","impliedFormat":1},{"version":"2996e46b71dadb93d38b47e420444d91ce3685f7ff0f0314bcc6503c6018c00d","impliedFormat":1},{"version":"03c9cee66774b18d3c20119b5fd25b3a94f95757aa82cb5bfe9cf7cb91400bd9","impliedFormat":1},{"version":"0c7aadf8379300a1dba35b087663c682f440aa5923ea96a8ff9ff483b891766c","impliedFormat":1},{"version":"70f8da676faa53028982803fb3b389b98119fb0b41df9020a3130b008ac6dc74","impliedFormat":1},{"version":"2409017257471ec5e3bf053cb4a0e0a5f3a7f11901a247118c1654980d7e1fe7","impliedFormat":1},{"version":"8b729a1d9b4f457b23b790a07111742b5c2714b614da768550e0a7309350e438","impliedFormat":1},{"version":"07ae8276b1ded724db7342f1271258bdf7d897ad47affecde328f89543fbef71","impliedFormat":1},{"version":"959e21dddaa3d50c7a9573c033371f8a8eb75e1da2e4f0d2ffc2e44862a7939f","impliedFormat":1},{"version":"5c5150c7774dcedeaf599b1372b8158b3f8a0e665d602a35e34470822da59f59","impliedFormat":1},{"version":"fbe77f3f07a47c30835cb7b88d1aeaf605052884b819f0669977a9977bbd4a8f","impliedFormat":1},{"version":"07cf01ae7f286d5c1390bec1fc2cad285c2cd96b3778b61eddfadd2305786040","impliedFormat":1},{"version":"d07829c9a6e7288abc6d1df2d0e3ffd31a2418b38e7bf3d374437042844ed17f","impliedFormat":1},{"version":"7299ae6e1cd70af673d26872a2c7616ea6fa287021669473bdd7842c94094f61","impliedFormat":1},{"version":"820567b6f3633584ecd3e57c8cc192a6a18f2803edfe730fd1531d9cb6fed891","impliedFormat":1},{"version":"2ae462dea06b9d0a202c1c034ae686d225169038c33242052f4edf93db00b254","impliedFormat":1},{"version":"5ffe14c99d9130074d6bbc1edeabe4b4ef9168a05986ac7aff84ac5735c4d77b","impliedFormat":1},{"version":"86241fb7f3594bade8e6589a5426d72a23dc6426703d43e1c8dea0570d78fd14","impliedFormat":1},{"version":"97444e4665407602750452c136344a02669cd8c42e07fdd4e1952bdb5df13a90","impliedFormat":1},{"version":"b0765a00e3e002773a6af233b937dfebf23fce20a9a7abcabb44ad6b7532e6ff","impliedFormat":1},{"version":"39ae6f648b10326364bae3e24c5735d12ade5ef4ba6ae6cf17e6b57dfc7d736e","impliedFormat":1},{"version":"fdcd57d2575b4e00c4c92b1a2fa15c791365aa763c4d4c901c3f3a362acd27d5","impliedFormat":1},{"version":"860d8f412e586be2009ba1806760f46f0501aea96880436a53956915295ba847","impliedFormat":1},{"version":"0a02b0f5641d72d688128db3e2608d927612131c11eb4ef6ee28c880b8577019","impliedFormat":1},{"version":"8f1e32247ec8ed20364777038b8d11fef3b6bd73b8596e11005481a3ea2a1bf1","impliedFormat":1},{"version":"9a7e14707830dbb4968c875b9f8ab510f531f11d3162c64d4188fab2ab0b671c","impliedFormat":1},{"version":"fd2d5cc8041746b1cc71ab247163982c68b4fad0522b2a8f555368d57f1aa134","impliedFormat":1},{"version":"7f56883fceba869ca2e3bab049cf33272bac1a6937c235c652e0bbd9aef67624","impliedFormat":1},{"version":"af1f132f95a4a56d9304f8dfe68f116d2324b0368411574932c55cbe2fafa214","impliedFormat":1},{"version":"4e0a5de3811fcb44426b7f6bea3458d1c794591d0b4a715c51c3ad5d70f08ab4","impliedFormat":1},{"version":"68b735874e866f37d072bf674535a9f33965132ed9e1e4164a6fbc494d590604","impliedFormat":1},{"version":"9cec0cc895584e0e42d5c97c4a16ebd25a4fa60730c186edf0d28df0a5bc3702","impliedFormat":1},{"version":"94d41a754d3dda0f2517d49d181f1ba1d812b85e7bc2c822c40be945328edc33","impliedFormat":1},{"version":"842ffda229092b37ce0bc6748b09a38aaedc8490a69b3a10ec48ebf47baa3224","impliedFormat":1},{"version":"0449afb9de90d153878437b4004c6c5ce1e2f8f33e93ace383d387b683bac845","impliedFormat":1},{"version":"358999876ec96fa23597d354ed2fe6ae495d7c654e720ab3be349179133ed14d","impliedFormat":1},{"version":"c5de19123a4ab7d152289dd2d7336ff852e78c134f0443cd1bf40b7cb9cbf0c6","impliedFormat":1},{"version":"4f84447ecedf492742cf058a1dc4a3cba63b932778463c858112e4072c79208c","impliedFormat":1},{"version":"74e3a9065b290394d3ee7fb111bb4523d846663d898aa21bb76c8e9af979ffa2","impliedFormat":1},{"version":"c50e1748196272f55890a55bb1cda5173fa615e4a07b3f741cf4f24eaeef838a","impliedFormat":1},{"version":"e19b2a73885f75f29b2adcf1077c8dde7d69137af24c065b5ae7d7fa9bd3b820","impliedFormat":1},{"version":"03770bdff8a4fb0c206e60d6139aa924b5c0bbf94e924d6625f553f59a6a27fa","impliedFormat":1},{"version":"2e54730070d00c443dbb388a356667bceb0a4c7ac5711c0cfc6355964cc7ab2e","impliedFormat":1},{"version":"09846ba03c1c6081fd8600716d65328421d37356a3b584b898325d5d013ebe77","impliedFormat":1},{"version":"7008aa856c52cc0af5aa6c755abfba94dbc5e0a9dac3f9a985eb5ed52e6d535d","impliedFormat":1},{"version":"27551a71453552cdb14453753b2808fb405e6b1495f53b1de318953ac4ac73b5","impliedFormat":1},{"version":"0bb991b7c106b013ccd1b236bca252a34d8cfd2a61387409c1c407e8e07acada","impliedFormat":1},{"version":"0c9fc9833804b2f941bbaca9c55858b3959b7ab386bef26ffb897361430bbe0e","impliedFormat":1},{"version":"be0b2fd0e29c0ea2153fac7069219f520447b1925eb1e9bb8a26623a0f1883f4","impliedFormat":1},{"version":"8a3ba8887f57d779b22773c0a7d988654bfc4ae2c7c3dfb497b8e7d0414f082e","impliedFormat":1},{"version":"b63cd507f90ae6676915db153a40ce7d6a4c2796d4eb9d607a6739f4017b04e2","impliedFormat":1},{"version":"360816468e738d7f3a96116575347aa1b1d3d470a35be1c3953c50cf6d50268e","impliedFormat":1},{"version":"8e9f6de6a56197fdf0b0a75ae16f95d3b067607ec1ea389b2ed97f4a9d83eeff","impliedFormat":1},{"version":"3e33309e181bcd36a58ddcaa2ccb18426e6497338f5c4d2f723804b2da440834","impliedFormat":1},{"version":"51c9f201ce3da119ca046f465c131ec8bf1e4dba44cb21fc7d3b83f2b75069c0","impliedFormat":1},{"version":"b79bd0d2db49a032b3f2bc488f1ae36ecb03adbcf7d15904cc83b29bdf0b91e6","impliedFormat":1},{"version":"881d0a25d25a83449b54d6b4f963c891e742edcef384bccb66e074561f50be2c","impliedFormat":1},{"version":"e9736dd1f0d9c3f00a7980640b2aeba560146ef7094546549d64ce5d51c34c1d","impliedFormat":1},{"version":"d6bf6f2a8f1bf3fdc6ad05035672d8c38a04f72f39c5a55db9f1844689eec144","impliedFormat":1},{"version":"ec1e72c665317467b85ad4d27f5c106e6a888116f8757d883c8600e5f299192e","impliedFormat":1},{"version":"414e31d3a66f5c79cb49e82c3960a6983f1c031df82de1008bd255bf7aee58ae","impliedFormat":1},{"version":"0b981a84abb59dd25ddc171976274c353809cb6c9384b6a82d1950790a25d0f7","impliedFormat":1},{"version":"0881bbb944fc4d723c4ac7dbd4bccec7a5bad8f1cbcb677d12126e88a92dacaa","impliedFormat":1},{"version":"5b022572fb0abf915827636c7d318a067ccf6d9836f020f2350e7c6b01268695","impliedFormat":1},{"version":"ad967a3bcad41e910ffe99906e78f68dcb5191b752327b0eac5120427212cf5a","impliedFormat":1},{"version":"cd65b4ea38085499721fb1ba4624ab10559d49b4b106f388fb35c23b3855f6c6","impliedFormat":1},{"version":"a6ae4e8d4805e76cff3fe699395c1781537cbdad8dc94417db81ed4f65a699e0","impliedFormat":1},{"version":"7808da70bee3b44c90aced70284302f34cca901e190635c26694ecb5b4ce2d6a","impliedFormat":1},{"version":"16a535be43c022b2b40c0fb4365841633beebf3d6f55f808f4999d830883e9d4","impliedFormat":1},{"version":"87fd703309f6f640f2a0a6ce79c0b14c02cbbfdbd3913d6af601be883ab8cf18","impliedFormat":1},{"version":"9bb021b1303e52cdc159ad2a254e449c68b9b5157ae26b9d918f19e2c8d94223","impliedFormat":1},{"version":"90c36c6e2722eecae74eb676d1d5cd8d6b788ff9d27bc5cb99571fa42fd0ae48","impliedFormat":1},{"version":"f5e0905774ccb56cdaa68627b7cdaa7d59634bdbc1d70954d69a3cd6b0b8099e","impliedFormat":1},{"version":"2b4ccaad2c242381606651a2cede50ec716b83caf254660849c60970b319c324","impliedFormat":1},{"version":"234153bf35fad77184c16cf346ce96b392c291f26d7cfd5f45f868e226ba7fcc","impliedFormat":1},{"version":"b01d9cda99bd4a3363a6605f0e20de677fb6942eadd642991fb05a27abbba73e","impliedFormat":1},{"version":"b69a0e1b607af744043ccf45c4f6ea74580cd60e173380534b77153300bd3f17","impliedFormat":1},{"version":"bce03a3640e013438c4567ec02683ad313d8c9ea64de068e5a508fac67554ac6","impliedFormat":1},{"version":"ce69ef779e504f2f48fc59e25e3b9817a4d813e77c7d3e936f213b8cdec7ac14","impliedFormat":1},{"version":"dcf522a16eb912d546550a5484d74f610ad6b5b156d50694c6c6478c8e865c97","impliedFormat":1},{"version":"01261c650c213c608dae5c038374d796190e03f048167be1f0b8472814caae3a","impliedFormat":1},{"version":"590d1940f1480512ee8de41f980c922393dfb61657ba825c9f73a645754c38d4","impliedFormat":1},{"version":"45fd7479cacb1b2cd1fbbac7091f65fa6f85c04366845dcddcaaf766fe30c7eb","impliedFormat":1},{"version":"7df3db55fcf50063348d0bda64af3f21e4a42eab56f5716c5cf36f2c309716ad","impliedFormat":1},{"version":"eb6d7e86dc52d6594ff5dc7d3d89008ab9f0f7619968ed907856381d75ee9259","impliedFormat":1},{"version":"2775d2fd8e2c6f3192db64cf04f38f44a2cd0ece93fdb6dbc7ee7ea07e7e4a79","impliedFormat":1},{"version":"70369948eff602cc000e532c12174f0745a9a5730617e677eaa75de36fe7f00f","impliedFormat":1},{"version":"c3c95a81bc454902bdffcdf1ecef500d4729aa6a141b5ed38e76bb0e74561c64","impliedFormat":1},{"version":"995e450c7834279b8aa2fa3162e86d3b47b9b7e63c7407ca0e3f01c053372a83","impliedFormat":1},{"version":"dc811d697eee6c86d36f29065e4cb4ea41da5336c3d760b04dc0f8ba61207717","impliedFormat":1},{"version":"66cf9358e50a8b782483f0b8a80d7483d50639642e12a18bcf5ffd0535fd926e","impliedFormat":1},{"version":"1a03a876422fc006fc5ef1fd93113c5c5330db37f2519f8c0976453015a60998","impliedFormat":1},{"version":"d66c7731f11b4eb8dfb79d23d7a3d8771e071a66b75dccedd7c70dbbbf600247","impliedFormat":1},{"version":"6e07d28ddfedcbbfdf773d3a29e3b6bf7b13a9550f2b4dd6c266efdfb3016795","impliedFormat":1},{"version":"aa9ff0cc14bac2b7f0cf2a31f00b0d0f1ac4203cccfdd1d0a188866e2b6cac3c","impliedFormat":1},{"version":"de12aae130d94131ce4fcc10cd578aa7210c55217f51e76a8471472bc155b72b","impliedFormat":1},{"version":"bc06493126b2437e3f85cb788e05d27e0584b994351c162aaf6322f9e51a1196","impliedFormat":1},{"version":"253b6652283133b8fe1c7ba038c7fed7c1b806f60433ae2e41c747f96e5bc9e9","impliedFormat":1},{"version":"02a62b658cac5a76affd34e0da9653fc32d882bb2af956b933fe0959d386c6b2","impliedFormat":1},{"version":"4ff33257d307eb3c960367a4577218e89e375ad5534bfc7c6f88962115ec567b","impliedFormat":1},{"version":"02a4a4666502a566f451841a1d570f615e5b51f9171e99ce0e31fb7b759e7ced","impliedFormat":1},{"version":"d8a2f4cbc18330f8d9cc1b4c9d723d9e77b6003b32355e03ecd1dbd93422450b","impliedFormat":1},{"version":"3a6d7502b7dc76ba0d03b02608e53a5643f4fb8687738940ec59b4e35332bc3d","impliedFormat":1},{"version":"a1feade8dbbffaefe2f6ce05ef37996eac654226a4ed85ce93c2a638a2e78ca5","impliedFormat":1},{"version":"d1aa730c32d9745b332733e65d83867d288bcf89508f51ea58f427df5321420b","impliedFormat":1},{"version":"a75e176ab191f710de50447bd5aded6ad53b9b26c6caee1704613e8100a40f22","impliedFormat":1},{"version":"49af458d05e16e37e5a4b793f65f38f1b739aea41fbf64ac7205b4d02a895550","impliedFormat":1},{"version":"730fddecd22461d03108e74818c2d54070e377cb7547f09b9560405c5d532984","impliedFormat":1},{"version":"443757ad22c5d2103a673b9babadba21930bc98e1c13ff17c57aa26fc470ea7e","impliedFormat":1},{"version":"8b71038e115f0472f4e2ec02bef4a314e1b0bc1d876835ab3e72bb74c4651526","impliedFormat":1},{"version":"00055fce37576a751e128d5c2f5ee5d9a22f2106f8ea21f62a83e6900c92ed73","impliedFormat":1},{"version":"da37d0c35a6f0c058c2d5cc7aee9b975a7d1e6def82e9baed4b1c166447a2724","impliedFormat":1},{"version":"cc6e8f8baffe0c5f6a8c6e73524f385eb6fd95c81bbc349b275d303f00b6d69e","impliedFormat":1},{"version":"58fdd698d0969ea83496975d1fafa3f17565b92eea8c66b794ae734c77887dff","impliedFormat":1},{"version":"04e7c703a42c32a95e5962a06e97adf6abc98975bc19488e6dec4cc03b1daf20","impliedFormat":1},{"version":"914025161b3436cf3bf2a0c1c3cdd05d86f60b4c9d1b06625fc02ffdef8db980","impliedFormat":1},{"version":"682fdf0737120f52e9a6564a99bba087d49dc88c2ee621777e5918b91749d184","impliedFormat":1},{"version":"44208d7a48cca101644c03a154c8544ce8919708e39c060ef1d6526b0672df85","impliedFormat":1},{"version":"f5bf416aa2e3a04499a4dff617266806656a0cf874dcbb3f76d932ea7cb6d6cf","impliedFormat":1},{"version":"24ad82b2c510e55a69dfec91b8f93a6a6bf2a1cce00e7b362398c81b94641645","impliedFormat":1},{"version":"9a90f27e9b4523bdafadd76b6b0b956c29f4ff2167014b9349b458a3be8efc52","impliedFormat":1},{"version":"2e2b206abda66516276ce9f18afd72ef035acb2cba7f911e194a9373ddccc359","impliedFormat":1},{"version":"53175b3b3e8cc44c5a822ab73a9330aef96ae80dfa3b080fff4ff3b4380b6545","impliedFormat":1},{"version":"7d21b5ccd3d4f1b0c538334c4f285195b01421f95c639f164a59fa907f55ddfe","impliedFormat":1},{"version":"1994b6a5d9d2483e6ad2655ea1a28bbe9f157afd93b368956cf1100837196978","impliedFormat":1},{"version":"f930b047fac53fb453faf71202ad06856e435b85878428bb1461d9e69cd513fd","impliedFormat":1},{"version":"efe8071011585e5a4e6f98ba486a3b50a4a12962546b07fcdfd31d7cba3d61d9","impliedFormat":1},{"version":"6340f1ee07636c61d7a54866ca967549453b4b65cc9ce01a930ac7b80352ffb7","impliedFormat":1},{"version":"ace6e06ee0d7abcc780df319cf590f40403bcab66c1f7aa34a23c4e71c8abe9f","impliedFormat":1},{"version":"343f8ca55d746684f18bd765aac60791e271027a57d125001f6f6a4b2c45983c","impliedFormat":1},{"version":"8748bb756dfaeb47f1735753cd8563e06a28fb21f78189cc2ba3252dc7a6f909","impliedFormat":1},{"version":"97a8be08b25f03642fc989afb8328ef7fec3597d5022e69b606c0e97e23c4662","impliedFormat":1},{"version":"8c2a41579a8f79d974c446ad30abc568843d1d71bc9fd5cecabdc7909ac19148","impliedFormat":1},{"version":"770fd130b6d7d71b809e64c706cd88ade602b28645809ab4718a31b8a762464e","impliedFormat":1},{"version":"7e233539bb3a821e4264e9b2788316344c0a6a8f251aa63bbe018981348599fb","impliedFormat":1},{"version":"cd59d351674711db4f0cf99f376dd42ac8ab981d0acfae34d3e31b5e58cc78c5","impliedFormat":1},{"version":"0ffd9f8c9629c76a235b9d6aad4ad683f1104777bc73772905f1d4380160f058","impliedFormat":1},{"version":"f1d3f11dc691e6fb07d4fca0a6dfdf6f256a456f5eec578706fa58d1dd1cd91e","impliedFormat":1},{"version":"f4365217529ac5c0c792444c4c655556dc7a3dc1cebb12c7b8f22bf68688e61b","impliedFormat":1},{"version":"0e141c1f5c6ee7ede3b42444fddd1a7bcce1f02af671ec43c617d4edde68201f","impliedFormat":1},{"version":"5fc1c764111de7f89d2b2e86f8bc22fdb273baa275f5ff8d2e7ea828a05206db","impliedFormat":1},{"version":"d8a142b313b3b38def4b4b89d0c4365e5c7effd1c3a448361042beb0bdcb8a44","impliedFormat":1},{"version":"3b76d6b94e097335fdb71f16db7b0cfa71bf9c126fd6b439032e55d01d2bc374","impliedFormat":1},{"version":"80db7661a743970459b71c85168a92e5a51548b91ef52e485fcc239597f5ada8","impliedFormat":1},{"version":"6afcf19d308f72d41424f2cc3eaab84b522e7f77ffc987f1b28cb9ddbb0bd65c","impliedFormat":1},{"version":"f117450517ebb04a458d699bdd3b03e8f38ae58658e18667e2dc4948684b9433","impliedFormat":1},{"version":"64687220bdb074b06df8bf800d9eaebc7a0fe28c868c80923d56af5d874fe7c4","impliedFormat":1},{"version":"c4b37157d753d187fa5927fa7bdbe8935b68242ea2b1f7a340d9b905793678e0","impliedFormat":1},{"version":"6e9bc9303ef4d9a0ee5a17e9796b4d22168c1935ac336c69884e3acd474ca4cd","impliedFormat":1},{"version":"5d81f18f4180866039e07668a32d898df0a6336a0548998919ec7f497f85ba6c","impliedFormat":1},{"version":"0b4167292c88a75b47853f16665b47018c6695e5cdebd3a7d9cf70e1cd209e71","impliedFormat":1},{"version":"77590aae8633c799521101260537d88360c87e72ea092d66f839ba758679c6e1","impliedFormat":1},{"version":"696eeb83cdcd2961631c0019cddea0e38a16e5e687b54e0cf527461f93e7a010","impliedFormat":1},{"version":"cb17fb0d8ad59206358300181046dae39f66cdad102562cd0088d87ff6182135","impliedFormat":1},{"version":"8437b145e4ad4df671bb99bb79a49ac646495f2c85bd945edecabdb08ceabf6f","impliedFormat":1},{"version":"1c7db5e9ba17e5d4f4267fb1ba6fd4ba398ffbd38c8335422557a7950e995f33","impliedFormat":1},{"version":"647ff75e7b8afe76301ce5b4067fbe87be4f0104d03e627d596520706fc695df","impliedFormat":1},{"version":"f5e604855265b390c923f28fa485942779ca5a685564f9539eac36ae73e1755a","impliedFormat":1},{"version":"b87ae8d3be3434db924b9c8dcb10265cb46de954acbcf6389c44c2d6cfdc3590","impliedFormat":1},{"version":"0cf85a26ab1817238cbf59001093152dfce7d96357d9ff4b026c455f87e20fe9","impliedFormat":1},{"version":"2f02bf91bb2cfcf721f18db1af83ad6c50e079002ec79c5a7ec1bb014137bca4","impliedFormat":1},{"version":"af85028497e2e9902a2142ff8edb93585f453c49b5d679ed1711cc2cafb9db1f","impliedFormat":1},{"version":"8a9235448502135907b4cb3e3256376385455ffc07595310177fb0a617b06171","impliedFormat":1},{"version":"e8881bd2fd230b04d815bd5278b974d9cc6e798cbb44a6e4b671ed1974e1a9cc","impliedFormat":1},{"version":"d806ab6ea9b1099967a11f9e8d939f40785f4482c14178166e9a0386c6df18f0","impliedFormat":1},{"version":"d2d246de7bcc81d4009d4c28e8a8c4920bc34366c63b93845d132deb3539ee72","impliedFormat":1},{"version":"3caf659fd3ff85ea7f962801da208de8d31f575684eade9b103caa3b45d1e3bf","impliedFormat":1},{"version":"97cba43725c4b9dea426e3c31e81de5ad8c5a85e8dcf4b2afa39aef117ee0f1f","impliedFormat":1},{"version":"d4bbaa99c31e834ba8ca2f4aa32d1ce76d2f3d7301d77ef87d6e4c22f0966a86","impliedFormat":1},{"version":"fa866d4dba8ff3030ed22ee15335d5bf5e7c20bd870142574ff96bc42da453ee","impliedFormat":1},{"version":"2732846b3f2c2d4155e7fc57c144805f75d43a16f2ebc610195d7a65737c9c03","impliedFormat":1},{"version":"1dfb40e6629cf803267a65920a3327c3fa6a5e42b4c6fb8865cc503a5b7742a1","impliedFormat":1},{"version":"f35c1a8bca091f454997d35340379aca49d25346e51ab1e15126760ee2e171e4","impliedFormat":1},{"version":"92230275025180a19caae70b82c704d73b2de644c2b4951b72b24101a19093cf","impliedFormat":1},{"version":"a2b176f66f0b708241265fb3b417597c9c9d21912bbb7f5cc00d99af551c2078","impliedFormat":1},{"version":"1554992f76d012249bc7c9d70c69ed08023cd89e64fb128b845fdc51b152a195","signature":"df9be01d27ac7410fd77d063626766dfd9c1fd3038e59ede44a364991c12af25"},{"version":"dc1f71679639bd7696b70303bb9e793f58fca39ee74d0fa4f0260a15c02e43d8","signature":"2fe254d451ea1578c280f5c697da36536eaea69b19c16798083adc043d4bc0b5"},{"version":"28bceb5dbffb3d1fa4f7b2d263042effd3f767a0207a29fc4096670834b74602","signature":"bb849c7aba889deb48e09ac203ace4cb7c96109784ce4f3d40f061aaca2da633"},{"version":"598f87d999dd4ec51f76c7774916f58b1e6acd70b7efab27deef4fa6f8f64c19","signature":"10dcb4f2538a0e42366d51164c5a66912a812977fc4697ddd221158a050c1069"},{"version":"1e15a67a5424cbc3cf7ca7683160802fa8a362eef43b2a2edfb19b71d9ae323a","signature":"eccad1416e8cf900cc473e6897939d4e4f896eb55d8f146d1066f41d2cfc5062"},{"version":"9ab380dcc182b9f143564102f63b50bc6d6861b20d4a410a3a6fe4ff3d672d33","signature":"d8e91faf7c09fa9d280828aa9530ff5c7619c1699f25c7d3319cf0fafe3a250c"},{"version":"834f1925b7608416d278a60c375a3926dcbd0721ab4b25e6bfbf552ba6e699f8","signature":"bf39329839288baf358b181b55bb5863fae02f14da4da3392ffeb770f1e7381a"},{"version":"dcd1614ccfda268aa46032d4afc4bc14a767973963bf522a4c3ab25b6f96ed5a","signature":"44a6ec8b837a70b644ebd1879a4a9eba2bba8c654b0b187c2c6346ab9e13fd2c"},{"version":"fd461ae7cd388dc13227b799c6eab047670bbe862b795370310f608c132a4a82","signature":"4cb85c88569583ffd0eb0bb007c27708067840cfca27690919f4fdbd85f62d79"},{"version":"817b02ed162974885f834c7439a29d353b60c5a1f3889d6ffc6c6024d8573e3b","signature":"d2eecc5260c72f6c835f42c3867c680422a16c7b5b47be9ebc30c6bf14fd9b79"},{"version":"4477b15bd28eda6966f45c646eebafd06bfb7c7a324392e593997a21b5e670f1","signature":"a81a99c87635dc8bde9cae67e7613ff373ebfb0d4abba81d0426641d391f8bad"},{"version":"7631f02ba11b1d4a007a422a163c6825dc6920f674083c803127e7e285dd8518","signature":"af0515334f6df00838b777f39d970536edbc5be22b4dbef58055d9189376b459"},{"version":"152e7f05f6c7f103146a86db054513a655e639d6722597859534e0421da2a4b7","signature":"54f227db089b1928436811c80e6ab8b12b14d465291010dd786c230f328ab0fc"},{"version":"5ed669cf8ee68df27d30cfe77ca6fa7cd6b491fdb638bd3efb5f54977b8648fe","signature":"fbe4b092ecbd682515996f8cd40e51a45c9eb45fe2d51fe493b737b36614ec0c"},{"version":"e169b249a2d973b77fffd09258199fc78e40f3f7c5be49f392103408c8b69a99","signature":"ea90cd928ea059f037d8904adfe4b98c7213f12f967f03697f7b4e73ac57c2d0"},{"version":"c5786e3d1e0dba4926f01cb03e9009ec1731fcd53b5915610b82b58928f5fa40","signature":"43dc4e7330bf525c144fe7effa5e4b6d46278f19f15a14abdec689688027f6d7"},{"version":"1a26b81342f1c6f52fb7eab4d1570b2dfa373e0be3f8d487dc456bbaad98871f","signature":"2a6ecd283a49102c720cded87a490eb4591d77d218fcf71e0cf4f8089e19c561"},{"version":"32f2705ea5a7a911137140ead4ea7744d7e3ee275ae2399cf6035e97c631250d","signature":"e962a82c077bed44b57341624b308c18933d6d9e0acba8b71e8bd4f9041f14b4"},{"version":"2ab805b3b9c0ebc70503d80e8820db06509bd40ba4830cad8bb0867608bc94cf","signature":"15b14e85bd71d44128aa43050c83912ba18f366202fe680bd7fdde79e5509c00"},{"version":"1a10b0b4f772c3821ce1cb2f9c35d3166957af9952353526ced131c1631a2ebb","signature":"55722cd275e496f0c100342bb2a7a8b3f66cc6f59c9af61cec7774d389439a0e"},{"version":"379e7c97b4fd4e801114f0f663c5afc77919da2848c90b5223dbef1351941092","signature":"aa3fbd244ff3cc308c50831396e10c1ad14c86bf38e7c83835141012f5501fad"},{"version":"481a8f5c1d3317aedee87dc1bc184c5122a5e86b5ad5bd7c18c7d08b346ba386","signature":"8950964805344fab3e2aa2bc66b92bf8ce1eca4036e9e26c8635a937fa16c229"},{"version":"7bce4df25ff20856493d1437364a6853e7ade477dd7b2a5bc2a030258769851f","signature":"bb4c60e8412068c5a8bec36fefdbe1544971a9d084f1b5b2e338b86da742f53e"},{"version":"2498428edde41654d7ff71333c83c7e50cc639abe26b730f4b4b7f14925b47d4","signature":"485c4faaea828b8dcc01ef0d3fa90fbb0cf17b61a27ba22e33324e0eccd0cbd1"},{"version":"948849acb58142c43f9390279524ec373ab2e9083ef1c7ebd92568aa329b0388","signature":"7ed8e8574f2686f9270ca5d0f55d1b601f6e6a0a3976e0f4ceb8e04d1cfbb9a4"},{"version":"0b50c82cb7f888599fa9b9403c007a20f526c99ccc0c31dc57a9ee933f9cee0b","signature":"561aacf7d745e72f9d4a0456429c504e2b3181faf7b0610d3c135a175ee66f8e"},{"version":"81effbcd03ea1097b2c782c003eb8cc8a65bd71c7633de597e8273468f72de45","signature":"71c14c2eba7f346cf381a84ebf248ee72c0c86ef0d0c7c428705dd81187cc9f4"},{"version":"06406f1b71ba055c5b7c21e234f083ed1d0047b24d255447cf41a475f98c0aa8","signature":"75e3cacdd72c4f5a8d66ae43157693745efb3cbf8fb93e6429730a7dca8938a9"},{"version":"9a969544b0dca78e5e1aa0df8d34379469429f417545cad50fbb53a857f03bca","signature":"70870e2cbb742e3d577d670a601e432688354f784d50ee7a73b6c8305afb5cc8"},{"version":"d6674a943e6e810e3f9cc51bdefa99c330f89078780a56c4dc9b6030822e20c5","signature":"5d1c7d1e4fde9cfa5fbf1743841fd0b5056a7125e7299b5fb13f336eed7b9a4c"},{"version":"f117d94d2332f69ae3d7eac000d2f36f34fa5f0e000aec8e994b88ae00d18d06","signature":"7d041c97c4f8c43bb0dd49b028e97f16c85e5ca22025bfdde860d21e5a2de691"},{"version":"3bc6da7fbdc5294758e391d2898cfb29dbd4c077eb196cf3cb21a3c703c37575","signature":"d8d7ba53a1a35005b7e1c4c64bbb36e9a389dcf9763649f0abc0e87792263605"},{"version":"3838c2c48dfc95eaad1fb713531cb391cc9954717d95ccaffc6b006921538c6d","signature":"3f2b7712a1826e69fb7922488f76fc690c3a62cb646f375ccd98e7c48f0cfce4"},{"version":"8fc18b184af91cecef9f735c6c1202c24c18cd346be2b88a27ab56e3b6fdefe5","signature":"dd17924371b57f57077ae3366583c5d060e67ecb40b3c6b2c4ce0279c5e95b3b"},{"version":"d166f048a25424decc42d789e660d57fd34ef2f1ff4b17c6a7ff6ce5f33afadc","signature":"d4fd8c26a8cf7338498cb17479a6d22c8629c2aa1a5e75a472dc2a0a37ea0e94"},{"version":"5fdc7dd414d3c218df3764629bcc17ddb4c0fbbcda40c20c3a4d911ccf75cd46","signature":"8bf47f714857e0ef706d06deb927bb7a39fb9997c815d8c5015698c59028541e"},{"version":"2c14248a23c029b2e8d437f7e0ea2668a400e4fd9918ac24e1cbf82b58a362e0","signature":"791da0d3b2669e3d98b91e8d1c4c26c30ba6d57d2df076942635815529b32104"},{"version":"535e831016edaeaa4fb7ae3dc631a812c3cb923ea3e54e81857c041bf2727bd1","signature":"0ceecdca86d1a0602b7c8c32b261a0d9a5ee743cfa882f0a09e78545170e7723"},{"version":"409468afa6b7f2c897ae8b3481b2672e88a007fe80f702e7b33a1b7154af335d","signature":"f74d4541d1c4198ddde91a0fb625d79b97a2fdf500233100c9dbcf734766ef58"},{"version":"a7d7d010f8395ef56426a32cc112e00acf4ef9db074c836df8e39efd84ae6551","signature":"64c6d9e98b95f95c165b55c0bc1dbc924b1f4bfb30caec4ecb54330e0ed530e3"},{"version":"373f340d426e8a46e0d5c28aed262862af1a963aa6aea8a3617179c1d3e78da7","signature":"9928eaff977ffe86cc59371955326f61600c2eb91b94ef096e5da3293b7fd800"},{"version":"e29baee5fb07788a7d84a3b538932b5d7fe7da717b26f7e1726442a4a958790a","signature":"d833128c972c0c3a618395bb6e2318dd4b76ec50ab71a6cac30a0a2ba5c29ca7"},{"version":"ef8ffb9b3013967e22e481d5fbcafa12fc55d93bcc7ea11c7fbd0f2ba36c5051","signature":"9ef9c99a3bd00da04e7f444f862b17412351577543b3e2eca9daa188bea64e03"},{"version":"1e2011cccf775de681d19697e17b7c5d1cfcc2a61fc821e961bf7bde9a16861c","signature":"5c798cf141cadf4e335f64cdc6743ffe274cb451fdd0ddb5f781361369bf8c13"},{"version":"63576409d9e3fcb4bda3765dbb59d75b4048bd3a3befee8dddff60d4eef0a5ba","signature":"e39a46fa75d7193c411aa805bff51bf48f4567147e4a93ea25c5fab2633cdb6f"},{"version":"4dc8bb602aca52ab758da46bf72fc750fb2055200bae30c14d334242d9fa1ec8","signature":"615d6321b92fa28a948f6313e760efb37095dce553da52630b62d7835203d4ba"},{"version":"20f6a4df917ea9b30714a7186affdbb5781e345bd6ac9d5e14bdf522ce3d85a8","signature":"087b6445beb6043592dce6ab426acbe0e6ba84807901318efb9c582518f347a5"},{"version":"62b368348d4595681e2faf46083816a3f9e8a436f2b8a1e4de30212046afe419","signature":"d9e78f45af9c544f2fce2a4f0fdee8e0b7282bb7c2f9bf591262c99e913eabe3"},{"version":"4aef2fec816bd9f69a4034e9d8abe06c0039148f34fefe830ea5d32cd53a9ed6","signature":"bca51d2bf990b92581261380bca472d51b4c64548ec7d32ad2322bc8a4c94903"},{"version":"d95147bed7dbcfbb42dca9dbd0e638298efa57339c8efbc51a96e6ba47a8b2d0","signature":"2be7aec5d774a34b8429e2181dbb7a656ddb852934289146525c55a6a62132da"},{"version":"0ec08ff261a413199b1a7fdadc4a3f73dee6c4d9951fbfb7e3c6fa8cb823fcb5","signature":"ca20d610b31b3e5ff7db54236c85bb4f6152060c97462dda72e6df90d2a51586"},{"version":"d76f8b37eac1587b2b57d1cf335b498d2e2823bf7e4ce958d689924ba6321643","signature":"0d685f7a4135f89433d20f6ad5df2dcc16ab4d9389027cb104079a3b5aa999ee"},{"version":"c7fbcc60fa38745f180114f83534eec5da73f1a67102b319ebc27f4c8a16456d","signature":"7e1c01301fc4b369b7677d7ed9426665aefb7798ec9c0aa754334c54ce397c29"},{"version":"d20f3c309bd0bf482a12b336228d596fd239a4245f887b8ae4b40775f2d29d33","signature":"47eb5dc2bcc7e73d1bb5873e6e76e22438622457d6f73293e042d1ead22d33db"},{"version":"828e8383220c6749986162002f08832bb6728f050ddfb1dca0fd798fc488d23b","signature":"a48d25fbcd6557f25ab745ba1eb1ce086dff6fefb2e30ad365b8a4a63fa3456a"},{"version":"9b1d903005406ad050e592a360f259d66087eccc470e9fc60aff593cfdc1f12b","signature":"227b46b7cddc953ca05c88ac2abc1a653ec5a153d52e992993c0da6d96865b77"},{"version":"db74d50a2e6cb55a07d0edc567bb09ef8320d0272ab74ac709d3035ffcf84aae","signature":"f33cd63ead3d620d07c4ddfced90747d233bf76f10f99fe96bf37a9036587d0e"},{"version":"da0f159ef4d0311db3f9488554bfd77cec183ebe1feb9217430c9ceed8927469","signature":"525c6bf34d3a509a31e8f9e32abb3d4b03cf11522751d016d4af632540a85f7a"},{"version":"c1977b0af9c07d917605c2a86ff9e4249d10f2733a6c37a1e364be995a527541","signature":"60e98af90ac1741af2e2fd4fdbdae1884ed603a959c5c82980a177d98bcd4a9e"},{"version":"b9806f83bb85a7f1abba443d5c018885edf3221a8189d245dfb2e46e88bb5fab","signature":"704c2bda04e9cc82b84a45037507b7960903b57c9af3c518c33fed5333191398"},{"version":"500fb09f0abe33d5bb07f3db4c18a974228b384ff0e3bf41b445996ee757d962","signature":"6b2d38b86a5ac634b58bb23d4873991fced9eee275ea8b4dcc7f8eeed21c6b96"},{"version":"ebf814a7ac86d17be53fe3ec1600348734bde2f17566e60a01598f4d45f9d42f","signature":"6d9bc9a88b2a392f5319733ab79e67eb130900f3dea45455db899b23386bdb55"},{"version":"f0de36794f2bb1ffec956c7be1d5ad8888580b99d0b7e40ebad10753e16aebf1","signature":"5e5bcb83f067468b74af3ca897523bbd3f621d56e70cdccf7bc4c555c3fa0f2f"},{"version":"3efbcaf4e1d15e188d79d8902a578c75390dc0d5ba8ccce1e1e0d656868a8be3","signature":"4868aae010c650a0a31e0c0da3df5c8ef50130dea774bb992418b14bae93e58d"},{"version":"f095b2c1297bbb1443fa3914c6e7888ea6d81c3efe97b4989f34fc0951150f38","signature":"c059e6b886dd14aece01c4faf7d8074ae5e8124ecc677739bf84f65be33bb7e1"},{"version":"ddadccf5267b084dce7f372791710abf9db1e95dc19e93aad44f6916625beead","signature":"6678f1503d7ec3d36805da51ca30b5ca2b485e66e78868d68b3eac7ecfba7d5a"},{"version":"e85fa58498fcaef64f365abfe2f3d2585542268e8689d3a6677c3abb89975654","signature":"3965cc0ad0c17b0f597ae7c796d984322b07f88a69bd6bdc97a155de5b308e27"},{"version":"d3f762959deba41fddea15a887fe466d4c81d3184a63f02f8ad7dfa8141c7094","signature":"f49c5cefa43dede1a0493910025017d22c7981f553ba040abf431d88263ae6fc"},{"version":"db17bd58076973ec22207b8d451cfee5ffd0944723f0858abef744fffdd9a629","signature":"48ab5e2bb1d237cb3d67625180ee441093456e2fad3450a5fc9775ad032b0172"},{"version":"09949aa54d0af6316860f6886f002d1524e9e73d986e7cbd7f7eba507c73b72b","signature":"dbbb50089d64e1804d49dbee0d957588531d4c165fa640fa92608333a4ac0008"},{"version":"3913aaaa3cfacd53ecc3e547e73a4af9815288da7cfb807287f3a1bb6d204dd9","signature":"f16208c7c4b478c451b81c99f984624100823df8691ae442d306618cf2a2a5a5"},{"version":"c4e31f415406d54c11f7251a6ec431ef6c3070bd9174b5ec8fc86099ede0d7ce","signature":"b94df4c23ca8376c63cee0e0518eb8d5aa1bf7ab24fe5bbe5bcbe348a9a48bae"},{"version":"661a6813c0c961496f8ea64fa1799e7d87b1f8ce7a6917f5028d370d5ad9dbe9","signature":"c11213611524bfac48b81fb7e58598577b10978fd91360f9c324b8ff995a6e52"},{"version":"465beff9c9a2a511929d7f738e7ec965ab8f6a93036a0a123b093aea6453e65e","signature":"744e1058cc7a61fe057de94ecd331f3b3b80694452bce8407dd28782c189835e"},{"version":"9a6fa82c8ba263a1ccbfbd0b76be4853e4ce80088c19a05986053d1236d43daa","impliedFormat":1},{"version":"771736b1b19b7ba3ff81131b4e4e3f2602affe9014bc348d055ab1f29fe8e111","impliedFormat":1},{"version":"a0cb8606eb67d99e44ad83f758850f9ec0f93ee41ff1438705d730ab55e2152b","impliedFormat":1},{"version":"10d4be4e93c6aad31c4c0596e8faa91c136fce7f844b9857c43ca2e772f29af6","impliedFormat":1},{"version":"b631b94a548848e50a2d73cb75d665fe501f56ed71492ba35c71588e3ecc3d24","impliedFormat":1},{"version":"82bdf9c5e4707f7e99d4f2aa9d0322cc7763beaa4f4aadb8975a17958a543a53","impliedFormat":1},{"version":"a2e59f767d7d5ba0ad1a3e163959d89329b14cb38cec4fe1a1c3f00855794933","impliedFormat":1},{"version":"debb3ab7b5ac10a524327fcd29a9954f0d4d3bbdb7a139e50a52cd7a2d9714bb","impliedFormat":1},{"version":"905938b5b97268d568b5079729896ab9d85dcc7580a7f15f312669fd324aff34","impliedFormat":1},{"version":"d56beda91955d2405674b64e09fa5d3b201ed523fdaa4f22106f265abe0b52a6","impliedFormat":1},{"version":"8814fa8398cd0c43ba36ab4b3e0cbeb684929c2ee75b6dfc17fca5779a54b312","impliedFormat":1},{"version":"09cfbdc7ecad1afdc03f862f6b055dce06adf42d4751469dde62bbd0bc66e4d2","impliedFormat":1},{"version":"6d1c39508b7cc34f216cbe7fb50f4cab696a701cc5f16f8557fa1dd92dbe22bc","impliedFormat":1},{"version":"0232138c115e763ca41fb2c99345f5d00b38dca6aa318f92496969ed3976ba26","impliedFormat":1},{"version":"7dc421746b898f625a41eef01991552dc2a736743b6896b78b2474dbafadd9e1","impliedFormat":1},{"version":"729f68032b902fce0e4518d639b90b3ae566f653322f6ee134fe074dde5194eb","impliedFormat":1},{"version":"f451bb486dbe9dca748377e54f3c6b265bab7ff92d4ada0807931d9d1eda05b3","impliedFormat":1},{"version":"dabab4ab3b7b0d64307a30a214aa8d84b3218a6c13b17c55adda70f3e55718b3","impliedFormat":1},{"version":"9c5c3add0cd1a4a2c3b148bef1000628975c30f7fecaed66b952635b17921f80","impliedFormat":1},{"version":"fd670008e1c0b86fca825b0870ad25726df1a3fd5e27b57d7c1cb0c30713369e","impliedFormat":1},{"version":"e5eb34a6d6dcbd488012f8f35d7fd747433bbedeadcc5061a7114b54265ef6c3","impliedFormat":1},{"version":"0cfaa3d88b15053e766547c7254bf7ad1925ee306a1e95b2d92784d4e9857e0a","impliedFormat":1},{"version":"bb4ccf042a5fc0a77e455b15a1d6dea6e646811797ba1bb271f45a1814d127d3","impliedFormat":1},{"version":"8805654981638cfceb30d0cb69d9d55e7624fd153a07934660bf39f6fe11fa7a","impliedFormat":1},{"version":"ffe45a0886ac155ac748f57afff2d307a24195d1365a9068805035a42fc36535","impliedFormat":1},{"version":"209b6ad1436a8bf2196ca677e430a8d05534e9fbbc7c049b0408ada826a658ee","impliedFormat":1},{"version":"c5c2d9df867da5ece1608d7e8f339f7af9c6e07b07bdb2389f979d5954ab402b","impliedFormat":1},{"version":"b8bdb2db25b4100161e1c2e14c949d11459b518519127fd60fdcfa0adfb148b5","impliedFormat":1},{"version":"751d1a0f997306f1717749c227c7d15132331fa0c944b0b9f8b6d7b94e9ed0aa","impliedFormat":1},{"version":"8d274b8270e6f65dbb08924820976639ffc3e2ce3762863f4726b920bfe51a86","impliedFormat":1},{"version":"a8afb2426aae23fc745745a06c6171cffe9eaeb6be14d8eb2e24556b68f2b87e","impliedFormat":1},{"version":"01f21dec5d7e2fdcf8f298e4eedac690b92ca61e741d8b642e1b96ecd24458f7","impliedFormat":1},{"version":"c80846bdd075b043f17ad6c0e5c9f042b96c92904bab296d5546b66a0715c5a2","impliedFormat":1},{"version":"512db3d42fc11be2a41f0b504025d28f4952f3dad5d6bf8c22b234cf86b1328f","impliedFormat":1},{"version":"18a9089b46e0de4ebc1e7d2cb9996954b92db8b00df49eee73fe0ee230f78912","impliedFormat":1},{"version":"13f7d75a5b0bf0e80b0a2f659d4845a80929b7483138aa7d3b7f18390cb81573","impliedFormat":1},{"version":"95c29bcfbacdc30a6cb53ed70ec819849d2b1a919064618667000b739748fa3c","impliedFormat":1},{"version":"0d8e5dab8f673629c413af62dc258bc8434502b624e4dd0ec74a6f5d78901e84","impliedFormat":1},{"version":"b4f40889a37d3418a4817a44eb2afe9244e8739cfbc5839988aee305cf0d0568","impliedFormat":1},{"version":"cadf30e71517e0bcced437a7b108e8ca6d07523a5f6a5c04ee94b73d9c74e1d3","impliedFormat":1},{"version":"2a1ca525d4ccfe46df2ab79badd4c1d297fe137d92ddb61f283b6ab67bf33b5d","impliedFormat":1},{"version":"272c8775891922a95c49a17975029ff4ff2ce8c1ce6fdf19c010b6fd4510b767","impliedFormat":1},{"version":"b12a9ef1c5688ed8753797bbf3eb3d178dde3d8a7e74632caf38c04d9d2df374","impliedFormat":1},{"version":"5ea9983857848c97e1076f184a1f66ae19d7c786fcc6ef2a7b6e432c3d5f6a55","impliedFormat":1},{"version":"e7818f9b49f5aad015055446fad21bcb18221d90866be6869eed9c094d895a82","impliedFormat":1},{"version":"9818c9ea597b096e8a25ab57e559f9aa77a0ae77e95ac7683a30114233e109e2","impliedFormat":1},{"version":"51853d4d3bfdd8e95f1ce82a0d7c1ec9918381f23e5c7a22e74eb4e5cab8520f","impliedFormat":1},{"version":"07b368fe05496c6190d450787b41f409a78650e692f909d33bebfaaaa39215ca","impliedFormat":1},{"version":"91eb21bd82cc331d43613634c7723068ba7daaa7447d055779568fa831e65972","impliedFormat":1},{"version":"f25684e86972a0268a5275aa9016965ba4f3559a0b7d17123fa42be8505a1de9","impliedFormat":1},{"version":"d9bb2597305abd711c1870db94603cadc5d5c104f1d7f48a66d987e44072dfe0","impliedFormat":1},{"version":"94608743beb1c14e145861941191ba8dd8b092fe9f952cc361386929edde5297","impliedFormat":1},{"version":"fde1dcc51826b1de121ca3f31a34515777369ede0d9cdee92d9f69fcebe8c080","impliedFormat":1},{"version":"4e1d0821975eccac0d8dd05114cc8a4af85384974b5cc36f4f794701148ca571","impliedFormat":1},{"version":"9382cd80f51e70c7fdd16fa8aac523300bb2c0f84bc4e68f97d3be01861f327a","impliedFormat":1},{"version":"b591e541ae1f473ab7156a78db4628669542899544a5d59a42a4bd496458f92e","impliedFormat":1},{"version":"648b01f7b1a4b0ed9ae3975354d423aa471b4e7553f6ef1603dd75dfb850bf78","impliedFormat":1},{"version":"3cc973ce6de3b1438b1d08c5a0b415e65334cb4093ab45288ddb06f29a0a80b1","impliedFormat":1},{"version":"bb487ad87083eb51dc9989d6271c49bf600945ff6878c30b0cb04eca2c69a017","impliedFormat":1},{"version":"7f24d392a436c24f098e5de4879f8319da3d582d3765eb8ae5445c97790094bb","impliedFormat":1},{"version":"323c61bc70555f8cb23aa26d69ae084bd6e00934053bc79dfdf7778671c0410b","impliedFormat":1},{"version":"14b2bb335421928573d218b6e0ef4d32e819743efb9ad53d537c8590eb71a9a2","impliedFormat":1},{"version":"e13fb28e621f669a9a710a71f352fe060a70be237fa78e659e28acd2dd3f6cb9","impliedFormat":1},{"version":"c53ad6e31d7d703f61512a6760525bd2a362ed9d71ab87f7149b1fe991fb179c","impliedFormat":1},{"version":"43b95f7d3137ed93183aeea38092eaad1a29184717b8ada87d46941d0e72a973","impliedFormat":1},{"version":"8cb4a0da83c7e8e6dd38fc0a35150947bdd3df3c367e913f64a4ff6bf90ab695","impliedFormat":1},{"version":"8ad2a2cf8cac4806b41a83d7602d74a544df79b49fc3ceb09c385505163ab57b","impliedFormat":1},{"version":"546979cac4ef9c08209d6c8d27845e9f951bc6b1a8ad5f0332fef8e0dd5c1842","impliedFormat":1},{"version":"695ebf21677adca99575db290ebaaab052cd9b67db3e58b96ea9beffa2b8520b","impliedFormat":1},{"version":"efa83a855d11ce8b33380473d417f2f37b551a06aaf7e3fa4f1f2ad58386ea12","impliedFormat":1},{"version":"e8b58fbefa86c7c46084ed86bdc3fa22350df97d30328facc63fbd9c3f42f5ea","impliedFormat":1},{"version":"95d3cc26a98244ac9df2be6ea562e5df482c72e4218ef0d4dd89af87022cf229","impliedFormat":1},{"version":"97e350f1868b36b39c7fdd0d64bd91dd48e355b452dc84de6fc4bd05196548f9","impliedFormat":1},{"version":"0e64fb82055afeaa4d099a6549bae27defee10cb79e47ca69c25f66fbeac42a4","impliedFormat":1},{"version":"49a59b64320f39788cc504f35e12f01961161f7ad92e2fb688e0173d51d2901d","impliedFormat":1},{"version":"f7750c9359fd236aecf8bc74e47992c17ae8596a8df28dd134e6c56b418def07","impliedFormat":1},{"version":"cd54e59c82e4cac94f28705937a3340add15d9f82bd6b9014745f6e47d4d7076","impliedFormat":1},{"version":"d4c41976704a37da423464fd9bd8cbadba63372a996107a11c919b449c1cff6f","impliedFormat":1},{"version":"fba20a0ee9a7515c3739f9006e2225d0371e8c87bc29f334de0cc509399c79e9","impliedFormat":1},{"version":"b542cc10988bd6f8d96b234974156f3a094609f66d8ea64000ff8cb4d8402779","impliedFormat":1},{"version":"19b8859e2fffe28d35df9074227bcd7aea54542366578863736c0ad847ba8f75","impliedFormat":1},{"version":"f243156e2be74a6d3cfc55c9e15df882344402d9a1089c32eaa131a43f38c718","impliedFormat":1},{"version":"7fc3c1b288bf173eb7c6de841002cb9e337d027c0cec0bfecfd527f2cff484aa","impliedFormat":1},{"version":"ab7e5d40b1a24221da3a66b6b695b345345b80a79f0a287a9e8694a6c4fbe812","impliedFormat":1},{"version":"7aa57dd1af7fab949ab2957a779e6d619ce7b73aa938aa7331b63035587d645e","impliedFormat":1},{"version":"b6ea8d8b96cffc63051a7c9e995f4e2236b37413958d6b421e8a1109d350556e","impliedFormat":1},{"version":"e7d30568e771aecc7b55d92186593d0b54238fa438a8d32d5e65cc5e086f2d06","impliedFormat":1},{"version":"03200c67d843971eff3cecac9aad1223c956a9c5517aaab15e56f84974f9cc22","impliedFormat":1},{"version":"8ec07cf076bcd9d7ce5965917bf7f00681f2faeda773a41929b538d425b0287c","impliedFormat":1},{"version":"d6503783c2ee47092c86a7244e24ebee738d98cf640bacc2a047a01d72624844","impliedFormat":1},{"version":"148b30bce91ec08b57b53310519a473c2146535948d2e9f66ed507b236c6d7e1","impliedFormat":1},{"version":"6db4629ddfd86e4a6cab3d8c37cfa9ea64558c7150cf51a85491c5595f415784","impliedFormat":1},{"version":"18ac0fb6ba31c2b39b750fb8508cd8bb0fe1ffa534c45af9169e96c50d02ea90","impliedFormat":1},{"version":"0ba6c5e2b5d570668fa700fa4989a2cf0478f9eaa6f4e068b808641d747ddd74","impliedFormat":1},{"version":"56fa44665cca7c25b7f88a5771c027200e5ccc32a1a4182ab15ead1b6d53c693","impliedFormat":1},{"version":"ae93912c15f3df75766e7c50a2d7b0d89f41560f0906dc1a70ab45d3b3f30bee","impliedFormat":1},{"version":"56886eab67b8aebec58790f12d701573eba1a69c4f370914a97a8b9c29049502","impliedFormat":1},{"version":"fea7827f32ef9ab636f46b24db0d928ea653aa8e337968b3e6a1d6371ea57de5","impliedFormat":1},{"version":"dd66aabbd86c4044cd215b2d2d4b2ff58dfa06997abe9940a8e92cea05289cc9","impliedFormat":1},{"version":"261318119ec3cb7c7994e9f6f3711d16886fe603fa88367b7c12a6fcb928b17c","impliedFormat":1},{"version":"5a5508864d7f56fff04cc012a936231772e267d2ff4bc979288731ce95d3b567","signature":"3896d43263187359bd05f07f0a7b6bcce523c21a0ca699bd286a84db330678ca"},{"version":"5a8ee51534bf078daeb50c3fdebd4f4c6415f1d0e7124f4674e935aa74e5fe0b","signature":"16a8120891d1e79dbc20ba9a47a26ed7bda080c6ff43de69414977e32f3907c0"},{"version":"6e9903b64d27b42b264d89ac6402e48f664fd8c7dd8f9f59628c2b1a920ee018","signature":"3334b621b44a4401d7180ccb59ea9fd928566c616c3827289836a5b0d7fa011e"},{"version":"e3741104d8f818406554e17bee0cb412c7d62a040d3004d548bb4db1f7c91538","signature":"9a9efd7230d914a2a78911ff39eb0c5d5f026bb7ceae9644ed5f158b41475057"},{"version":"d4896f4240d9f2dc41c53936ea2ae52cecccb02afcec5c01ca509591ff429a09","signature":"f8d2dca5d64713e2bafacdd11658c1cb3bf8365df03376e2ab0e83b7638e8b3c"},{"version":"2f6cd2806dde0c15057134e022b9b7888c374844a942b6e07022c94357e8f515","signature":"da4103ad4b4ef557d6deca0d6f6fb0344503b27d8939c2438126a5590c323bb9"},{"version":"d1f5c6c5f6a6fecefd8edd05962f6b05878783278dd9aa11df372b7bfe5718f5","signature":"746aa7164f34ca4276e9f9623563c5b63c3675153b1c83796b812de733562560"},{"version":"96454ec511f87a6013500bc6bb8b7107569d853570ace2413923457286229e74","signature":"4af326aaf843a09ea14287503d3139270667e40740625a74679bb0a57d96945f"},{"version":"bfb309d2cf7c1d004b98eddc388db0f7b51e294f2af88569bd86e761c4305ba5","impliedFormat":1},{"version":"7d80d85fbd6b4e0fe11dde5fcc9aa875547f1ec1a499ca536a39b55d4e1ba803","impliedFormat":1},{"version":"f758fa994a025fefe33dcfcf68d89ed5209b53443285561e5bfe547f770ac381","impliedFormat":1},{"version":"f611b23dfebb4e4ba6fd4f519180526491a72aad2289f7bd8393556879b37502","impliedFormat":1},{"version":"3a93e73ecbb7a89241c58fcf30ecfbf788c3e98d01f5eab4573ce0f8635b6506","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"81bf23e7aa704d44731f9de2c2ce4a8cf5b5f29b64f54ff775d3e789fa9dfbca","signature":"275bd0e37faadc815ede8e48df1d244e13cbb448ad0db2b32bcf5e798e9c1e8c"},{"version":"d6f2c66b20557b4d3cf962084fbaf6f7798ab6579bc6e0f61b92d4011c885590","signature":"1e502394b0bb550506cff92085e179affb5dcb03cf633cd7328c4cb752f610fc"},{"version":"c47d0d4b548e155a7783d9ebb6e018bc692054d15a9c0b57002f7ccd0618a8da","signature":"92a2114a2f3a09ef7a829297dbca30126c07c4c68a191cf7afc3e69d6ad798fb"},{"version":"9e0d231988cf29ecc876383ee862becd1c07f42c8ab8e057b620751f8796c39a","signature":"f0534998b964e4a2e3376304b4bcd5e1c56f82098a6ed5a720c4e4ef1a843c81"},{"version":"8583985d6355de62b1682d0638ebf25cede92ad07ab7eaf126eb738116a19b9d","signature":"15cd5c7b7b7893ceb0e96c23479df682f0330e15f607e147528353f5a5887fb6"},{"version":"02eba330df531da2692926d630fffdb02a1d302eb2a461ace652efb6fa9fc745","signature":"a195017ca2c5abb7cc7a80e6179761e7cb375f66dcb4e6b0824e60b6e8c19aad"},{"version":"10299cc8419102bae2ee57ec8194159f8e61ed0b41eb835502f67463f39c917d","signature":"d05ac255ea2d84cb5064c988bf07fe8f0c28f01ba3e5beec2ef22611cbac6e56"},{"version":"caebe63dbd0e4854c50462f473d4fdb8cf1c101bed0402d2b1630c1065d4bff1","signature":"300b9c4681dd15ab03bcf177974420d7997f2184f607081d1d978d4f315903f9"},{"version":"c93796675b1aeabe076f6649697526ea4b100c4e8513401d279ebefea50a58d2","signature":"bd763625095d0954358c2905447e8464fe21e7fb5bb9efde5aefc727947bc811"},{"version":"8168df8dbe0c2125afbae3ea8b0aa73e410937fad708b308fdb48b60bf4643fc","signature":"68d3b3d53bd8e24bbae76efcdbcaf230a58cdc6eb1cd002aeeac5ce6fa113f3f"},{"version":"7417b12dd15b441d11b6b640ce3924c1fcb85e32f382dff8de9f1565530d5b2e","signature":"8667579f1e6830b63f4ebd5f279f516cd08048cc75d1c47a01a2c4fe6f9dd4d5"},{"version":"98b549890b6592b8726158ab66e8433e22a11024a1615cb2de7b58ed6abae33f","signature":"92d091a6e6923b099c2f72748886f8acd79033e7afcf7b86477268534d79ea46"},{"version":"cbfb05714708abb6d0501e93e79ed0637399dbe66d2652d3a6032c47d4499285","signature":"ca30c38317ab4d7a409549f9635f74fc5298936a2a0362ca6965dc4f2616acdd"},{"version":"4dbcd4cbf87e584b4df2fe13e705c2edba718ad935971327198e6086c87359dd","signature":"97de9afcd02c050e374965ecd2552cae8f869d5aacff7ecb729a07f99dcde7b6"},{"version":"bd61e63a479c9c986a9c1936ca4ad01dc0bcba7acfa0a9ecc45822d1caaaaaf0","signature":"3992a426d4245b7bbf9d7155e07956594e14a6f9c9bbf49d20dd5a2baacada35"},{"version":"69d5af5c63eadd37c8d1bd4fd483ab0f4c7e2258b73bddb2847fe0611abf7d70","signature":"110d078d44505285b016309398cc9409c51e93101fcfb4670ed8af3340969f65"},{"version":"0aff60916520168485d0fe7cd35ce53b68380e619b232f93bd46a3eb73413057","signature":"ae5e3661cc55f4702836b7f578ef58bdf1b9b67e72b46981d8c58cbe4cdea3e1"},{"version":"07fba9cdff4b62ad676c9344e41e6a770e19e01fbb5a4d98e9460a29d2f78266","signature":"cdf8d87461e48d0c7c5a1093ab8eac9c2b021937e8f7aa00a139b7ae5b860285"},{"version":"fe26253eaa9ae90966406ae05e330190364ba16e8b7b9a61d7280c0f18c5561f","signature":"ca04659b8bac5547f95e0299caecc78b6051bf16ee0107678c3b50219ead5245"},{"version":"bafec7b5b2054503ef681938521b76fcdb4dc8c8f0766a13fc7ecd6b3005e94c","signature":"34ed5991593aff2a2d2985842170305560c5e4600f23b833084703a2b9cfa13d"},{"version":"e7922373198975bbb60de575e50dfe90cea8857559d75fd53b10e264a5d6818b","signature":"70b4215014b280b117f5a1889d31bfd373d951a46e653e50eebf7ea08cb64401"},{"version":"ac960b3200290cfd1b19f496f03349721a869e4d066db278063ecf63a0a235f2","signature":"7a3cd35a8f43139853a4948fc06ec81499cdb088bd9181ab94541b0a6e4227fc"},{"version":"b5d9f6d2f1ce130ef0cdce9920d8dd5ef9c1e2d96cad88cbabb50cccc8265379","signature":"90a8c2d412dae927dd1ff88e236311cefeb3defb824ada6db46c360f1f925285"},{"version":"08d301c2f165a60d7bff1e129a2dbf5d78f117dc27e1cc5a6112c13459e8777e","signature":"e3c3d01ef504a79609c20c5420d80ae70aa5a525be21f0a9304b07bf41be76a2"},{"version":"c2d3d3a566d0c25ef1b061cc050dc6c6e5576520c9d93ceed0eaa9cf76a9d7b1","signature":"77af8c159123f4c16e8af431dfce6054b88b116e09bbf982d61234e17ba7421b"},{"version":"70996f2dddb2ec0efa7a5fe04c5b56fd31e1aeff8b25e5e428b36ae8724cf2f4","signature":"db3c40f379f214e34124777300fa71e147a7dad7d392e10eeabb7a2a7a977f01"},{"version":"a66711d90cc0908c1c1cb0a748de7de644cb0f10fe74aa24996ac38a4686b8d1","signature":"5c9bfa84db250eaafa48cfd7f6758c382a8d55d14db5e8e07819d8fa55093025"},{"version":"078252552b5a0655aaf671a763fbe350738008f42bc0feea15449f25cec05cf9","signature":"27b7ff02cc68250ccf2750dacea10404c18da4e99aafa92db4aa18bbe7789564"},{"version":"bfe2ca7c83d2713911a32abb946f197f04845f8b5c0dce50016a84e8c5723a19","signature":"5e58a2e33d94f1a7f62398e19370de83d6dc0ef2502bee2a8f4656aaa213d8ab"},{"version":"513c99ae396e65702d1b86a00d3f1d9de5c0811ac18743021ddec2c0f5b928ad","signature":"2cbc6a183222724d6be31dfcdaedbfd4bc6a17162230a2271aee95ce1f7ea67d"},{"version":"109e4b9611a522fb3b921706ab978ccfdee4ff86e8d82bf531d36e2c1f538e49","signature":"021e80a91d4074d758f5e034c2068ca7c4518c79bc00a37617f4e27159d1e08f"},{"version":"d241c9e2391c0cd6aee5b563397edb84a398ed49a2b2d258eb0da4e511ff0c3b","signature":"7b0a4c368a8bac257f878a2a68741c01b164f4b614d42037870843e65bb83b1e"},{"version":"1fa74b338a0373a1abea910f587dbb2750d4122cc33083e7c543413c0938e861","signature":"8372c9363817ef009572541d6eba3275b2e527cd9ac83af0d93ca55763f8c76c"},{"version":"73dc387e5aefdf90bf2acc9bca51fafdfb76e20e93f8693d8d681769ec5f8d49","signature":"120d66d8ab777f39ac9a93dc999dc6643d3ce84f98b7ea12f5d972e4f6bfcaf7"},{"version":"d09375d0b8ae3a3087d6e65b5315b85dfce9bef32a67d7e371d81e1485ed6df6","impliedFormat":1},{"version":"9b94545ea530fa6cd42bd804c67b323722bc52a5c307acf0745963647b2cc536","impliedFormat":1},{"version":"4de12e15f46dc063a7d8f35fb0e3f20a21b5363d9325321fd04aa45bd7fade7b","impliedFormat":1},{"version":"9e008fc8fd36d97bd94efd91794ebef485d21322769ad41dc2ade3ac93b500e9","impliedFormat":1},{"version":"5662b1180edff300f59beebdc535d28e63f425eb3188a4c13962513a41840e34","impliedFormat":1},{"version":"769f6ffa952c42f785831c23b9ad3a15d0f4f5a01965d869cf152d5324d10b1f","impliedFormat":1},{"version":"e8232631fadef3d175e6166d05412bb7745f236489c722fccdffc945b7a8c66e","impliedFormat":1},{"version":"571593d0fb872ff3f69a446cba068ba2dc738daf753166bc700b966a38e1615e","impliedFormat":1},{"version":"802247498d7d71a0ebf77bdab5cbbef06ecd3118bc5d35d616fce928b19a52a9","impliedFormat":1},{"version":"520d605654bc634b8e981268d4034f32dcfa15749357660a52086d9ff6417048","impliedFormat":1},{"version":"e8db60b139d70d1a4b6585c134ffa6b198ef45ab8f077bc75ed85100a535d6f6","impliedFormat":1},{"version":"d624c42ac2874e246f3d3d5b870b294db926b740eb49e41076faec28c65b6ae3","impliedFormat":1},{"version":"43ef1c3e7572e2db8b19d70805800279788fd8a21ef25f326de44a23107a2445","impliedFormat":1},{"version":"c45d005d37db6779816fdadfe3bc594a06109070502f6cb7a178939a8dc71e2e","impliedFormat":1},{"version":"a79975a80dfd05111a9ab450c89e940fd3cbfab72c5c5acbf5c083a676cee686","impliedFormat":1},{"version":"a51c8acd33d3cb34f88328ff2ba373bbcadb94a7aa109db85936ede5aa33d769","impliedFormat":1},{"version":"9f45d143097c192941d2233dd1f5631db1c354b479434443c24a68f612e35756","impliedFormat":1},{"version":"cfe7d9d063781df70b0d0bf6532e0db215bf78df8bf89cf14643ced82cec838b","impliedFormat":1},{"version":"7e0b695a04d63fef9a29271c43dbc0134731df95bafafd02fabbab958d11b84c","impliedFormat":1},{"version":"fafb62ff0d867f9cfa65772b844ee315b024ed1920d99803b48aceb4461934ef","impliedFormat":1},{"version":"00a515caf829319169afedab7105318f3e0f84059be730501f65e55bc2aa0f54","impliedFormat":1},{"version":"d1f1a4985bcee2d1ca4968124bcc941adfbf788c153d23da9b3618cc6d0bdbcc","impliedFormat":1},{"version":"b526691dd48cd60a9b3a0948fa2ec362abdf76b7075ac0b63d9a1d287a1412ac","impliedFormat":1},{"version":"32ebeea9bd221b5e4b48d00dd464ad8f808bd6e6f122479083a67493650ef4ba","impliedFormat":1},{"version":"04f24e7bf3b4757881dcbba7f6a7e76406c13af2c39cc7443f6e0a8c4333025f","impliedFormat":1},{"version":"69ff38c10170b949db2fdc1bbdf034ce10c53dc1e928c46f557f764d028b58cf","impliedFormat":1},{"version":"7e6ac692a05e88acf9bd70f8ecb004601724639cffb744cd6c6f00184b291dd7","impliedFormat":1},{"version":"97b919966e292a86c64d5380d1e740229d643401edf5ceff3b5b4c317ba0f10e","impliedFormat":1},{"version":"a632f74d889bdabc3fe45ee67669f10053e8f7e2382b73592b602415f47e2cc8","impliedFormat":1},{"version":"545bdda99bb5193b7185be42f9dcd4ad220623219f9fc55f67545580391c7019","impliedFormat":1},{"version":"9243758ba5395f14daa32abc91f57bd8bdebe434c0fa9d032167423d92144617","impliedFormat":1},{"version":"72d8345cfead0c29e8b4d3bbe7af86c87ed50af726403e9229adf69727c6fd50","impliedFormat":1},{"version":"238f11f508cc9fbdc6ce283680cb1c1bffa83d662f5588479205fbcb1c5b0039","impliedFormat":1},{"version":"d097201802fcbf25d79999817d7fa0297c9c3f8dd16ea4addd9c0df82f22496f","impliedFormat":1},{"version":"a56623b512c17758214fc7d63f570bdd649eec80b90a45150f3fe7100fdc91f0","impliedFormat":1},{"version":"e0f49a9341d2f448588783315482cc4579bc49a8003f001b40717d206bb6b29c","impliedFormat":1},{"version":"97d47743ec3962e85b183e3a82f92cec29fd77cc04e80ee4d8daeabaaef83cbd","impliedFormat":1},{"version":"6e5cfaad17f7186a2f37e795eb858f38e9a8ca659b51f0700339a00b4eab4c23","impliedFormat":1},{"version":"b32fa7e38efebe74c50ffd3b2bc03bf128f3318dee482562cfcdaf44a50ebe2d","impliedFormat":1},{"version":"6e7cd9be73c8f27300383838ebee513db30469afa6848b37a2935557418bd144","impliedFormat":1},{"version":"157d3c779e63d55f6c985c83384b19d32b604296c051499442a5f398e2fe32a7","impliedFormat":1},{"version":"86a7c300d71cadd326e4299133ab3f384e56c90c48abb9afbcbe21164d2247b3","impliedFormat":1},{"version":"de55125845f2f7617549a4d091bdcd526a8577e26203b5295cef4f6dd826b1fc","impliedFormat":1},{"version":"8485c13294467d11debb2ce65e2bd5a1a06345d664ad759780cf9ea0f352614f","impliedFormat":1},{"version":"292d44873aa2568c467dcf4dedadf666afc12bc72ef19f54a8dbce1d424cc64e","impliedFormat":1},{"version":"5954f628e30fca5d16f0f944104307b71510a35ca40cc779b73aa64da5454912","impliedFormat":1},{"version":"ac21ce340a05038e9bea7b94e9f471a0bd710d412329733253832313b52ed8e5","impliedFormat":1},{"version":"daa5a337ef92b665c4d3c7d1b423b42f8437b028973e04656d67cd90d3609f97","impliedFormat":1},{"version":"5b544f6cfeac0ab86e60fda7f8df2274ec269236cefb0d4de414cb911f8ec366","impliedFormat":1},{"version":"50b6050a2a9e32fb09d4bd87a043d1e389f72d3ceaf84c89f3a433b0bfa8b13b","impliedFormat":1},{"version":"5b94bd8f283e44e5fa59ae2ff9319201877eab8dc4e82d1b9548598d9836e562","impliedFormat":1},{"version":"0abfb15c92d4296ebc1bbef2be397da7e32ae115f368e7bf3183d67108f74f92","impliedFormat":1},{"version":"91356a9d3bbfbb3d6cdcf2bf266d164a4806316f41b561cebd439af209cf6b67","impliedFormat":1},{"version":"080c6ff951c38f28fa8d874139d4db12844acb1cc7144c574cbe58279a1c609a","impliedFormat":1},{"version":"86fdb2e51fa437f8ba429840263fba58bc1b50298667749003cc919454dd63e1","impliedFormat":1},{"version":"a2f3810fc6ab7d0c7cb0eee0aa5d8c07b3f845a6094e7b69ec4f8df22bf9dcec","impliedFormat":1},{"version":"18fdea817a1f9b2e4ca9938257e7074e4ba0c1aef5c07c77fda2f965c5710450","impliedFormat":1},{"version":"04e4c00c3c81419a67c283f67b0f7568f1f52f321ba45ab08ae8ccdd1eea737a","impliedFormat":1},{"version":"f20a3f58b03d543c625ff4219f86df730962a14d75307c14f1809851387f18e3","impliedFormat":1},{"version":"5552811c7987315c84381b934f7ce9168dea216e8979ff761158129106c06f53","impliedFormat":1},{"version":"53f8e61ff1ed46228ad430a8d6c8b4d8c2e7d18764c6993cce1919c35ea5b968","impliedFormat":1},{"version":"1de5a81eb2d35f69be1a241d5811f3f77677d194d8e17afea2a6cb1364aa1843","impliedFormat":1},{"version":"a0fee15eb5957c146aeb0e0b9063c3722630cea51b338e2143141eab4d413967","impliedFormat":1},{"version":"c6139768378d2250104f2430f41623000b0e936b057a87e9bb6ca74c4cc35667","impliedFormat":1},{"version":"10558063f0741148d4adab4decae8695efedbe2dd3f7531948b3613ef935e02b","impliedFormat":1},{"version":"c81a7d5024970bf34cb4e0553d6554b864d2d1ee9b1cb9efd89129d4c3dac997","impliedFormat":1},{"version":"5f65004422c2d5e97f2050075e03e22c4b30bc942facf4f0e22dc4c4e5964e6e","impliedFormat":1},{"version":"a98ab8329757a47b4994e17005f85fccc187c433a673b01734f9c6d7df9291a5","impliedFormat":1},{"version":"03e5cb0828c073896ce4ddc5486af04a0f1dfda965ca8d40e83cac0379c9b36f","impliedFormat":1},{"version":"72179ba23ae57a181141f641b8a20930d7fb0cbb75decf76bba539f72720ed21","impliedFormat":1},{"version":"9dffb9c5958e915e107316229b09d7ae543bc71cb4bab77fe46b62019f3f7ba1","impliedFormat":1},{"version":"95b02dad10c7cfe9727856ad7b863bc98a794d9c9dcd78fbbef0d59e82138e1d","impliedFormat":1},{"version":"36ce8f4245fd0ebf1563225b5b6d6255aae02956575659f07a1118f273cc61d8","impliedFormat":1},{"version":"487bf870931b1db1f6f9b05455fb532d484535a54b89ff5e92cdd38cbb27e4b0","impliedFormat":1},{"version":"1bad797b2f31970a1c2095e694586a1100c4bf90ecb3ea21ac766f7c5ffbb5fb","impliedFormat":1},{"version":"014662f66b41db86f71fd72454b37d334f320cf359ba22474dbf462b90f92faf","impliedFormat":1},{"version":"08f11d512e5e5566163bea17f0a00ed178ed80f92ef54961178f9d38e0307e65","impliedFormat":1},{"version":"e747a86b6011e157e4938f1653d0b9d49864c66af983c04a9cd8c676ac60530f","impliedFormat":1},{"version":"4e07e9a3eb8eebdd8604b03f1969b559e626a8ccd749b98a97da609fc1717465","impliedFormat":1},{"version":"9457ef4968761ef140a335cd82df0ce0c182fe34873f46eeba539b8b18ce7197","impliedFormat":1},{"version":"6d8e5b25c100edb24b96f751fc32e65a1e0bb5f4ebe3c37f248bdd4ac6205fc8","impliedFormat":1},{"version":"bcf062f7c610cba057f8266cf7bd9dce1ec618426ffd107c95718bcfd2b068c3","impliedFormat":1},{"version":"d392d2dcb4c249799f356ba61b5ec1cf1562919794e53290fe4789fe7056e172","impliedFormat":1},{"version":"533b5b627dc3ab444434b532e711f938ece553177314da07f22e42b2a3b25fe1","impliedFormat":1},{"version":"6ba1e80c263737a6f71ad46b17ee3b99bc6910b7e5d0ec18887ab8bc3df93df8","impliedFormat":1},{"version":"910feb38a433653f4f052913f92dfdf85b59777972067218cd373cf2c4c1f352","impliedFormat":1},{"version":"10509ee359d0b852f024946cd75a2f43641ba3cea7b764d5f0d04644d30376a5","impliedFormat":1},{"version":"6577f73a847552974d5a78a57855753bff3c48dd08c7218759030a1f9398aab3","impliedFormat":1},{"version":"c7fa3c5444d68c08f751efe1463831662acca5e985724d58573e977f6e42a32a","impliedFormat":1},{"version":"22dc5d3b866360756aa94850e330d4b03ba0e28aa56070f79fb3b9ab59b641b3","impliedFormat":1},{"version":"afa6e9f245b974887a05f632fbaa77b4b828cb6addf929c6c26d347642d1f3c4","impliedFormat":1},{"version":"2db262f19b477b62c729ba70cbb08b0dead495381c71564eb89eba3ca5aa130b","impliedFormat":1},{"version":"f3517ad5062e51dc990fbaf68bd74b2feeb74f0fa5d1d8d19843bfd76380b975","impliedFormat":1},{"version":"4783515107e6231ec0aa4daa73dcc649d9878a13581471d2c1eb1e07b33382ac","impliedFormat":1},{"version":"bca9e95b232380ba826a7b9a5691328c228be42cacec9257901907486a7f6071","impliedFormat":1},{"version":"00e85be72d85e468dd04fbfc418083abc740a1374f7d769d0f4c3c32ae191401","impliedFormat":1},{"version":"8a78dad5482aab2a0e7266e4b4243605a3a3de1b3e3cc5c4013f6d268e3fff18","impliedFormat":1},{"version":"046070e19c26a95cdb399347a016dfbef1e98d18bbf54e5443fa8427c1013494","impliedFormat":1},{"version":"2ab90c0c8dc7e54c35012d7045b02c26ecd73a1f13a0af3a8e1d9276cd84cfe3","impliedFormat":1},{"version":"cffb93fbfb4eaf5d8f90b5c44c37bf51625c0be1a850aa3e7f635d96f9086a17","impliedFormat":1},{"version":"2cfcc254bebb202dfe8adddec625880a660ca8d40b5f93826efdb2899bcf5a4a","impliedFormat":1},{"version":"7000e97fcb16524d327cc0c71310b81b4f2a8cf1ae09da5a46166c4f521ad307","impliedFormat":1},{"version":"ca0c4d6597919034347e0d18d7cb093b24d2832a76b7258e4cbc47d6cb2b80c5","impliedFormat":1},{"version":"298c1e539abf43d7699f3c56d44e9c200d1a8d6783595a9e9559f8829f9aff82","impliedFormat":1},{"version":"94b42d915abd6a0aee7757d2f634120cbfb2b25a261d7726d345dbc04d8a15d8","impliedFormat":1},{"version":"0cb778bb1c376970e66f64baa49b010349dd6ba6738622e940479bdd8adc3f5e","impliedFormat":1},{"version":"fd9d3b0a5960bec1ac5b17995198f9ddcec9f0f22d7f4d0cb5a1794d0015842e","impliedFormat":1},{"version":"fe754283ea9ea1660658034e18a5aa5d4efe03ebf18bfa3f1f6cae10dc5fd216","impliedFormat":1},{"version":"ebde3951d3800444b3932fe47bc66edee2022ffc1fd66ab2812006af0c1fbd6b","impliedFormat":1},{"version":"b0c0c004ccbdf5851beb48121cdaabb105fa0fd81deec1d38239769aa51c1f0d","impliedFormat":1},{"version":"d217a63d18c6235f73995d02ad14c2dd2c03fd226cab19188fc721622a0c0e9e","impliedFormat":1},{"version":"ed49d2e0c72485ecb8fd316cf8dff9dfb7c7ace8faff75a9244a74b790d9fe51","impliedFormat":1},{"version":"58bd448e81da3af57439dd71817cb06306753fd26ef984040379b2871aa0745c","impliedFormat":1},{"version":"41c11d14bc03cf4dd17e74e71a377bb221d167f50ada10d4364e3ada5dfcb4d3","impliedFormat":1},{"version":"b1fb50f63377766bed70615d76c7933f2f2674f14eb34b8e0011931a22cd0b6e","impliedFormat":1},{"version":"2556451f7bad6e7d62c30594cec691e1b71667f006460ea73da8acf04021f1b8","impliedFormat":1},{"version":"01e035ef4a842296517dcbdbf95d7e86c4ad90cbcd7eec3ceccd1a5ea60cc919","impliedFormat":1},{"version":"c722fb42b32b0d8af47ee6afbe044aff0133d9e23877f3a5e555edbe576a935e","impliedFormat":1},{"version":"bdf55e5467ac502b47f8a6dad728c049fc38419edf6f335900602c3017341680","impliedFormat":1},{"version":"71d6c834fc80129f690120ac1648e33ad8726f8374dd244984f662070df8b7b6","impliedFormat":1},{"version":"1dd3473de87ec322004ef3632b8855bc6d8fd27a2a42e8019d2cc83662c6083c","impliedFormat":1},{"version":"beb0c09ed3611843698d0c9bf7f8b20678f3f2ca12a7e3563a1f0c19249970a2","impliedFormat":1},{"version":"042d6ab69de7a2f2b2fc50c990e0f63aae492ca99dd402756a28662c0c4aab8b","impliedFormat":1},{"version":"484b879b6d2e268baec6f63dea3de40eb7933ba6d13dc659213f79ef30be3c3e","impliedFormat":1},{"version":"6748aa4c56b72d961cd17357cafce25459c488f4c0f58c4c814da61aa955b185","impliedFormat":1},{"version":"0630a3c42067ce65f028bd2aa1a792611c4a1473a72a5af0e4f13cd361aa8a19","impliedFormat":1},{"version":"ecb05882658fb9a3511e96bbcbf5afdd6ed6211fa8bce9621beac9ac251a9c48","impliedFormat":1},{"version":"3fefc78331c4b1cba8e4decc3b99a60b20256fced4f7ab82a83c9876fd91d288","impliedFormat":1},{"version":"c0e5e58ae13b3e1c124978538ff1f37305d014a02e72cc1e13e8bc39c19c2ec7","impliedFormat":1},{"version":"d1db5308bfcdc01a1c882f9adf61f15eea173ed96c0efc89334aa3f5f4554028","impliedFormat":1},{"version":"2aae2c03fcc398cdeb2ebdbe0b2ef75f931a3047bbaf3a7e595b77a1099c3a62","impliedFormat":1},{"version":"74c859e463a329aa5979fb2564d62c539c73872d486c059006fa3451f89009df","impliedFormat":1},{"version":"106c8c8c329985a63a36913def4cb31e8b5b3ada152796cb053772b4875f11e4","impliedFormat":1},{"version":"7f5eeb603ec42ee6452bac0c5a6fbe4bdb2282b0679e29a27a7a8f27eda4ff2e","impliedFormat":1},{"version":"e778ff5aa4e8690b60f314b3cc069129bd925be0d0b725efbc2e0daa096729f7","impliedFormat":1},{"version":"f01ccf24ea873eb216a9d97b80a80f438eefce6231ab0e815092763ef0349c66","impliedFormat":1},{"version":"1ccb5f714564122fdcb0165c4243727ac8391d76696693bd936ef04ce10a1411","impliedFormat":1},{"version":"f45fcbb319bfba9d2e055a25235485e4e84d02d29e5048a5ef641d11e23f8682","impliedFormat":1},{"version":"2900af127a837ab452778e2f1c81b29b283bf21127e0fcfe72ad7be4791229ca","impliedFormat":1},{"version":"e97079d84be581b287039d077a863c0852d8155d66d7f33b7227985bd60f1e10","impliedFormat":1},{"version":"ca772a303716d324ba2fbdd37cd7b6f6df4dd6282576e5f11e6b99c06b7566c8","impliedFormat":1},{"version":"66876ea4dab4349d6bfc74ad4801e9c59ae15a3ad55c0bbe09aea172055cbdb9","impliedFormat":1},{"version":"d89854c62871dfbcb5bcc617d9e70dc64fc17b00d6add0a450dfed1426a749f8","impliedFormat":1},{"version":"08b381422213ace211a0e21b7a39387c73c23ffb38de2c4c96aca2c816173323","impliedFormat":1},{"version":"0b03f4f2c255d5e0a22f958b99860b151ac86b2ba33afd0518b97252913fc71a","impliedFormat":1},{"version":"97d64a33cbc5d29e3109818b61e7b8bdd5e9e16ef602cace081dbbb30dc78deb","impliedFormat":1},{"version":"eb756a53777092f4aaaa7446c1fb120a6b5192b9584683c2b93f11b1f7c1fc94","impliedFormat":1},{"version":"25716553e4d38c13c085a146a80e2d59880e41c8bdacf0f35b5bce7c40df1b01","impliedFormat":1},{"version":"f94acbf4bde891da6f4e5733c9a0a89d5fd612556c5c55a78e481e409aac3729","impliedFormat":1},{"version":"27d444659719b3f998d3abdd948daa29b61e17cd156fe215648b45c2a01c4acd","impliedFormat":1},{"version":"602e5a8fb7503cd90bce7bdb89c8991ecaf03b959ea011dbe15d820a92003c4c","impliedFormat":1},{"version":"aebd9a8270db1d6e4180bfe23d291509e0b7b3346c6c3a74b96063e229496f7a","impliedFormat":1},{"version":"a16c109042a159260ea652e6f04869ca2cd802d6e29e88c460ebbaa56bc49937","impliedFormat":1},{"version":"72cb37e13f5f75a6342873acf3b5b9e05541f4ced091ef9377be76f0d38198e4","impliedFormat":1},{"version":"c9bab1cb74f13513911d7ed9b6f54c6eff06954acf3d93ac3f63673175bd3c08","impliedFormat":1},{"version":"aab2cb5e15dba1be5c2bde01631e6ab896cf2f939360e20faa6bf0d6184b75d9","impliedFormat":1},{"version":"040eb505f604c7cd6d19784ef39a28116688423757e45bd205cf5c180732c21d","impliedFormat":1},{"version":"ecdc8ac5068c76f56ed409332b30f18431303689550f29db4459be42a8d57759","impliedFormat":1},{"version":"bcdd86f750376e91b8b20fb49f89651254b10673925835db5e15bc87446f57ec","impliedFormat":1},{"version":"12a8a4c86e052cd8826d49341ca9dc6242eadd692c41b6aacc991c4fc3221b39","impliedFormat":1},{"version":"f2a81818ce964220422bd65814603aba213c97c9d3e288c906c501b13c3a7e9a","impliedFormat":1},{"version":"e00b39cc3da81347fc71742ce3c368d5d698562968ea9d7283aac177e6dbfa4e","impliedFormat":1},{"version":"4c10ac04b1ec81b106c268ddd1a230167b42910ffdcb366af7c8a9103c04977b","impliedFormat":1},{"version":"a5726d11d9c2bfb12713a2f24944e37084817ea1eff815ad3a37bd831c9dfe94","impliedFormat":1},{"version":"3862bd74c44b62012ad5026eda4655ed9f80d9a51a60b4a75c78cd672849a2e7","impliedFormat":1},{"version":"c868836435628bfcbe89c5b26700aa35e19f1dfb4a2f48705aa5bcef5e48e58c","impliedFormat":1},{"version":"be4ca0869e80b17354edbebf4ca6089c7d2b0b94cf43232a2cbf51576a59971d","impliedFormat":1},{"version":"7998940ad8fb5f194944544c411f4501e99a751c89b48d2279a84a77f412e05d","impliedFormat":1},{"version":"96441db1cfc4321f1ef2e4d49f38f142df0cb99aeea3fe2771f9e0467666e4d9","impliedFormat":1},{"version":"5e96adf1673b37fa57e1e61c5f2d6449dd86f63bf396870338579f5f4b83b82e","impliedFormat":1},{"version":"15de2dd257478e3b928c67b86f20489309aa2ac332fb63d0d700d6d07a50235f","impliedFormat":1},{"version":"84236d8d439b9e35fbef52a75b4f839d50a9a0f3c7186a6adc04a082713adfff","impliedFormat":1},{"version":"a8bafc1a43e787afca38022e0abd4eaf835bcab89cc45b423f83fbd3a35eba20","impliedFormat":1},{"version":"62f2c846b033c528716e5e75ad5e6198ef30e96b52cac1072dfeb44fe3e2e332","impliedFormat":1},{"version":"ff8ff08d83b50336c330ce7e09058d249c2718606a79359e18f279c375f4260d","impliedFormat":1},{"version":"0a34cac0ab7a31ee9da7c1e3ca5db9460bed3c7055c811680292c84c95fe2a08","impliedFormat":1},{"version":"57f5eb93a6662cb82d2c6de9468a1290227a5b40ac6b1d00a5d1bd1cd32b074f","impliedFormat":1},{"version":"a7ae2e27a082e3608df88a7c540b16c430c54c1db42f630d3428198778196d10","impliedFormat":1},{"version":"ee233ad3f4510c003e60a0388c652abe363a6d9f8fa76829eb2829208ea294d9","impliedFormat":1},{"version":"1aed5213987906e5b6823e651a5b943b8ea6ac35724967d5677a6ab5ac92da3b","impliedFormat":1},{"version":"5a8ef4c2202ed36f937660ebfecc93cd8cae285291297b1e55d276930564abca","impliedFormat":1},{"version":"ccf06ef2d3ec5fb22b25389e22b688b9bad80e9f8066d6c582888f05b73afb5f","impliedFormat":1},{"version":"0b8ffa2f5624dded72302a105a47e7d935f73294720eddd501c1d6c672c25f44","impliedFormat":1},{"version":"d81ad031b1862baeeba60fe20d672b3ba54dfa0d975cedf52307e6281daf859e","impliedFormat":1},{"version":"a0f29eba1654dd97e3f419481e51b963b367a5e13612d8e6af83542e5c396801","impliedFormat":1},{"version":"cbe7a8cb6b71cf29eca8803798b0fee90b71fb6e10e7abc7c56bfdc225b8c567","impliedFormat":1},{"version":"b47fa7ccad7aea4d0c48b6f25705c253cef3c682a6476b975c7b163dad4bbe06","impliedFormat":1},{"version":"0d1eb7e0728f88697993119fc18c0a31cfb99b928a28b61317b94bd07b6db2ba","impliedFormat":1},{"version":"b15cf570be51c525bdca8d1a6af80f7ca814a1d8eff02fcc0a90c3796d0cb80e","impliedFormat":1},{"version":"f1e0a503d7be571155b2ca2a3ff758a8906e736d54b2014e7e607f83b77a3414","impliedFormat":1},{"version":"9bc2b4e914e8f47546da8f6fd41adfd066a7924cecc808ad90c5e12f898e1003","impliedFormat":1},{"version":"abc8fcffae5e218a332ff53c3c6e3b5a0f462fab23656f6c068b02f70e8db16c","impliedFormat":1},{"version":"052f2a7a8d2f15cbdd5f7d7f85f9f6a8534743e7b467b611a5135c9e1ecfcb24","impliedFormat":1},{"version":"40c17eaafa266e9579865b0936f4099c9fe52cda31037a822eb53e7201ea6afb","impliedFormat":1},{"version":"6ceb421008ba131ea222660229988d13ac432d119a3129106cdd33f99246b8cf","impliedFormat":1},{"version":"20e71fbbcbfb66f81c52b3471d97e059a7333dc55a149d69945889758e89c3b4","impliedFormat":1},{"version":"4915a72d81013ddc15f5fa5e21f77ef37b568df0541674c0e706f5602a61f633","impliedFormat":1},{"version":"f0d40164418127a95d611f1015d38fa68e438fc0083568936d923b49bdde843b","impliedFormat":1},{"version":"708d531e1292e4f313c6f5b659d42da70cdb2aa4aff1f1e2bd824076440144dd","impliedFormat":1},{"version":"70030889d375c18b9c238cf0e5653f2e849af7e3e97b4a72563e56121e3981c5","impliedFormat":1},{"version":"d1e91e42db77de51b4f2640dad51faf46a98d569e9ea0cc0a90b23b4d89ac8dd","impliedFormat":1},{"version":"b0aec9ad94bed92e08808b6d6ab579caacc97dc4c4bfdd6a84e47dab789efca6","impliedFormat":1},{"version":"fc6b08ef226b0cd1fc0e25491d4787f8f53816c6126aa166f88044db0cce8a42","impliedFormat":1},{"version":"aa3e69f92d8a3e3f85799e1508b99cf28473c94603a70ac2f3d04390c84ed6f3","impliedFormat":1},{"version":"12019613c8cfa8b85821174d369d63efb775ca21f19254e75055a18a61c3b5d9","impliedFormat":1},{"version":"26fd823e635748293190cf4b531df4a6e4e70d3fd08c83721c8b4b4e9902f849","impliedFormat":1},{"version":"29aa7d0616d4a2b985d6b9156700fb51ff56b6cab5018486bc7508df91c903d5","signature":"9fd4d121c749796d02b13ccf617ddb5956dbd544322d10dcef3077c41b78395b"},{"version":"651fa70b8cab3dddd1e1544bf4e8edc739e94fd750a76b86f9e362d4559b6de5","signature":"90a809f8abc9e4d04aa485940fc6a12190c043961f92e6d74dcfb354f21940c5"},{"version":"e2c5284f904e3de5f2470e4a356178ab5809706c2d6aedad5a0120d4da90152c","signature":"173e778fd09f556412cfa787ffedae4b01e3f9ff612cf8f00a9e51711ca4d496"},{"version":"8eda4e02cedf921b930dfd2c3e20179f92ecf04423a0ca546c6d2d6799b5b475","signature":"27199489d7264aab094cc27e76bcc0e5c8f1a765b6f5097220211bb78865f89f"},{"version":"7d2269f195d2060aad03ae48281fa2b90bd10de77dcbe1f4d19c8d16cb2095d7","signature":"2794bce0a31af80fed0f77a8db58a4b408fed6347e61a1982365473a469314a4"},{"version":"c64049d8f85344384858cfa7ac42150d120d670f9a0ec01f6c4f24fda77a1cd8","signature":"7f038a78ae704d98bb928eebc97d69074bf8ccc36d7acd7be12fef0191cd6dc3"},{"version":"5e249ca1ea0d43dbec4a87585b0b1d8fa4f94ee2e5feead0693aeb78341093b4","signature":"4a70b2ad16f07909b6b10277e1e76900d9e807775cb6321c90a3f590261c6c09"},{"version":"5315a00e9f90009e83ed089d4c9a657c8fd10e1cfe1ee8484dd4be6288b8580c","signature":"695b4e1a5fafa7f275c5651d567506aa60453465405eb5af36a1943f5b51320d"},{"version":"eef4276e518de4bef04708dbd9ddd31a187652c8142f1982af30f1f6fb4b102d","signature":"90864c089624ee890d937b1d5f80e52cbf79acba4242f9830d1f7523604fefb1"},{"version":"6d8b4a1ff8b81b5f9c9d37234427435c18e4627364ec105f1caabeae5926a995","impliedFormat":1},{"version":"545669924f3f3d15a1b948660a5af8c815184616e3f4f206c29fd8a1d2348459","impliedFormat":1},{"version":"195b43fb03896d064b88cb04d5f749809f6366415a6e433b70df295bc3cdbece","impliedFormat":1},{"version":"0ebd8d27ba65f83621e8324644dc19c7e64fe3a9d979137cd80ebf40e28b0462","impliedFormat":1},{"version":"a4e037aee1bfbe770ba3bbca72963ad00c8eaea2aade8e03cdaa126a8a1ac60b","impliedFormat":1},{"version":"8ad2f4ca786f17b3aa031e7474c7270e60cad48f17c0e3f4655ced6a8d22eb19","impliedFormat":1},{"version":"1847ed6a92c0558751560e0c8d84cc6192244266536777a5a23f6d7e2a7a0f1d","impliedFormat":1},{"version":"925a01de67c40bb0536e511bf51721f5af6ece57b97d202f9a3d5ca9e3935591","impliedFormat":1},{"version":"dffa2b8d7f35ccfa857bc50f164b9535dc04b8e3171c60d659073577ae593558","impliedFormat":1},{"version":"9c52064fc892b8e85017bb32400c5131c470b0a67caf734e606080712e3b8790","impliedFormat":1},{"version":"c9bf780f99a02f19e2ea5b06856774263dd623c54bb5196e5aa7d11ed0e6c846","impliedFormat":1},{"version":"98034ffa6b681874efc880f751cdba60bdebd485e98822ecc4b39877781de1ca","impliedFormat":1},{"version":"c7cd5cbacd5f544c87dc15b7d41097bd7174dae26d7d0886ca695130de3ab23e","impliedFormat":1},{"version":"04342b37fe58ab10f85a82cb095d0ac7603a38048b742856a82682e42e5374b6","impliedFormat":1},{"version":"b31f82a41bd1098d4564074dde6beb2aee7dc3923b4f036e3fa3af2ac919c2c5","impliedFormat":1},{"version":"bdf095a90885e0fe7b3c4107baccda7ba734a879948d8536d7f69f0142299cb4","impliedFormat":1},{"version":"02467e832e085a7868dbad2a24ca24b1b64f2355905914403c0004ce72fc2e45","impliedFormat":1},{"version":"7720346d5c7bee837106301ca87d3d385a530a6e81251ffadbb3a7fb87a2bc05","impliedFormat":1},{"version":"1a20e2cd64c713ba19e3d8d68b48ffe8e3552674f0d10ecaf954b9595e4054a3","impliedFormat":1},{"version":"cbb46766815bf68c1831c2cf0e9ee0aa9c4ef9fb1f2f992a1ce2d20138cc3c1c","impliedFormat":1},{"version":"1de21a323bfcc897a20c0f5f41ab2f3daf2742e7cb68191037286c63b166bc37","impliedFormat":1},{"version":"bad0ad0b881099cfc1e1ee670eff334d3fd6f17ae748951dd75c261789172a8e","impliedFormat":1},{"version":"79f33d7f269c713197498beb51fe18ec1708f783e9ee36e3aced3bc01d9b327e","impliedFormat":1},{"version":"76fdf754adb2b5614936b8ec1853648145f1b28f2f2ff6216480b92446766f0a","impliedFormat":1},{"version":"a6a6683feb381fc8b72b288fce7a6b32e1a132503bbc3b51f31c6da6e6375232","impliedFormat":1},{"version":"3865a4974e908f332b03a34ebd228d770db0f7a1e311f2229d7df46c0ab2bf38","impliedFormat":1},{"version":"8ec4ba730b04d1afb04c131d25ec72efc36dd305dbb62b37f5045424a4cf9341","impliedFormat":1},{"version":"ead634afd934ba3124ff72eaeb02350bc1d8ffbcbb1b5541b297b928f10a4d7a","impliedFormat":1},{"version":"6a58f3a3ae8a7f2aa8988aebe392f8544dbd3bef96795deba908f3f1f90eebe7","impliedFormat":1},{"version":"6b5ea9f0d12bd3a0014c1970cb841d99a066e0b71d2e268c68cbc3389b2a9527","impliedFormat":1},{"version":"c3b360be6b8292032f8e567175acab932bdd53f9641a307c8dc0feb5817c5757","impliedFormat":1},{"version":"3824b6994efd620592570559e58b10c810c56de51ec9e0c918064aa15d9a0a6a","impliedFormat":1},{"version":"91992163e20474f706604a2bf8ddbbef93475a7fc81a2bd931958f2652fb1f22","impliedFormat":1},{"version":"7318fc87ac80ab31b5da74ee7968ac0e7a208fa3347615caaafa1dc17d83499d","impliedFormat":1},{"version":"b7ad4e29b4608dcb74555bbcd889c453337d5d60dbe9c318e6ecdada4d850ebb","impliedFormat":1},{"version":"5831fb954ea120dfa81a3380c9b4b3825f3a775553c5f35acf9fe9f0f8dc8ab1","impliedFormat":1},{"version":"72bb53897694a7aac8be9bd0c1b707d69549f33fe157b488488546a5901a09c1","impliedFormat":1},{"version":"a3295b4807d520e8ef955e53cedc428da9777a9bace75b61b06b194abead2f01","impliedFormat":1},{"version":"8abfe66a868fa15d1a9f5793a8789d65598901468376b7d33684a183f540da94","impliedFormat":1},{"version":"6cab2a965d8cf4c880908bc5427aa111297197e0976f4df8dc072a12998e5c5b","impliedFormat":1},{"version":"9fafc8a20a54f227f8ffc97fb526e01d7d40e7f746b97c3c4dcc786a8f583e0b","impliedFormat":1},{"version":"ac81562b12319a11a039148b94e618c626fd338ee08df2fe916711a50893108f","impliedFormat":1},{"version":"e47bc0c0462b2836daf4029ee3793b18485aab82ce6df16a1d1dae7dadc08753","impliedFormat":1},{"version":"c27521131d17fc8dcad6a2bbf32b1453d39163130d28db1e02d966f9fdcb0b3c","impliedFormat":1},{"version":"2199d6152493029dc0fea17954b65b1dfebe3f421ccbf62fbb65027964f7464d","impliedFormat":1},{"version":"bd6bb3e0808673fe31479e678feb4674750c879bcfa6247960f51098713186d4","impliedFormat":1},{"version":"7c57ef3e76406327d7a127dadd6d48a25e001fb35fc744b5032b56cb7038cbdc","impliedFormat":1},{"version":"b4d9f3f364719bd0ca9e6710a62670662745e16a7e3281298ded10b2797c9e02","impliedFormat":1},{"version":"f7ba7111e582efd9a804135bd0b156fdc56c20c57f030c30012f62317d6642af","impliedFormat":1},{"version":"557d04797b262fd5e0a6911f2bd8953e849f6e845385d351fd6c0a43ca7ed3b7","impliedFormat":1},{"version":"1d6fa4d92a8099cbbf35c8043965dd29aebd41aec0c21cf8795c2ba4791ff876","impliedFormat":1},{"version":"1b82d649be97e143b557351194072925fd8a6a3922ae25a6307faa3eb22989ec","impliedFormat":1},{"version":"3f9677a142a54645e39b2dcab82af8008723dd53d024e205435675065d3d85c5","impliedFormat":1},{"version":"da2f1ca5b91dc3d9c621b0ec31d3bbb281e1d50f4e134d433a5172cf9beef103","impliedFormat":1},{"version":"f6247942f8d5c18d46f87cc5c69efbb34940e3994023112053119aa6a4154776","impliedFormat":1},{"version":"ed78f3bd0ebc494d018df20548b8a49b114795c62de629be63c32c513dadae64","impliedFormat":1},{"version":"58da935c41436c0f1bb50506f3db51db2900f7f66c528f4080b47a9e2d5eb457","impliedFormat":1},{"version":"5a12e1a765a358c8a868110068be3168c4bb2622f2a2cc6bedc286fc4982626a","impliedFormat":1},{"version":"ebfb84bde4ad61f1f1304e84a8b68bf38ac95fdfdaa15c3f820967e3cef24e61","impliedFormat":1},{"version":"805ccf9dc6a2b41f75d676696de93aa242b86daa40eb9d4764370b266023bbba","impliedFormat":1},{"version":"e25289f80df4cb60ae6a28ddaaa67b3eb6eef0593a856d559609259e894581cf","impliedFormat":1},{"version":"60c97ba6ec78a39b5fef27c89c1d53f724e1dbceec301d6fcee4a8b5eeae9e5c","impliedFormat":1},{"version":"ae456a438db6b13b89582d2c2ffe847b1c180254dce2885b29079ad23dfbfce3","impliedFormat":1},{"version":"4125d11c36d4430e986018869ccbf63e33dbdc0ecea1b05ae9261af5f3855f22","impliedFormat":1},{"version":"95427c6bdbf8371f1ff39cf33d2f165b4912f80f6e4b92c1d44325df64ac151d","impliedFormat":1},{"version":"1e2fd8bf569648448b7aac6d2a2754aef1f4d4abdeb651a77c7169034221bf95","impliedFormat":1},{"version":"4bd93c7f2437e4efae595f9df306bd61a3994486183f48afb5f5c9dc6437a6ff","impliedFormat":1},{"version":"e5c67dab955560d09036c7bce6eba00b7db004270756b9b59825deb2f7c2dfa5","impliedFormat":1},{"version":"0c1ec9da17b07f968bc0eb7498ee8e174abbe4f8fb9dacc80b061b9299318151","impliedFormat":1},{"version":"e3643917f9c524d32fdcdd7712f0c4cb12b99a8acbe31c39170ae31c58d553ba","impliedFormat":1},{"version":"c94f3dace4b68b6c128787790059b67a4812154d0d34764092ba403de64f0093","impliedFormat":1},{"version":"726ec25efd9f3fc4147f787c5353de63b29e35f42e35f700514d9fa8a8659706","impliedFormat":1},{"version":"d422778d3b801a52a3d8cb9d72659e05ff279ab5e6d52da9f8f439ce1a7f42e0","impliedFormat":1},{"version":"024b1fefdb53328a9750e67dabe3512ef527757bd037785ee930d7fb0fad7ff8","impliedFormat":1},{"version":"1ffdd22f172945afeca614c7dc70503de50c1a026bc7d1c22a418f37fc21d4ae","impliedFormat":1},{"version":"db9d3c3f87bfeed613d822f493a3b266310da244d63e159cb101f9097bd165d6","impliedFormat":1},{"version":"06657c2de01950b2e35a8d354c1ce19cb05ed85aeadd5a13813f760c9eb5fe25","impliedFormat":1},{"version":"331461a6784e9645aa85262f3665861581e68d20af109ef5169a4599a948a0f9","impliedFormat":1},{"version":"3c600863f391974b6d469c025ed4de6300e480b7b217182f267277acf1767e12","impliedFormat":1},{"version":"a14e0a2e0461bf1329698a4365d3daf0bdf56ca55200dd17ba0ce1decafb3a1b","impliedFormat":1},{"version":"157dc9988dc42d73d0c41a3af153fdfdf97413c899ee4b7d8837f33ff9d11bb6","impliedFormat":1},{"version":"3c65ea8a95ce5a11f482abae178572a13a337216664fb481e151bfe696bbe9e9","impliedFormat":1},{"version":"a0a8eb2072f7780bc794b545e3e760685fe67575d7c3cec11ba6f134624fa8dc","impliedFormat":1},{"version":"dc04dc2984778ba15a13361aea9c915182f64a1d5bcdb5b4c430168e2b154daf","impliedFormat":1},{"version":"6eb767cbc4425973586c4442987c1a1eb607650891e4a43a9b8ac782a85ecd1d","impliedFormat":1},{"version":"da2ca744f4a1810b9a8c0ec68412cbd053459ea7334797db526cbbe18f022b89","impliedFormat":1},{"version":"1d2190bc27ec5febe38395dceed4786120cd18abcabb2621eb57d8595fec0d69","impliedFormat":1},{"version":"139aa892d9f6359b19074c70b6d2d7893f7176ff08c6694ce392dcb9ea7404fa","impliedFormat":1},{"version":"1bbed39a47c98542adf6e51e89b5fe8216744040352d5bc3e3c9280846fa575a","impliedFormat":1},{"version":"ed3bb32dbd5fad44455953437ea29330438250ff4399bc1a5550bf422d085ff7","impliedFormat":1},{"version":"7059703f16b3a21b84cef695602fcec57ae635ebcdb3db9c1d4973d7d7b49287","impliedFormat":1},{"version":"8d438b33107c05ee65b0d6cdd72ddf63e1b68e237b551ee4d5ac9c3b3a0f0bc7","impliedFormat":1},{"version":"be02b769ee841ea27c1ec02c1737e1966ff6a1204eeb9bfbcdfda9dd8879a7f4","impliedFormat":1},{"version":"cefcf164e983cfe9c73e7322ecb81149e84b2a3ab1cdc95b4d118560f1279b91","impliedFormat":1},{"version":"b889acae3f1d102dc1e19ae6bde31065becce6b4f3229c7cb5c155515cb81a7d","impliedFormat":1},{"version":"5e3cea7a1cc38b2bb8730c66f3b3da817370ea208ef3da43e6e2a8208c32679b","impliedFormat":1},{"version":"ca053bee7713e04e7f4a91acb76e04a77a0e55d3abe5020ef24d29ce0d448d53","impliedFormat":1},{"version":"f4ef7266130f87644a6f12501854cbbfa1070c42cd87faf33f87d64335abc23a","impliedFormat":1},{"version":"0e7e76615747c3248426731aed1e1452e7a2c7f063ba35eaac3e6a8faae1a32f","impliedFormat":1},{"version":"64ac7c1e4f911961a13f32954a53c5d722c57b27c390a3cb9cd05dc74026b480","impliedFormat":1},{"version":"5dd4777dd32f8b35fdc136233e6e560229e564063ced9afd5d0dc3ba3900f8a4","impliedFormat":1},{"version":"c7af068c6e67ee986d76cbad0acfa58de2a338ab9a1ca306f877973cf5461e03","impliedFormat":1},{"version":"899da2fb199e6e16fb89a92c0129e62ad7c586df1a0e5fe0fee579d391b72ae4","impliedFormat":1},{"version":"64b1028b92467d6e87b1bce90d012e5cdbbd4dd35dba54c5f53be19095b11b6e","impliedFormat":1},{"version":"d7c7c82b8a1a5e9aeefe804d2ce54891cfbd2f8798ed426cb9f44e67338b32a3","impliedFormat":1},{"version":"cf1b35415240b6dff4ca871afe1a6eac0d94cb2dad2480576ed05c9512c281fd","impliedFormat":1},{"version":"019721ef1a2e4e735bf370f32ec513ccf719f4919e3be3b41acf9c0b353c9d16","impliedFormat":1},{"version":"4726482f05007d15e6c371514a5baf33cadd38dbe285bac0234c75a9463558b8","impliedFormat":1},{"version":"a26c924fd884bcb2e55cb5fd0008ce58113d459bbc1cb924c99847767f2a8886","impliedFormat":1},{"version":"df29c17f6bf025612d273e1cc897eb4c8221578498db75da3b822cf90ea1c42b","impliedFormat":1},{"version":"a9ed29c4dde54a0c6da1a8092a3754e24de6b74040c68612081267506c7c14b3","impliedFormat":1},{"version":"cd2886147f367b00e24c700f7e6d6e7f4cdb27284f5d9463f412a172b094dc24","impliedFormat":1},{"version":"7409ef36c625d60a86ac1407e49de90da47270bda094e2b9a37d470e26c10c4e","impliedFormat":1},{"version":"eca94a391666cab6347eccf0ec1eabe75bbb012ac3c8efc72c2fc1096ad84bd0","impliedFormat":1},{"version":"ea53fd137a25bd7a0e99f1bcd2f1f190ce396d6818381145ce233c9ae68576d9","impliedFormat":1},{"version":"08c998a1bfb56141b71b737887b9b5c38216e6d6d906803d74b5d53583f39484","impliedFormat":1},{"version":"8a48faaf1f07ee4753492b8fa16e4f496c58fb1dd50414788bb1b9ce2f013d24","impliedFormat":1},{"version":"71ce6907f7e00a0c258e61bbdc49b8a47ca987e2fc0acc065955cc05897d1ca2","impliedFormat":1},{"version":"f06897e7240dd39f8c1a6b79a3943788f03d2aed9858e0c756d045705064e8dd","impliedFormat":1},{"version":"32497a0d6b963314a64a46a427a316aaa9891674e43f4b5409e81d00a0f5376b","impliedFormat":1},{"version":"3dee34e5d87f56f24506eb8cd1cc37d55f942c976c16b8b501e0ded32445c978","impliedFormat":1},{"version":"54513d592a8964918d79a1bf82c41172958ddf48fb13b8da0cbb9e7e7ee33eb9","impliedFormat":1},{"version":"eb950fee5f2afcd1ccd541ff892b150c22a36e682bbf7262701c97f7aa6a49cc","impliedFormat":1},{"version":"f78a9018e694de0e92d90f6454a545f7a778b90a75e62b9f582f3bb74653ebe1","impliedFormat":1},{"version":"a6fa2f52fafaf68395d59f0419cce9e5dfc532d3c39eb0b4ca5d2f6314e5a678","impliedFormat":1},{"version":"dbe240f7eec217e2d4a77ff6f66e29f8be67f060188c4b021908a35fa99d2d40","impliedFormat":1},{"version":"fa66f4d1d69c409c1ea535aceeab5bb0ccb1e93d92e1b2abfe9835900d828adc","impliedFormat":1},{"version":"4604dc1a62ac42e1fd3f5d0a5478359ba92958b237aa2e24403a86e7210c53ac","impliedFormat":1},{"version":"98de9ba7c7ffc2ed99fd98aae6ebea602bd1bc402c34212401c0d2d5084c600f","impliedFormat":1},{"version":"2df41b284a9220755ea0231eb9fa1f9dcf4b16a9ab55e2cf86eeea53f23dd032","impliedFormat":1},{"version":"97e95b0d014b9cc1b330d0c7d3accc30221a091b349f2b79f5958551fb57510d","impliedFormat":1},{"version":"533065f21f74aaa2582172ed02c74e8a2b7a5bfde460ebabb4cbd447ed57ad2a","impliedFormat":1},{"version":"612bad7b6de5dbec926fd384a9f1f6be35d62a64cce688fb5acd511659ffd5bd","impliedFormat":1},{"version":"4186554f153f2017e63dc5edef1aed5bcf9b2dbfe30f9d846463d8300e54fa03","impliedFormat":1},{"version":"4dd9591055cebdc6b78701de85b5f7f03896ef6bb382dd3c120c1a165e1a7e7a","impliedFormat":1},{"version":"c5d1e1c2f9b903a2aa1214ba378cc81f5d4cb9d6d4ac308ad680c4572f8a7f92","impliedFormat":1},{"version":"77bff35abb1af437503fcc6664055c947fdf967869326a2f5285587574e3442f","impliedFormat":1},{"version":"9e38f76f68ee422f536b5565eb38dc8f57b56fef6f4ea498d6c2d81d243df745","impliedFormat":1},{"version":"792f975ba01585fc7050dddded36700a9c50662835235cd44a7d6228f0944832","impliedFormat":1},{"version":"7eb762cceea1cc343e266dcabf7a2e667d8a83035ba1c2558630c4549a8123da","impliedFormat":1},{"version":"b68868886ec1fdf0ebe9ac77ffc64f551d4e6fce20ed81b43ad269482635a44d","impliedFormat":1},{"version":"e42e97e41aaa1471fd1c32f58b21bf8188654c4972d70f2510db726fa9a69c6c","impliedFormat":1},{"version":"eeb540679fcd5c9ed3f0b0f0d2701678f6ac3048f2e06ed794ea9207036a2474","impliedFormat":1},{"version":"40bac16ecdd7ab2d2ce0f91c7e0752f6765453568512f32ef0d7a77c58dc6d3f","impliedFormat":1},{"version":"6e2c902f3f98a271441605452a9bc6b9617c64421fb4b95230f15e4f819bcfa4","impliedFormat":1},{"version":"3aca4dbdc398645e18b35eb1ff3da172c75738f137efdb0a60fe49696f7fa431","impliedFormat":1},{"version":"e940dfa74c91fb34e1e90800d02c20f3a4822c16bcd76e4f661ea4973a53a42d","impliedFormat":1},{"version":"2bd525d2a414347e1a686e5a7361e11bb78d447c0ca812c88971137b971d3672","impliedFormat":1},{"version":"c1f3febd0b3b773f44d15f472fa06a1456c445762acdc72ee481415619a5b204","impliedFormat":1},{"version":"032103352468437f55d36ff748a317f519114f80a3e495b7680c75e038e5886e","impliedFormat":1},{"version":"16f875256267380bf369ef248df4dc10fd9efd4f268e8f0bee7987497dbfc841","impliedFormat":1},{"version":"d4cc1fa2579465da35331a1d00014c55117fb7676996ece38a25b80a1036069a","impliedFormat":1},{"version":"e464a0c5a7c76f9e775861de7e5b342aac7922b1c0b6e2948a5db0aa931e5010","impliedFormat":1},{"version":"c23090199551eaf0130c1ae8400726a7e4749a915cf1634e9093dea8d25d70be","impliedFormat":1},{"version":"ae5d03618590d27f170976005e97817d19ed73d620ec58d707dd650e4c1c3dca","impliedFormat":1},{"version":"f20534945b8b1e39361089d8aaa53c96a3565fc717668c4e6d4850072ffe40ca","impliedFormat":1},{"version":"2e7c6fc58ff11d5df58982645f08fd1edf66d0b987f31dfa29cb430f6b8e090f","impliedFormat":1},{"version":"1ed7804b318cc5f9d51974b8d7565beb90347a9d40a97b2a09a794083399196f","impliedFormat":1},{"version":"a1bcc8bc8f1197eea9b81045844e86dd7f44e3c370314527ee01a9172206a384","impliedFormat":1},{"version":"aa708748232db585634b2eaa61cfd23610df792d60998941c8d977e6f95f8801","impliedFormat":1},{"version":"edbe202e2470e988dbe68756d73b35bc658569d5c98bc2143dd8a531fac669ba","impliedFormat":1},{"version":"caf6506cfcf52d2a66b71503f19b70202dc13a468f9bb0faad8f900ae34f3a2c","impliedFormat":1},{"version":"59e1cb320549256f1170acbdd2b18577c9d0cbf44f0b8c1fb819df966139850c","impliedFormat":1},{"version":"68abc3d16873a318412e6bae31025c3ebb946301311d52172398c3e59d396866","impliedFormat":1},{"version":"296aa66fb1d44df18aaade53bc13bcdf3bd187a6fbaa73b5450823e572b92cb1","impliedFormat":1},{"version":"09fd7e2c1608d3181cd69e23ea833d25653f60012331a2db0c9bab6616090338","impliedFormat":1},{"version":"9648eb8f1f1e8bacddc8a852f0d3ef8e43fe13fff6c4b0d53eb54c256e2287a4","impliedFormat":1},{"version":"cad46ccd68670a60603dd7440b8f2a90e1e89f7766ee18cd95f03ad75d451253","impliedFormat":1},{"version":"4717ec6b7732c61a0e7ab383a476c9817b29328b16b34156cfc5ad0b9d7d8f58","impliedFormat":1},{"version":"9584cf628af3669890d3c67a7a0be8e7cf36454025eab012b6b3ae1c1ed41e8b","impliedFormat":1},{"version":"38e4807e6727998e7092f5b60dd0318abd6e53843eef178ec5f53dab55454fbc","impliedFormat":1},{"version":"dd07c93b6e66096fa542451304bc3f5bb2364a78829364a82b8196d6fd179d1b","impliedFormat":1},{"version":"2dd2aaaf73eef844dcc54c5c3887350e817d128c539a1011ec1b2b176c7c39aa","impliedFormat":1},{"version":"216328150a2fdae431ed0e447aa3295944af1bd199d9cecce4aa4924d162fc66","impliedFormat":1},{"version":"d57321e94af48b9e72edae49243b8d5479a8d85d43c5e21dbf7ee40e578b4a15","impliedFormat":1},{"version":"c0e43217feda3af69a9c044588b132f1a77acab87bf8ee8a0671ea57f191cf2d","impliedFormat":1},{"version":"df53ec34e7ad7e938aad672dd08d9b75c16123464bbe1f387164a3cc78d1cc51","impliedFormat":1},{"version":"2a80821012d3997c8f9a42a16f82f6aa4b2ff334d00d0b057c0aed333bd5b116","impliedFormat":1},{"version":"0425b2bc79473346386335c4a98730288e27e6b3be900a86e42f0492b74a7222","impliedFormat":1},{"version":"e334b173acb82236c822c76c2a9867364e53375cfc8d0020cf4f1302cc126645","impliedFormat":1},{"version":"404d9d0ea2927613a7e98143ec064879380eed25f66be8913d133aea11ad06d8","impliedFormat":1},{"version":"8cb66810e5568395abda507ecd8e8bd362c616b0e5b4a596b781168c67d11535","impliedFormat":1},{"version":"ad6975af7ee038b7f75af724695551c7fc9f3e28a1aa7497d78f5f6bf9e95d46","impliedFormat":1},{"version":"ab6aa891f03df0f4369739449c0d00933e3d130a6eac0d7cfb9f665f3423ea45","impliedFormat":1},{"version":"84f17aecfb88b19c6682eb3c4c5549cacc598d1272d86e383f3c5b9e651392d9","impliedFormat":1},{"version":"83b8e13fb9894aefd5815a4f8db86afeaa0e2a177636110a63e49da567d12726","impliedFormat":1},{"version":"be332510bcd8c40389283bea35e93f1f902cb9d7ef180cf2c0e927a05f085369","impliedFormat":1},{"version":"4d1a546b25a9a3ff60b1ac5e55a3878519c2dcc300dc06a04a6708d3d8e640b9","impliedFormat":1},{"version":"80a62b835bfb91609d1ccac81cf94a3276d2ed8aa9024eb99f629e9aa0f27460","impliedFormat":1},{"version":"d7c9728c0917636fe16505ad9c92348ae4235a589cdff377194bfbaec4a24519","impliedFormat":1},{"version":"8dc226f7642712a4bf68991506ab095578fdb540c2da229c3abfa67f7fe4c030","impliedFormat":1},{"version":"20a143dcd7b924cafc6ab6b5b70a94a04e318e897fab3bf0a2eb21f67f67aed7","impliedFormat":1},{"version":"112e1088892011d916ccf37834361c708e7c8e204ffcfcca78ec22ed48f517e9","impliedFormat":1},{"version":"9d173b6991690ad49bb604ab45e8fddfb8eefb1fda36e68235774c9c45ba5a77","impliedFormat":1},{"version":"a71944ec810edbda13dd7523dcf48b4bd466ef18e21343f17cd48d50b0293736","impliedFormat":1},{"version":"e4709b4e4bccf69b1f25c419310c35083e8d0f6b02acd9a0d31c1863ff4cf9e4","impliedFormat":1},{"version":"7e2b2b72bcd68ec1a659dd2652a4cccfaf3b341202270546c8b9ad9207e6a5d0","impliedFormat":1},{"version":"d152cdd7a87c3ae967813e29eabe763f47a289a38fbee947c23dec3720e87cda","impliedFormat":1},{"version":"de78fafaa11172c4de3e99f0405b938bfe1c5b2889c8beea5c0fc21dc9e277cd","impliedFormat":1},{"version":"69d98158a8e8542ca2624f554255f3d4067b3c95f8af4adc7c523f1ecc785924","impliedFormat":1},{"version":"f100519e3b2430408da9555f6b86fac193ec3b7d63d63e8be5671937e2ab27f6","impliedFormat":1},{"version":"565222b8a4fc2d084b6fee31818cea707d0d372b67a06a311db4d2a035126f34","impliedFormat":1},{"version":"70774af3c44d8599483041d731ff3c5f87fcfa422eaceba7b65e45f5ece90f4b","impliedFormat":1},{"version":"5f9866aef84db3e9cd0bab95029c5a4619625e75e813058f86b92058d23bfed9","impliedFormat":1},{"version":"b16e1eee3045f66b1c5706429a14510da2305cd358dfd8e3277ed36805ca9f01","impliedFormat":1},{"version":"7fdf1aade7cb7f843ccf273ae0fd426cd526a31323faf51f029b029c043c87f3","impliedFormat":1},{"version":"bc873fcc1bdc8fada5654939ec6d24376f376a688c252b3b0d659b71701d0475","impliedFormat":1},{"version":"5b2acd68e2450a06612cd77a2980b24e85d4ee66fed4ee65b528ad10fcd7926e","impliedFormat":1},{"version":"1449deb449c18236f39dce4ae6e8066063be14faaedb51e4a3379f201b0d521e","impliedFormat":1},{"version":"849a24a41402fc007986121747da0ded06ac5434e33396d92a2088fbecbcd66e","impliedFormat":1},{"version":"2623288f4e5208829979e3892de673760de3da9abb7884b4a346e18eec2e6418","impliedFormat":1},{"version":"bc4a8b849345970e5bcb4b2b09cb55145966c23deb96dfb958c3e212aa8f6549","impliedFormat":1},{"version":"74be6f2d91f8486556e00d99b87e535c104c0db4a7fc46989b137797c1f6032c","impliedFormat":1},{"version":"ded16fe486cad27d4f31bd99134c106fc85863a76679ed7dd7e4cff819943bb3","impliedFormat":1},{"version":"d3b5d88efc7ee78ca3d4250ec060bbfc58cf74b28b4339cf8efb672fc5d60ebe","impliedFormat":1},{"version":"45f9074ac03fd96db661e045bae3bca046d01da19b505d5b3f07401481b218eb","impliedFormat":1},{"version":"eb6db105038e5ba14923208161fa58ba2b6d108b0cdee8ce4c73e6199336d7a0","impliedFormat":1},{"version":"2c7b1d5fe345850ebfd5a6b7da9647dfa48047c238c9c8d925c04bf73e24ebce","impliedFormat":1},{"version":"7bf38bce0c3a2c1ab1e08b22d4e66ab9466d3512a4dc27138b01c299fa65cedf","impliedFormat":1},{"version":"3822fbda54201506f0daaf2e5f68c0b487dc07aa8fcba8efa0546409361e5ad9","impliedFormat":1},{"version":"4ab33959bb8e1d202521904adf074c34f0025fb1b9cd90d5dd434a2cdead98cb","impliedFormat":1},{"version":"f12093eca3c838997b2a0227def106995f3a7b6994fa240c0dc2c120b59e39df","impliedFormat":1},{"version":"dad57f2c0560b4d47f9b27bd1413192e52ebcae46d9c3eb00a587eeb5bdcdee0","impliedFormat":1},{"version":"4df8b43f9b70d121452d6d0e2fef3fefa2ac740887eaf81e59ae4e59774ffc7b","impliedFormat":1},{"version":"a9c4875e7cf70b355b32ebc8a9623017be71a5ce18cd01b21e8a5369ce1c9e42","impliedFormat":1},{"version":"906456423f006f67de143ed8a45e067fb19bf207f28ebf1cd111fd02263bc6df","impliedFormat":1},{"version":"d656ca3edc31daf0dad3bc4c932322e09cd063906d0b93ff6666cb92c09d6dca","impliedFormat":1},{"version":"f2047388ef0faa883e6a6d7c1bfe5791b57054f9576f649f92eb150d14f07e2a","impliedFormat":1},{"version":"6edf9b42533a18aeeeae10a8814f4cedae8f80223baada16df13961f35864856","impliedFormat":1},{"version":"8592fe0eea7d1f67204f0b48c793be466fdba4a90a2c3ec7b1f181753d1282f0","impliedFormat":1},{"version":"8bb13585d1dc18a5ab75f64e7f7b253e94fa96fae98d2294b351f3186737f019","impliedFormat":1},{"version":"989f246c3daeac4a4b6dc39fb8ab044448635fbeba7837a59f029cc110ad3340","impliedFormat":1},{"version":"25134baffac2b03586eda34193b974a821e567b1a5bffbc9cde61d0e16074b4f","impliedFormat":1},{"version":"a36f879f34d812bc8c5466490725b8ed9a9f9a315eee41de20a9af9b7c2a4820","impliedFormat":1},{"version":"122cfa505c7ebd278f37dc10283f4c441d150445c92a6c2bf2ba3e9176bb2f91","impliedFormat":1},{"version":"dd311b5e4f0b0d716a435526c715cbc7fc9e3b1922c1366ec6eab2da4de63a53","impliedFormat":1},{"version":"71d3a5b0a9e744b1ccfc3522954cde5dff68a6d9ffe9e142b2f9e76991d4f113","impliedFormat":1},{"version":"4db55870814528d3d643a9130589b8ea0d0addb82b2f4bd8a331fc7e5636dd8f","impliedFormat":1},{"version":"46f1bd13ffc3cff3ae73c62a892eff25b110adb2c38d9aba67feec3c90cc9469","impliedFormat":1},{"version":"b557d45c168a6a585beca8a23c5441bac7708b6a033ab6f8713116cea1118453","impliedFormat":1},{"version":"761c60844bb3c797e811a07a760fa44e61be419a1b37dfb710260cd5bc060815","impliedFormat":1},{"version":"53eb2db64558891b5dfa684e391e1c90ec47301a9f1505cf428f16f6cf7de5cf","impliedFormat":1},{"version":"82c68416e0136334a66275a03d68adfd7835c94cff3a33826b2bd20498db6ab7","impliedFormat":1},{"version":"6f879a6a80158e94c3d96259396c06beeb87fe863915dbb03e632872c664263f","impliedFormat":1},{"version":"85c296c62669d3ee0345e54a238426b30cdeed628fabc2564172188adbe55c34","impliedFormat":1},{"version":"2a36ce8ae2e791468c660c444ebdd93f2c22c5e1b214733217149b989d2d6eb9","impliedFormat":1},{"version":"de6a9d567a8ea5ca9ceb705c49abb5d5dff446332e3c77f5bd585e6bd4df8970","impliedFormat":1},{"version":"21e456a1960552dc0c81e102fb476854d09875e17a37b0fe52218683c9f4b3cf","impliedFormat":1},{"version":"c75691507d19b3b3892a809f0e51af04b8423f81963786b1e4ddd54b178c1f35","impliedFormat":1},{"version":"b9969f5526b4d78fb5f0001f8a2a654d61c7ce601535013e5416c4300bd9db7a","impliedFormat":1},{"version":"33803b2ad29051bd8c4d3ab9b1cd1d5a2603b1b48f03f21e62602c7a610a7ea3","impliedFormat":1},{"version":"624c1c00baca2051458efa764fb4537039b3ce88f1c79db7dd27033041b93e8b","impliedFormat":1},{"version":"b74233955e8b40025dddb55303ac2cb5189f22685b4f30dbf21ce1b1a5656ded","impliedFormat":1},{"version":"dba0925c06074e9ea94c9fce95887dd8bae26262e1522d52b769e59e90a311e5","impliedFormat":1},{"version":"d85050c0cfd804e9b849d9974741773e8577b6ae5636dad052f07b156972114d","impliedFormat":1},{"version":"99d81b8b1866cdfb815b440fb151fa1689b7c5f534d36aebe8065d902b3f9f70","impliedFormat":1},{"version":"8a589a19d5354a1cc02a28ab5084ac04ceeb00658a95ec089144b3ddb226e259","impliedFormat":1},{"version":"9bab286af000d35a72ce6e8acb45aa31fc89b8792ef937ca51566a996fc827f4","impliedFormat":1},{"version":"54d5e3ce769fe2154d98368f2f723c8003f04aab9c2067312fc515d8a20580e5","impliedFormat":1},{"version":"a37d93b145fb91cc40f6460a785bab157da27286ba72d0b50ef8119a5b16b85c","impliedFormat":1},{"version":"59ab362eaa15d882855ae715d8eb0cc757b7d5770ad97db5db4cd1319d760c2a","impliedFormat":1},{"version":"380220f859692be28024967a3df11402112f41cb9b029f02b7a4ac6e6176b037","impliedFormat":1},{"version":"ae482fa3b788e4923764c1ff3aad5d18157e0ffeb8cb0b012482721a0f613af5","impliedFormat":1},{"version":"11211b24dca9ffde7b9c863450b189b47e33f040f8264114b7761d07bee4c57a","impliedFormat":1},{"version":"96009cf527ddfe50e728ff6339f2aa58b687b678bee6a3848f6f7d56d8d34987","impliedFormat":1},{"version":"0e0649c37dc6e85592933fe6f5ad004ce1853e8d21bb6ac0e3f4f9c924e6ff2e","impliedFormat":1},{"version":"68d8782bc87568b260473a027c7c6e5417e1498a5f0a38037da23e2552474546","impliedFormat":1},{"version":"b6c0494560264d7dbbf8a495e37c709cf59cebdf7c5c03590e69eb743503bc89","impliedFormat":1},{"version":"1ec36a3efdcd2023825f83cc7e73756fa6572e62019375b727cbb393a7095e83","impliedFormat":1},{"version":"0a348c23db159a9d44a7ca1c310a59e05bb2055fe5f3a0283201fb5535f97deb","impliedFormat":1},{"version":"9003ad7177e5658410de57fd2ec0660da9a53250c30e8271f78b1e89a76b5d8a","impliedFormat":1},{"version":"dd5680f0e0a9865a764ba3c55face588247f30d5161a37a4e3b379323c4b7320","impliedFormat":1},{"version":"d1b784232e27adc314aa27fa529e56176bc0698dd17d6050e2c5fbf4d3e84b21","impliedFormat":1},{"version":"e618d6b5e3730c3456a83994fb9a5200ba6bbae3fd0d9eab87d06e18466fec11","impliedFormat":1},{"version":"1825c30debb44b2a193dfb2e2028599d222cef3160abfce320e9ebc6330b556a","impliedFormat":1},{"version":"a5a1623a24188a7a18679e3f89f2768fdc10c4fe17bb79598582c7196c454971","impliedFormat":1},{"version":"5e22d0ebd4e1c30152675aa52dece859cc8c9acecd7959c4b2b10ce8d051598a","impliedFormat":1},{"version":"8ab7225e3f5cba6478f7a4313d419b6579b631f5cea92f7ff0ee893513206216","impliedFormat":1},{"version":"f10acd2fd2f128bcf3f0828feead49d6155fc25300816af1471cd1f980137914","impliedFormat":1},{"version":"729569c50801e4ba94051cfb82b64c37f4598795270e434f0ee339c6248ae24e","impliedFormat":1},{"version":"3d2ba9e7ce503152d42c4478ac6bb424d78b42874259512d8472ec7674a72ed2","impliedFormat":1},{"version":"15b942189aaaa800907422b19c53d062eaf6f33bcf6d3a407e3a89054bab2509","impliedFormat":1},{"version":"a11fa6fe4e386a644807c6bab068dc731c1eedcac07584844d5b0885758dd00e","impliedFormat":1},{"version":"bee4497d4cdbfd126b8673f7408000577af750758e4670cd4acabb08b089065e","signature":"ac18a1194fcfe2b98c44022c765e8f07b5d7e0409722e500cb7a2357f2330849"},{"version":"9e779892a07d92b81e1c8ae2364309560897396d88e52458bf3f31ecde882cd6","signature":"63c02e4afdb95c14b1bf2a11c1ca940a400c6026e4abe96a3e21943e526f6df4"},"d8c420f8b24dd220da6b5a8be77a266078eafe053eec3d0ccdd94526197bace7",{"version":"9b251e3eb2569890cb4efcc3625007b3d337c8b074563fb54973c23b54515d1c","signature":"441d7b3d140367cf9794f6152a56bc6fcacc159837255b54d00e5c0a30f782b1"},{"version":"2db66d42c5b3d054a2b25905c9f3f28a30d1a033fe592014baa20314820746b6","signature":"4fc33152d5cac23957b357f13292a553f03a46b9072552d4c6a026f20d8dee9d"},{"version":"3796725f5d2494f42fb89182dd8ee1d61fdc7444064775b099db521e727df05b","signature":"802cf49eba74eb89c3778f1652756e7177c7bcc95c7f4f544efee87b27c58f65"},{"version":"88409505b33d8c6f27d600e89696931c95cb4e295296fa1bdeb6193892b9b6c3","signature":"5d01b4055cc4e30aca3e1f8408609e12881f22b927103d861e6b04c6bc90e353"},{"version":"4f8646b206a1b6da553a9ee1f78d35e8b388007d880194c833befa440d3d6b2a","signature":"e75cbe3d13ea8003b7e188ddd8464d431d852e8246ad0ace34151f8ea45c7972"},{"version":"cebac50dfae3cd30a9f6f8331444f1ddefdb6e3900f5d02a4a78d0ad67832824","signature":"5ed2b0f72fa76bbf34beb345c43878bee193162e3e7090fd933f3e912e8253c3"},{"version":"2ee167ee733762cd51816c467bc4be5d0de8c448e33ab5d5b4ab3cbc9aee1037","signature":"474e6ea988596f0eb5c403be08a28738ce030caa6ebac66e4f013af30c1d9862"},{"version":"6bdb5cf33bc38438f85233731ad9d95abc3efd0055148b77ae4d0005f8e33cac","signature":"93ddd7611901a13eae58fcf1ec1fb3b3503a1325576109d6000e0adcf5b548fb"},{"version":"724853d2703114e5ac7479565790fe926f8e40d75b25e3cce54a8e0abc794973","signature":"d8e74d15aca02d40482b5600ae8664aa35d71fe92af7dd18c17eb454e5e60592"},{"version":"68ea64fc3b5d32dadeaa0cd4c34d5ad76c0e2b2c75ff794b872ba7ed64aa2b36","signature":"3878279a78a4df7a018d781dfbfeb4a9e6dd51bb3267d52abe05fc65aec5ad19"},{"version":"d7fa8c3924c2b23b7451adabd3c00ffe5f4f65af45abfbb819c58bdb0a6f26c8","signature":"77741906a04b1db13436e1fd5be6cb1e6fb862fae40994d4f68ac747557eacf5"},{"version":"bc5f88d91022a8f29a1fdb964c761eadae7de860848ba3e159aa336dae3fd62f","signature":"b0a844f50b549a626630dc8d318bfbe6e4eb22db570ff19a952c56bb782c6370"},{"version":"9dda5b12b38066a0c02f45139327a943f0c1214f0a3cf4d05d9905c3838e9a89","signature":"82b3dc78365edb9b0748867b06c1b4dc8198b8839749e8a24863cc16151fdcd6"},{"version":"b3b116f0db0b6779bc5e1a61021a6ae55c568b9c420480dd4a8efaace9a98311","signature":"e6337eb3667b0d17183eaf55bc905e4da4d24e7422146859403a6643b2d041f4"},"fd149b7280ead494b912469a143ab5da50ab448fd697d2fafde21efda936a62b",{"version":"d9606acf6a26aede5f2781b61ee83b753b11edd1f4a9ec4799bedc5a81ed45b7","signature":"d62cb79c69d64312d1893d5cc74e5baa836666b52e10df3b6e5789ca2b56ed64"},{"version":"dd6b64421e787d06733734caa461d0286a179a6fb348b80be1f4c83338b99b65","signature":"f7d0a2355c43765d18780e2e6e0a959df5275f5bfceafecb5041e297cc508f66"},{"version":"6c4fa80ae9373171d84e1b4817280ab0882a22cee343342663f2ba2ea899456b","signature":"c752cff5cfcdfd0b647713524250a7b08c8eadb65a0e46aad43ada7b0878675a"},{"version":"791bcd8ace43d85552a0bc288e15ddccc2b297244bfd4292526a96c65b185bef","signature":"68f7454d61ef7c6b4d69e368c0573509b844d76852565a6eb80ace64d6bf10ab"},{"version":"10ba7e72eb2f68b57b16b46793a55c0baf1c173c0262eab79370193159269c8a","signature":"2bc4488ec029a8785e1af861de03e0d854b9b4fa5270fae7d6438c0b106969be"},{"version":"eaa624bc78a86b3aad4ce8fa1428996e27bd0d4b32fdfb220b60d8770074fcb2","signature":"12f0dbaf7b8471904e8913a66e0cd6656fac1367ef472e5fdd56e67654003864"},{"version":"6a20bb6fe1f259181c957db37d5b63b4a615abc235a2cde54ef9d7708cf9ab5f","signature":"893aeb7ccea7a5d70cbf9cdda8413b607b7fbee7ee494dd61bdd7706f3bf5862"},{"version":"f4a57de72d6f9961610aefefec02bc1c0a285a255ffaf37c832bf09b7a867e28","signature":"b77eaae24c21438619389fee4cf142805f9b728f59bde6141d3b03fbdc0aabf2"},{"version":"be647fc2bca8c2ba204e4db701e40272bd002f14c5297285ae5d20dc93a21cac","signature":"b916ddf48febd45157a6c70cb28fac56d5cfc9f5590aefc4afa0b31dad7ae362"},{"version":"61cf841830f0d74cc12e8e74adf4b326a5f16bef8cb00bcc0599f0a52ff4b044","signature":"b0ad3e9e778ee90dbb48c8e4f8af9ceb8f9c22bb7d13d15dafea1f79de5a4807"},{"version":"fc1ff6e133ad73d8e52624de2fc7b57708a1251efcbee7a69c9c4472d58c555e","signature":"b43d0e19d53c3bdcd65727e4f6bd9e3822d6eb7eafb64e74d5f61ba62aba8099"},{"version":"872e5498b30e8194d2c575275b6c1671e43ec5bdc2e8553c6bbe42e9ffacd7ab","signature":"c61fe9747b6b3a21e6c8980bbdde9df1314d00c99bb1979d6370ff28cd8643ca"},{"version":"e82f2bddd3577deed0e328f33d604f2e297fbdf3e7f19f1476652349febb6af9","signature":"a18a6900e9c7b225ea24df0955565a40b6fb1b17a1214d0cc63f7db478a3cdfa"},{"version":"2656a4e31055163539061faa7fe2c5ed914b091e3ab6c785cbd65ab69729aaa8","signature":"b0352dae0d4b8ccfb8779786491d35a329fc57ec212cd5dd6e4028305089de1f"},{"version":"4bec8dea15c7b3dc21a92fa11fac87463150dc9685fe4913fcb1bbfba517db47","signature":"f14a8825d24a3b9dc968e29c6ba4e1bd16a0438d4b57b34d8a34bf8b48869895"},{"version":"d2c6ba26ddc3d2c5c1ef8323f10822d19b8a52eed2d70e3774d29ad188f68bf3","signature":"d47bd2d61792498214f0f81b56de644d0e3779cd9d42cca9dfeba10c9604f373"},{"version":"9255e5c1ca81335e4a3fec55caadf71022ed1484b7a724dd0e01bd0c5d354cbd","signature":"3e513f00e73e718f03f8a2d437c3a8e13b33b674086c8dad7d00a3dc71aeea81"},{"version":"559f7391a02f31323fdb76e19dc7c8ad8e46a15f9b728b5d687d2256c6c77bea","signature":"47a9eba5a1f0f4a499c26c08a15228184e5773319b650448ded3c139cd10d846"},{"version":"e570ba04f706d7c0390ec02c9520069a1570db031a6606142d33d9a24856b214","signature":"bc8a0ce8c77147db6a4f75bec1e1356ac5d13db3a8e4917c93319e1703fbd99e"},{"version":"ff2501a2fbf4cce9fb7c4bf8ed5a709d60f678e62bdfda44c26f956b6c0c91f9","affectsGlobalScope":true,"impliedFormat":1},{"version":"398d3b753d3741fcf0b069a6b8a1315fe527e328e84160119d81d8425f798c33","impliedFormat":1},{"version":"a02124c0ee850443005ca9a4800b743c1afed28f9752afaf8c95cac6baf83877","impliedFormat":1},{"version":"4db8335cb921acd3448288c991f721ad7f4939069946819221b8b3cbfe9b6c5c","impliedFormat":1},{"version":"554acf414793d442a5602055e4f8d0b83edbd0e4a977bd4a90fdcf7b0b1fd898","impliedFormat":1},{"version":"52e2d806ccaab9371209f3fe262252e44cb4f79592e21500559bc6261c534d1e","impliedFormat":1},{"version":"b123d189617fe233217ddb7c8c0fd06b4581fdef69e77d63728d3006499f33eb","impliedFormat":1},{"version":"3cd4c2711fd3496674d4267305e67b69a5a42caaeca1b45c294a1de5ea3897c1","impliedFormat":1},{"version":"117c9403487420a1fa5936658fee58de9d77c07b2c7510aba24e9dbf92e9b622","impliedFormat":1},{"version":"3c201db56028e893e4de5bd9d048bb804daabcf820be6bf96bb5905a0ffa1024","impliedFormat":1},{"version":"a29d69c75a5a7b1d451b30dae820b745eb7efb5cb74abbe546451b1185c8b339","impliedFormat":1},{"version":"f2bac29fb3514f46c0c1ea981340c674098aa74c5fffe1e7630d31c177686450","impliedFormat":1},{"version":"b5499e8d3e39a1523d4d12718f77f1e2dcfa3f825f67898fcb90a9edb680e43e","impliedFormat":1},{"version":"96fc3dae2f110377fb32c48acf3efcebffd12df01c798466287183ade087719f","impliedFormat":1},{"version":"a64e1342f2c32cfa1b9da3f5282b2185dbdd0989e30b48ad07962e7771244a18","impliedFormat":1},{"version":"cefc795bc727964f7ec2991703fafe975a0b265ef4938d342f4dbd93ed7a5f86","impliedFormat":1},{"version":"0d390748eee58a959b560b93da0d6261a1d3ff87a246f459d8b3e7a20391b62c","impliedFormat":1},{"version":"fb652d576e7c73b08eb6f9a4f322aa841c1b857195745e6ca436724c179de2fb","impliedFormat":1},{"version":"d192c4305f2add7ebbe22e9328f788b341fcb66e5ce4bd23cd2b1de097fe890f","impliedFormat":1},{"version":"925c28c5e11d57a08d05059a147f7a91c0e447ec27726dc8b79655fa1ff05301","impliedFormat":1},{"version":"8c4242fbbba473b36879fb5c23f29482335ab05e4150f06c22edae4e44c894dd","impliedFormat":1},{"version":"2681cb7b5767f12041662d1db8c0994012ef98c8b41d26d2388497adfcb0cc64","impliedFormat":1},{"version":"4ac4739a6edf9fbd20a18b5b675b08291fc860dbf89784fbd85f6f303df8047c","impliedFormat":1},{"version":"1840ac8a2b18e0982da57d80a09f5e5ec0e38d18afea7ac4ce069d9bcb3b3cb6","impliedFormat":1},{"version":"681c823b35bcc451c501382a6ebecf0b09fc792d83efa1279a005aa62285ff7b","impliedFormat":1},{"version":"cff0422eb92de48515743c3638bed6c73cd7d1312513df94030dc4c41090457b","impliedFormat":1},{"version":"d478539c608c8ec78b2d0a7817c29efab421e29d80b641ccaa074a96fb577f04","impliedFormat":1},{"version":"f97e2644e1e7763c6332e1067695ab3a2e51c06baab3985507da46a6e1200958","impliedFormat":1},{"version":"e3c8c01adb8d63c65f122778d8f63911437024ec3f4733622c510273ce3b8823","impliedFormat":1},{"version":"a12603dea0828662dc971e86e1169ec7b243a606e460a04ba1e01051c4f52f36","impliedFormat":1},{"version":"b86d0df4f4c8abcf28e629ace836c0f6423ea1509853178f56c6329b2a26ccfe","impliedFormat":1},{"version":"0e62d4ab3949b67c679fd23b39e55ed9f19597c0afb21d8ceeaacc4716ed20a9","impliedFormat":1},{"version":"04771a6db3f7b054afac1bb6d540d18efdbba7439415d4bbb759b8f39f1f5377","impliedFormat":1},{"version":"d0cebbf45fa0f4b492284e0be4b3cbd1610f05e33ed201ba8937b1c147bc974d","impliedFormat":1},{"version":"6a1b55618aef82ea35596613159dd7cd7805b07dbfcdc8fa288e41745f3ec98c","impliedFormat":1},{"version":"572fa17bfde079d0d5159c47702addc4f2e0060f8abb0437a5ce9d451473f53b","impliedFormat":1},{"version":"9c2971938ec0bb237bc330aeb026d82d1e7ed0da7391c8761263e717875f2b21","impliedFormat":1},{"version":"8db1b5e284bdd0df8797b1f70406cc7dd126587fca77be01e711910cd04103fa","impliedFormat":1},{"version":"31549213d7a9f3cf3aa96845b5860144e3900997771713c689d60276b4786664","impliedFormat":1},{"version":"822a8277cc73b8d96ce336ff56a1072c9f66485a64a562cc0f29cd7e550a87fa","impliedFormat":1},{"version":"a097e76e2b3a5a7ab5db2db9a5787dc4a3bccbc65228951c243fc0d58675467c","impliedFormat":1},{"version":"e996cc50e5bae651f0565e8499873d38145d8955e521e758426ba73758eb3bf5","impliedFormat":1},{"version":"8ad61067b3ba801965c04c2815c231847631a61c4da2b1987500b5aca6db161c","impliedFormat":1},{"version":"aadd40c020be82d01ba79caf35e1169bd3cd53bb6b999a4ddc5f00c9db847a46","impliedFormat":1},{"version":"f16df5990c987807a817d3d4218335095cf2783a1a7521e2871e64b8d0f6648e","impliedFormat":1},{"version":"81320fc91eea90e06f8781d5f6bd0d3990e0cc7a50e766a78b56e0a1cd44a332","impliedFormat":1},{"version":"224f89650a8724c67f36b98b5e5325d4a224cadfb9b387bf076adb76437443c7","impliedFormat":1},{"version":"36338d4f4ac9768967f2cdc092734373a3d0eb70b808def5222765825dcde534","impliedFormat":1},{"version":"3bc987bed2ad368f08e5e0b05148783a46621d3386c4a123af145d12082a3850","impliedFormat":1},{"version":"27ae73c6c31329e8268cfadf194e127a92b5c6ada1164693e784622576b94f81","impliedFormat":1},{"version":"0e5a227256596eb516def2d3ab823c2321cef34c28cacbb559c924b2374143e7","impliedFormat":1},{"version":"718d456c2624bdff0b7683ed67041995519f657b98f52b7890f11cdccac36f89","impliedFormat":1},{"version":"4b2e887e533849e74020b1c594604e990dd8fb3abf693b1d82c96d5079b27ea8","impliedFormat":1},{"version":"2f4f0059c74e8ecf9a5e962c6a8fc3aa258941dfc18343f50e2efc2923ea5c56","impliedFormat":1},{"version":"92e0c20c54604feb984ddc519b56460c61dd9b285fbc30174839286545ddf848","impliedFormat":1},{"version":"54a336776a1161336928376c78fcc9deda2b5890f9008631c7aea700b6727bb5","impliedFormat":1},{"version":"14d18076cf79b3c6ff515123a71836644f50c2956312a2ffc960028111489316","impliedFormat":1},{"version":"632e5af6af4bc7c3977dd4782ad03b37c0229806de4eec9666fd79841b6a68c0","impliedFormat":1},{"version":"8c3e1c25eff5752f6642204351420c99844c1b2a73aa0dd5f81b315cf38b32b0","impliedFormat":1},{"version":"2e51565212c8cd03202a9492d57e93c431041114762dedf69ac3be0f62e7fb20","impliedFormat":1},{"version":"06f894fea5d5bb81048440482e750f7cbd4932cabb95e4d485cb0b9be1d3eeaa","impliedFormat":1},{"version":"b617019b6a719ce7a920e1909f3e62be8ac6a914746667bcfe10d8f350cc7089","impliedFormat":1},{"version":"cecf293195c298e093742c82e5995cbde08af76d41f9440224de7f83e077c4aa","impliedFormat":1},{"version":"575ce340b9a859c0589a04305dd9846184710eaee71a4effd071d5d406377ffd","impliedFormat":1},{"version":"8c08e433b7d3a998a2cadff9d54e3da445c26a13815f0c2fc8e7f3fdc7cbd1db","impliedFormat":1},{"version":"ed872db0e2a3622d6d92d9b110b7165d8cf23d44b6993783328e0088fdc6a33d","impliedFormat":1},{"version":"e34adafe9efbbe6d7af7e346ca7df8bb2e77a3a851d8207ae6199357b903b192","impliedFormat":1},{"version":"42b2b30e00d58b57a4204ba1118422e1bf83634816f950869e7454472fab4e35","impliedFormat":1},{"version":"958fc2e0308e04a48b1f3a793d66aaec672278fc1ae0f31efb89febb84dac1a9","impliedFormat":1},{"version":"4e771fb2e12b05ef96d1a215adfd119643c057ad3e97739f85d1d7533a18caf7","impliedFormat":1},{"version":"02ffcc56317b8d9ee19f209b7cd8e037074ab508a1ad06754a2b1f2e77911f66","impliedFormat":1},{"version":"ab570c33c53acbc83ad2e24433a433fccf12c28389271cf3f5c44b871f547b2b","impliedFormat":1},{"version":"8d6c8608ae5cde40483ab309156f5db32bbb2ee86486f7cf0f1d85733fd39f53","impliedFormat":1},{"version":"f4529b8473a9022e02fc7a4b5f92e93659d1874809f2c7b38fc367441a93a339","impliedFormat":1},{"version":"b92c58600fd18c32ff687b783eebfd0796cd995e5965a86ca17275b523d1fabb","impliedFormat":1},{"version":"ac46a79d9cfb4df1f024d98c886e4a47ea9821a2a467e4cc9623d96b8f753766","impliedFormat":1},{"version":"ff8b5fb2bca480984f53b064e53a0f3ebf4400c0a1a4b53cede4f77de95002ed","impliedFormat":1},{"version":"ab1a99b4017155d8040b5456cba7bfef33bb767da1eb8e4ca369d5185810f349","impliedFormat":1},{"version":"0c4286c0a84c20e05c1f5cda5add26956e696aa817b33d45ee2458eecb3ec375","impliedFormat":1},{"version":"eb155438a82c3e7228cfda102f1d6e1ab4652aa83cb8ca01d8afeeb782803f1f","impliedFormat":1},{"version":"1f0012e2fac75a6ef2406eba7a9ca9ea16c553947583d663eb726c97a26880c3","impliedFormat":1},{"version":"54ec65aad2d7775fab779d01763bf55d7e768920d68f7a05946901eae49ebbfb","impliedFormat":1},{"version":"ae1099212ffebd47c3f0e51162fb0c1e5d4b104421b8a66edddbdf920899334d","impliedFormat":1},{"version":"9cbe0b736b34de9fcf54ba1db60133cfcffd413bc87ad008384ec6442d4ccc14","impliedFormat":1},{"version":"3f713c2dd9b26d5e3e475c811a8d7ce219f1346cbe46dad4596dc6e1d8d35cf7","impliedFormat":1},{"version":"d538fbbf8fd0e073bb11279bff9a37deddbd192513362737f98cce00f2fa3c34","impliedFormat":1},{"version":"a7d869e34e5b3201695b1fd231884d163cf41d24975e1e6a407eedc31d7b9efa","impliedFormat":1},{"version":"d5b6042c1806e7f8ef08b9be9cb72ee50cb7b991a28efbda30a61434b1610216","impliedFormat":1},{"version":"8d30f52bf78ba0b0435286cfa393e2f62077d64fb9536eefa9cddd62c1252884","impliedFormat":1},{"version":"431e627e173edc76477913c8fe99dbbb96f8fde04aab57f133553f359d4eb20e","impliedFormat":1},{"version":"2d37f551e55c0ecdc324b7af4d2ce823026a3109f64d8c4f5d9fb050a01460da","impliedFormat":1},{"version":"dc6e89155b11ff755ade6d9eb497d247f1f7beaae87f64332e2c1e5ef05a3cec","impliedFormat":1},{"version":"4e2fb39d94dac2785eeac18a5d53df1f73bdbad18900fa22e46bef5cb3ad789e","impliedFormat":1},{"version":"734e259be474d6c6c2b63e161974d600158d3defe08e78f378301fff846413d3","impliedFormat":1},{"version":"ef71f578ad24aa892b5f52e9e5aca43fa56434ec07ce5d62423a6499c15708f7","impliedFormat":1},{"version":"176d770c6577804c34df935fa0d0fc3f60396ab125fbf20d95698e35c08bf077","impliedFormat":1},{"version":"314c4b1b0b4977f9f55a5854a6c6effdeba1342edbbb89e7492e550cc38ce4cb","impliedFormat":1},{"version":"b03941a397cbaac2a62ea7094ca56acd26e90d73865d750788a439c68c75dfdd","impliedFormat":1},{"version":"67d0d710465d9f4e26c3e55865d110596b95e84f7598164ad3046345f422931e","impliedFormat":1},{"version":"34e8ade0345445320e23a677a1011f78efae36e8653446fda313b38957865dfd","impliedFormat":1},{"version":"abb451358f4487f36cd7fe07120948ee6b700f0e7f5ad07591bb63c2b5c79b71","impliedFormat":1},{"version":"a02935c93c12620b4602f0adc823f20e6e1846d3d7a7a8bebfe303794c9671e1","impliedFormat":1},{"version":"465d478cecb928a2cd1b3ba3a07d1a2837cf3779375d5fa42fb46e9dc1fa4a76","impliedFormat":1},{"version":"42843220b0536af6997a420dcf33a0be817b4df50eebd20d1fe154cb6e8b7cfb","impliedFormat":1},{"version":"a0df4b1e4af6007211dbd710098e3ab753b119886c94ef877730644c66c166d7","impliedFormat":1},{"version":"91e683674172f9b5318021598e99f907dade21763a157a204faa7590771fe353","impliedFormat":1},{"version":"d45edcb4ac7d734f90d9624f33b01c54106bdc39c659ea6f36a0f7e935324f90","impliedFormat":1},{"version":"a7187425d04412674ff5f311012c11b459860cefcffa463b286068edd43c0776","impliedFormat":1},{"version":"b344bed7a9efe077679682e1ae59030c40b377cdb8d9995b024ada7c6ed61eb9","impliedFormat":1},{"version":"940b599dd43946b484a70368fdfd5fcf240e5732cfac5e08564b879f96470eae","impliedFormat":1},{"version":"55da79a1f289545c2d4d5fdc56f797a18074e62c252b8472d2680f592a159310","impliedFormat":1},{"version":"4123ddc1f4f7b4652c4da7aff4807605ad15279f100341ef4f5ea10ce8fd58cd","impliedFormat":1},{"version":"e3dbafa3bd33f2155464db1ff01bbc7a54635625d3266ab3632a9db13366faf3","impliedFormat":1},{"version":"71b2749ddc97ea329d22f141bd0d2231dfbcfd0c77b673fc54f22a2560d570d0","impliedFormat":1},{"version":"d8bdadb41912e8bd858a06c7550282243d2a4b8c7e9b827ef06da11068914218","impliedFormat":1},{"version":"7047200b3de005a6e8da3175f01387c228830644aec97c07edda000308991eeb","impliedFormat":1},{"version":"d99bece612c3ce15c8888c25df7e0444a338049cdb815d3931fa85891cad7027","impliedFormat":1},{"version":"a8e3e304d5584080c99e081ccce819d05ee369062c2f0e1c174560ed2cc2c034","impliedFormat":1},{"version":"09bef2dc3067c333c109f7b7b79bde43b843335b32d8ae51e712778941f89d68","impliedFormat":1},{"version":"aaf6132ab602e07ef329ae6aea0f4b976988e7d88432d4309f2a9fd0e6d9f895","impliedFormat":1},{"version":"dea409b891c81c9cc218cbabf4964c43c6cdf247aded796b3a30fa3f0f14593c","impliedFormat":1},{"version":"82aa855a9e602dc5203cf517582f06f2f1bc4e4a43300433d14dc2e909408880","impliedFormat":1},{"version":"92fa6c066987a4cea71a0ffe9fbfb683b45b5300ae9f5584b02592f3a27b3ed0","impliedFormat":1},{"version":"a5c018512673b7e1ff6cae34d14713e89e94479fff33c14696f7e2153e4f4755","impliedFormat":1},{"version":"e459c1d4e7623343476da01e7e4edf8290bca1f1315287559137af5557f3ba39","impliedFormat":1},{"version":"5981c27079aeb53fb96829328f014ae7a5a690cec8b1c93815bc23e6fe7189e7","impliedFormat":1},{"version":"2b69fbd1f361e82dfe9bbb786133f0b58845c79d7094fa5790306e5ec271e5bd","impliedFormat":1},{"version":"c10c88f1daf9fda0323c9205ee7a0fd63ae4f67320d3b673468242d89061a459","impliedFormat":1},{"version":"a68ae02c58a9b6ffb29eec100c886ce8eb80201e454fcae79c299bc2db0b37d0","impliedFormat":1},{"version":"d764056449904a73c1f2c6f8c2ae79edb0d1cc799eda5fc3a60a30fa97b94749","impliedFormat":1},{"version":"7e73db72fa480a32afd616f2ab23edb4702316c7b898bd2ba6b5eff6e8ab9412","impliedFormat":1},{"version":"916e84931e102ae5091d09c1ac5aeb2cbf5458f11e0057b23157f5c062254999","impliedFormat":1},{"version":"226d624e4776b837abb8c1eb775f27fc265d7ab4c7473bb48f39c535cac94285","impliedFormat":1},{"version":"4173e4d951eb16efa7943068fcb21aea81bdf4c996dd047ee78625874836dad7","impliedFormat":1},{"version":"9c219a351e0e80e556380fb3372a3fd2c54fa3f1bd9574710ab4e577ea26063a","impliedFormat":1},{"version":"ac18a2d24df81dbbb885e1601fe94fb9a7ba42f04c98df04d16e69f4ca9ee9db","impliedFormat":1},{"version":"8a9b3c96ea397dc289581c1aa4f045cdd2f8a55fc5d917c56d40370a83eedc5f","impliedFormat":1},{"version":"5b289d52c1414fc6737fc451b85fca5f70ead22c2294f5a9484ec1ffbe233a83","impliedFormat":1},{"version":"30af3be0483da0faf989c428587c526597b80c1e368d85281a3fbc95e360987e","impliedFormat":1},{"version":"95f50c554cc9bd129a38fc61cabf1b0b56851c3b4db34177c909c2a03d3e2b9f","impliedFormat":1},{"version":"a2b35bc1378fbc1443e1678fb3ab6e8023269500146537b5a098e8db214327e2","impliedFormat":1},{"version":"4428a7e681f6ee541f90b54d6422325fe26263c333b27293203ec8cf98da4dae","impliedFormat":1},{"version":"cf25b77288f29a84be0a436ea2f5b8cc00bc06b6e142ff975f60a2252a6fc18c","impliedFormat":1},{"version":"9fbd375bb1f6ca5490ddc37165bf761f2fe89d93bd0de57e5bf3dd12cf94baf4","impliedFormat":1},{"version":"fc291372c7992060d4222381491902295756466f44fbc6f0889a6d4e28d0b937","impliedFormat":1},{"version":"6ca9bc3ae7c4fabade7fbf2659731cecce54a745d286d69755fa2496c545456b","impliedFormat":1},{"version":"e6a2e3b9fe781537a0f91c301de9961bebdd4b33187f98a582d694177acda462","impliedFormat":1},{"version":"27316752e6998304fd7db2d67ba0b26d8029b6f503b3ea89efe845c39fabf9f9","impliedFormat":1},{"version":"4add6412e18d83b5bd7c65dd07c3a1544bf6b31baa22473775ce967d685aca27","impliedFormat":1},{"version":"8a7d6fe5fbb7e37ebb0bb81241d59c4a806cbda97a5f1f15af3fb9c903672598","impliedFormat":1},{"version":"74efaf42f5842c4c4fb398d2f74d59a86792320c53df9f63dbe00ba8b24b88bc","impliedFormat":1},{"version":"08603c7d3cc9cecd1ac97cc1baef2d90b116759b541eb4181109bdabc64788a9","impliedFormat":1},{"version":"64068fb5c2c88a2b7016d34b02b03582e759b3f0ffb89e9e07f968838275a564","impliedFormat":1},{"version":"1825619ec278edd94785af65ae589289792cc6db662f63adfddf2a79f6bd4233","impliedFormat":1},{"version":"3c63b1b5ea4b20822405d6b7a8ba96429ee2b400367b61c82507f80be6202289","impliedFormat":1},{"version":"cbd2ae79794fd7bcff66e343af71f7e174a6eb95f17a8742b1ff62c8482f7cf7","impliedFormat":1},{"version":"e36a635d498defc980ef7b96eb5cfaddc11eeda2bd0de7ae249adde06b2d168f","impliedFormat":1},{"version":"5ebe388148630bc63525fbf1f408370c657ae888abe179001e64711b0dc8d663","impliedFormat":1},{"version":"cda3e014cbfab2a60d7e598a50f90dd30c7fcfeaa73a60dd916507bd9965260e","impliedFormat":1},{"version":"432ba4ec869745ed9de5ba6a12c76549dd76ae0a146faf0bfdf35ffd4a4e6ea7","impliedFormat":1},{"version":"a88437446e80a492b1c4d3f5c9fff5d80b5c5e52754cbb3eb2cfee3d3690ca94","impliedFormat":1},{"version":"bace2dc66c954f2a81c641fa9f0dcb1b36ddbc6db3635ea446ee10c47ada15f1","impliedFormat":1},{"version":"c5c7f25f198dfc5ffc62fe2e8ef3f25647bf21070a5f05ac200748c83ab7da4f","impliedFormat":1},{"version":"60390e7b89c19d160b3bf2c854a9e06414d001debd9947a5db54623004a4be0e","impliedFormat":1},{"version":"5909d075463236fbe7fd8a785248203f6f32d73b032c774d5018adc3c44846d8","impliedFormat":1},{"version":"c08e7bfca5a8bb244cad7689ddf7546cec8a5bc5367b18bcadc0628ae927f797","impliedFormat":1},{"version":"b7506549d0f8ea4c74e4b4b4263932090578f193cb37bf719b44c5f149a934f6","impliedFormat":1},{"version":"992aafb2a060c3e2099941c7128d88aeb9bf8f5fcc594e9fe561d19003b5e4be","impliedFormat":1},{"version":"9874f63b3f3167f344d2a30047722b409e2915a502d9b9a50a91ab1a23b49623","impliedFormat":1},{"version":"b55dfdbd1e893c0b6cf91dca75395f4bd8aab8e624007f9fc70d650d8b340137","impliedFormat":1},{"version":"1740fa9c57b951441b1db4478a7f6a82ccec9de1de650920cbce74ed10e08eba","impliedFormat":1},{"version":"6948d2c91da770f73b9a6459c3daf8ab23d80bf7b70e215551ca3219ac041b68","impliedFormat":1},{"version":"32dd24f732e9efb063532a95a730e6bd68b7c8e54a3d4f026fb4a1fa71d115d0","impliedFormat":1},{"version":"e39c146a2b8a3f48452973628042cabc94bb2893488bd6a79b3e04cfcd89c729","impliedFormat":1},{"version":"60f5165cd2492544cf497f3eb4e8a75fa340185b4b98b8aa87b62853d57d1549","impliedFormat":1},{"version":"fe9cc3f1d04297f8d6995789f4df2b531a1ee7f1d0c8add6371281f4a31d195b","impliedFormat":1},{"version":"66b9b5e8625e6ada62c4d070918350dd10d01fa260426674448b25ffc7009488","impliedFormat":1},{"version":"0d25032744f0015a340edeb2e84e685a4c79ee1c9066d761d7fb0affbc2dfdc3","impliedFormat":1},{"version":"dbade843dd62f6241d1a30d944dd80152ddb554ea86a2778caf0f3ff990402ed","impliedFormat":1},{"version":"c5fe75259bda7aba824205a9138ea7f3bbc47d20ce777cea79d40930685b6ac8","impliedFormat":1},{"version":"3d485a48053321817c3ce51afa41c07b180b462274551d53c5a4927a5d052038","impliedFormat":1},{"version":"9e2f9ee99f0e172ef91af1d571e09743304b3b2430d41a8bcab357b878114757","impliedFormat":1},{"version":"5d6257ebe252d97b3d6fe3e0a49a0f148cd7312849f5f1d6f6b7265d3d72b5d2","impliedFormat":1},{"version":"2c60950709e37e95cc5dfa2ca27c5da53521ee09c254f894f8d91ae8717e7885","impliedFormat":1},{"version":"8bfc090ffec588f44eacbd6714f798a8a0c3dc1b02855f5e12e567b4f161b30b","impliedFormat":1},{"version":"b302d3e1a806fc890c324ebe90dfe07a780e973267c66bd159d0dbc1f6e3d055","impliedFormat":1},{"version":"b1c627fa2a4cc9199f937f4d35ccfdef2efd6ef40d5525ffd384acb29cbaf66e","impliedFormat":1},{"version":"5211c6778aff2c045d7fb44236ef9b91ab424ee916f6575901dd0dcd9d58857a","impliedFormat":1},{"version":"39959ee712b3455499af3b1c95bbfc9ea59d584d5af2b01dcde120fe5dc6fceb","impliedFormat":1},{"version":"bc27582d90eaa5a793cc4f3e27acff890eab95641431c263144f3162bbd4a8bc","impliedFormat":1},{"version":"2992d19be476415c0296bd548764c20fc9cac2876e45abbbce23dafbd65438d9","impliedFormat":1},{"version":"dc117b16848058e94c39b68cddd38b36be885a63a0130097e6e992cce6ad9bf4","impliedFormat":1},{"version":"11bc3d6606ca11c982d848ff3946f1d978360e7861dedd8bb97348a21b4a8ad7","impliedFormat":1},{"version":"989b88698577f76069fe791b224d2157a0205aa2b029718dfd386b7b4706fa0c","impliedFormat":1},{"version":"fab62208329b9bb74dfe558a6b05f802bceda19274c763efd8ea0b47cb68925b","impliedFormat":1},{"version":"b7126f4bafda5e7424504458d53b298d92139de65e34859ee67f6f8f1d43bfdd","impliedFormat":1},{"version":"34e922a2625a8cc2f2c586afc7f7068cfdac27bd5119cdfc10f8e22562b2a142","impliedFormat":1},{"version":"d57be402cf1a3f1bd1852fc71b31ff54da497f64dcdcf8af9ad32435e3f32c1f","affectsGlobalScope":true,"impliedFormat":1},"680203bbde1377dd63b6ac43dbdf8e0bdcdf9073b36e25ac29bcec89f5e2b367",{"version":"f802525d218ecc342811e1802be8dbe71f130c1b3cf002622bfe19f2e97cbc24","signature":"bedfafeb11035f489d9c126bcb1ae7b5c7af3841e30859c841458e570c4ea64f"},{"version":"fb1d369b7552812b6e25e061e2f98eb95570f601180d8189ab4ae70949485aef","signature":"fcc72833c5e15bbf94dbc9150f7c04da8e8a267bba961be6b658f7e2697f95d5"},{"version":"9862899a87d5ce7e81af10f54727911791f870df82637dbb2f44616d4055e887","signature":"a73e187985807b51a8f112ae5878c918ac9d4596b50a6e915aa18b610cfdb971"},{"version":"419771f45d061e80cc9f252cfc1938516fcd27dc2b5f24bd8b2d7f4158231098","signature":"34d6048bb0b999ace48a71eb1bc2d8e7d4c750a000a2da29a3c686a8f21a0c97"},{"version":"53477a1815e915b8c20222a2ac8f9e3de880a1e8c8dbf9dae529b3d2e2b4a53b","impliedFormat":1},{"version":"6730b5ebe441181366c167599ff9ed47f49c5ee56927d2feff813ac7b004f189","signature":"7be3336b7bdc738a07e00ce2d1dc4d132b57fe1ff4b6a97c5c5dbd471c7b8a40"},{"version":"adef4e0a24a756892b56de5b110e0051cb2eb60cd568aeaf246b095f472f0d8d","signature":"8e609bb71c20b858c77f0e9f90bb1319db8477b13f9f965f1a1e18524bf50881"},{"version":"2419aa672f909d13dfd164095d85e02e57dcdebb2925c581ab33ba0244005b24","signature":"6330fcaaaac2f9aa939eadd88d82d1a85b0b6a6e69a8777c82cb7429102eb97e"},{"version":"847bfe94652015a13523a1e53431a64c55d3ffd7998838149f594ced76dfeec4","signature":"979f78cc24ed146ffecec0058da27d34791907ce7f177d6cc6c9f414b103fbcb"},{"version":"a467f12e45f0444b0ea55f5ceeaa4ede0e5f8a70040f1989c7b7dc7872b8bd26","signature":"dfdb99c2d86424a91a1b66f3141537e94f392d763d20c0da3715e5a8ba257e05"},{"version":"0a9139419422089c3c587b8346ee849512f60cff61e76f9f3662eda90083db8e","signature":"8e609bb71c20b858c77f0e9f90bb1319db8477b13f9f965f1a1e18524bf50881"},{"version":"673246d695015ccb37e416ed94ce11d2427269d4aca72d290729d73c91f3028b","signature":"5b9cdd9ab3b512a0871f08f904b72cdb9f97d075e5df72d6288327a0ab9859ec"},{"version":"ea5d6b31c3fa4bc3a6483ec5c7aea699593b9409e8e4a8a11eb7346d8f8a3b5c","signature":"ff95913fe6b2f7fa38f1c08ebd63f0330313536558c839144188b6f42d0988f2"},{"version":"854a097df742ab5358f72c33e0ab4e6ea94f3b53adee29bf98df5d54c7818af7","signature":"c47c32e6fc0077cf39dba38235c92bc4fe5283f4db855ebdcd581c55003d5292"},{"version":"38ad21890f11e291c38999262c5bf7acc7912250dbed035bd1f4c01b745256da","signature":"7214164cfa10573d2bf79da5174719e21aa58d3745730ba5464fd4491e9d5c6c"},{"version":"87b5c19561e83b7ba55e3e47a422f923e6f4cc45df742e759ac6578a6c625a20","signature":"8f61f444f0045fa63c25cd34e7d6854bb588fc30ebcf0f72f8703a5babf8d604"},{"version":"aa3c3bee0bb94d2af2ccdcc62778bdb7867142a99e8abe98b1293698f255e9d0","signature":"afc3124d4a264fa8f31e60c7fa02fef3ca66cbb61fdf24e1191e65d593efee88"},{"version":"f1a6a927bcad34720e03191c80e1d4a89e7de903af38820ae4a56ef29f303845","signature":"972bc67023c8577c5dc1eb3b86b212c4640e91044b838a6eb8f17b0cbc81826d"},{"version":"1c2658f19801c7ee5b12f70c21b0e2b50475642c4af3599b2cfb3666996b3a37","signature":"347f871052b9155c8c92fb3fb1b7e1b13a83222ad782b9c06a6b05220048e7e2"},{"version":"ac3e100ff0f0d2af86a0e3b2bf7a55168cd629064de596fe72201902a36bde4b","signature":"d2a93b45dadf0e50a965e739b097650c54742500f23fd607bc24c8c62b8efeb2"},{"version":"b4d5ec4023f7fe5bdb6d559a7d65ddded6c58459329856afe4629c0f1a3f496a","signature":"228e6b3701c575947bd0dffc7605f386c985464aa23ef98fe898b0bd4c3232e7"},{"version":"7e0c4c48981977973a2829ee7a934528edea949defe00094ed8a75a551d7edc3","signature":"86c6d5d5f5eb4099d13e9c6bb573bae0ade1f9e6302f2a42dba61f4cc81a0026"},{"version":"fd7c548b191c544f7208ef5d300a244bcaba1e49eb94ecb78bd05ac141e4494f","signature":"3d2635dffbf657de8f93f1fac494df51732628bde4d2ac5d663004499cedebcf"},{"version":"a2c7d8a23ba712eaf9d7b6616fde3a5b5836463e921dd209a784c1a2297784e0","signature":"597b27db1df53cf078d88e04a0ba9b5b876f3b4d88db5d8aebdf9cbd9787a248"},"de35364a7d8a1a504db3e2d92e8f6a55e0bfa028531d315846d2b8c61f6c69f8",{"version":"3e0989627868e045b7ba4bd061b62a5a143e96feee6b0bceeaa3dcf3efe1d0cc","signature":"f73cc97f6f64b3e02abb8533f8ab0cb9cc04bacca9d3af96308eed1c0f256722"},{"version":"d69a4081adcecb3e589389dc852758a6c9789e569468aecde84c8cea25a4ed3a","signature":"be2b2e3db777b1783a40ca539018957e67fd3be3943f58474c30bb200ee35be2"},"2d870c70e4dba5edb00b4bf1667a29b24785353a8df70e352f5c8bfb30b756e0",{"version":"44a6eb6a7bb83ee04e29ca8053eb533a154e0608137cd5262c70b60e1556b055","signature":"569c53fbd7f02f81c7caa65dad43a98bdd94d39205e52c87a09a741f2413604f"},{"version":"72ec1e56060fc6d4f0d712065c092174948f88ccca8107905a6f41b3bd380973","signature":"416ade3443cdd6fb6b447d78c431d5c1270c84fa1d6ca203cfa61920791dc3ff"},"7909d395a0a16a3e84a5948c4bbeff5f1e7230dc92090e5ee4f10a736166db1b","96e5be6af3096b7526ceeb751f425f3243bc92c834012472b86e1dbf024ce4e9","330fc45f05f5098d77eb088e3c129822acd15483a2a073b5b5c4fc5242d77110",{"version":"4c559d39dbb1db83c512dda48af3707514db6d4aeb2c18bf14fef65480aaae7d","signature":"30d13cfe85c9240cc5b1c85eede475347bad8b7832f44104f70a9eaa938ba267"},{"version":"4d184a628301e80a08894bde4ddc06cd2f635f7d2be0e6d6b1d7e3dc809518d4","signature":"fb93a375945d0c4e09fc13c423be21d82e99c4467f73ee329c1bbc8e59e39765"},"8c440684c3845d829916f1c75b983f69263707d7bf1db14aaee75fd890dddb26",{"version":"161b75949942f4a9abdb6c1b083cd61f2bd36894ac987fcadd5945a52199dba7","signature":"50951aed525523bfba5b0d887a9d10922a3b7f09aebe1f2b41b7766604670935"},"38e7f24c7ac2f3b8025ee5240b7ea26f219e7684b1b856ab4b938871e77a7507","07bf5700a763e91760556a4b524c4d6f1cb1e4d7e4c5b5d23dec8ff7ce40f383",{"version":"b9b38d7f77490cd0500cfbb2483f41b371fe43f519ab14fe82ea19423f99f01b","signature":"361395f780cd87cdf2149fb54475d608ae5c50db9f1c6139697398e6f77fd1a5"},{"version":"77c200990593b256a283cfeb926fa6e0d0bc6afeb9e8f507cac1f1dd9f80d658","signature":"08caa88d715651b88f934b90b704da3ab1a0ca10eb080d96114c10359ea397e2"},{"version":"a17fc4140ae99c7d9cff1cdc8c6367285c108f3b9dddd1f51edfd1d8450327d3","signature":"b837acb1964e44877ccc769ce8247a96a977b9da7e15bbd769c270f39436971a"},{"version":"005d274917f9fef7804384643784a6c095000caa5f833e8c6e11a827f10dc2b3","signature":"36a1ce3394a43cfe01eebea44e429c10af3d4bdae05f7d66c3b15414ed190f4b"},{"version":"42eb059f3fb628161c23ac89c9bafbee3632b7c593724d8e5c62cfe38fe8f49e","signature":"0e2f7b37c8ed3d475c413af32bd0f29c2cfb7059a9d538c6a37997933609677f"},{"version":"c861d7beb7017ffff746a0590cfb228288879f43fcb8ed7ef68f29f88eb85247","signature":"f6f8a2f4594785edf70c5e00392ef15e6d04bf0cfbe651312376007fa5920207"},{"version":"75d98473451055116db771cca837c103d44e7746bf737a0cfec8e49cfa1cab94","signature":"f2044555a4c76a4cc8aaac6f60c7dc42deb1c35b572e9777fc63a63e55f93ad2"},{"version":"2bad09c4dc0810666ef5b6150aa910dd711051ce5f2184050c9859c708092a36","impliedFormat":1},{"version":"eece99a6cf69ff45c5d4f9e0bfb6450f5c57878d048ff01a6a6343cf87e98230","impliedFormat":1},{"version":"f7ab1fe738bbe7fdd1e9bc9887f55ac0d7eda0d234a7eb35c77304430f7d6715","impliedFormat":1},{"version":"1f763e81e71ac8ce0dd1b82e67cd9d1d2e56e77c844b46349e1623c82df9b130","impliedFormat":1},{"version":"1a9c0db9d65449e9dbcbf23baa3b8bfa48806cddb2adc5e172eb8eff5afbb702","impliedFormat":1},{"version":"4dffcfc24d378d52ade01a1ae0c5baf6dc7a50c35f8302be34f06b9eaa5798ce","impliedFormat":1},{"version":"6586eacd77a813c50d7d2be05e91295989365204d095463ca8c9dfb8caac222d","impliedFormat":1},{"version":"3f2b3c5d3f5fd9e254046b9bf83da37babd1935776c97a5ffc1acfce0da0081e","impliedFormat":1},{"version":"8f1241f5d9f0d3d72117768b3c974e462840fbd85026fb66685078945404cf2f","impliedFormat":1},{"version":"8506c0b5eef3d94fe007f3886d427418739ba2b7a5cf58517af00473f8bdbb64","signature":"732b1aaa3d04b606bae1bdf8a629853c78d81a2c8efb0bee27896c8b25320c86"},"035c68841ad8e26535a70240373bcfeedef2e5920451191cf1fa4860dabd5dfe",{"version":"e0db8da553b3d1f7367cd81747ceabbf97e07a28c1280ecef4df32689730d938","signature":"d427aca7c2719cfc78aed76fe1eb93a3b3a1c1ec202605d333de863312911f5d"},{"version":"d77d188bdf74ca7ff037b5bd0f0ddbb3916e4e3fb58f292c461a0bc4b4ea4edc","signature":"5165ae81ae53c152e0b75e1f43a55eae37e44575676de86c95c44b7fd682bcbf"},{"version":"7745b1ebfb17bf789f497012f60d58c56491efec8dbd4fcb98058d762352ee32","signature":"45cf00eb714d66a8edf471c97c73d56e867885fbfa3f32c0170a9445a52e303d"},{"version":"4706bcc3b04fa608c868d2fa1ca0442239a673b321ee6465453671540962fe10","signature":"a4f2799280f3f51209526edea6d866193080a96d06485323dc38d84e010697d1"},{"version":"75e1d28d6e43fb4d68ed2f2565c95f24cd7715cd3562529ec19c738b1a6f7d56","signature":"f63fe66413602f13ee38acbe9809e4aa72fc71a10e9a71e4889faa83e2851b40"},{"version":"ab6b80ea01c6e24f8c5a0fe6b38f8ad960d5d0d6a3f91ead3f28d695ea7b21d4","signature":"826de80669881a355fb51cbad80290ad3b5788a78c0ad1455c3e855267cced6d"},{"version":"924956eac4e4f24e18da253d9e646c7be736c616e0a80eeedffd5a1e3c146564","signature":"6da83de6ab069bd38ec0186dcac008302e2271520b4b4cfddfdfd5342c673a42"},{"version":"cdcc132f207d097d7d3aa75615ab9a2e71d6a478162dde8b67f88ea19f3e54de","impliedFormat":1},{"version":"0d14fa22c41fdc7277e6f71473b20ebc07f40f00e38875142335d5b63cdfc9d2","impliedFormat":1},{"version":"e1028394c1cf96d5d057ecc647e31e457b919092f882ed0c7092152b077fed9d","impliedFormat":1},{"version":"f315e1e65a1f80992f0509e84e4ae2df15ecd9ef73df975f7c98813b71e4c8da","impliedFormat":1},{"version":"5b9586e9b0b6322e5bfbd2c29bd3b8e21ab9d871f82346cb71020e3d84bae73e","impliedFormat":1},{"version":"3e70a7e67c2cb16f8cd49097360c0309fe9d1e3210ff9222e9dac1f8df9d4fb6","impliedFormat":1},{"version":"ab68d2a3e3e8767c3fba8f80de099a1cfc18c0de79e42cb02ae66e22dfe14a66","impliedFormat":1},{"version":"d96cc6598148bf1a98fb2e8dcf01c63a4b3558bdaec6ef35e087fd0562eb40ec","impliedFormat":1},{"version":"f8db4fea512ab759b2223b90ecbbe7dae919c02f8ce95ec03f7fb1cf757cfbeb","affectsGlobalScope":true,"impliedFormat":1}],"root":[438,439,1591,[1602,1614],1690,1702,1703,[1718,1740],1760,[2240,2242],[2244,2248],[2266,2269],[2271,2278],[2418,2420],[2498,2512],2537,[2567,2593],[2621,2626],[3113,3115],[3118,3155],[3165,3172],[3188,3194],[3197,3217],[3219,3251],[3299,3323],[3595,3668],[3769,3776],[3783,3816],[4023,4031],[4315,4351],[4550,4554],[4556,4596],[4606,4614]],"options":{"allowSyntheticDefaultImports":true,"alwaysStrict":true,"declaration":false,"emitDecoratorMetadata":true,"esModuleInterop":true,"experimentalDecorators":true,"module":1,"noFallthroughCasesInSwitch":true,"noImplicitAny":false,"outDir":"./dist","removeComments":true,"rootDir":"./src","skipLibCheck":true,"sourceMap":true,"strict":false,"strictBindCallApply":false,"strictFunctionTypes":true,"strictNullChecks":false,"strictPropertyInitialization":false,"target":8},"referencedMap":[[2998,1],[3081,1],[2921,2],[2924,3],[3008,4],[3088,5],[3003,6],[2997,7],[2927,2],[2931,8],[2929,9],[2930,1],[3001,10],[3000,11],[2999,12],[2996,1],[3004,13],[2928,2],[2926,2],[3002,14],[3086,15],[3080,16],[3010,2],[3014,17],[3012,18],[3013,1],[3084,19],[3083,20],[3082,21],[3079,1],[3087,22],[3011,2],[3009,2],[3085,23],[2922,24],[3007,24],[2923,25],[2919,26],[2917,27],[2914,2],[2915,28],[2920,29],[2918,2],[2925,2],[3006,30],[3005,2],[3993,31],[3992,32],[3989,33],[3817,34],[3820,35],[3821,35],[3822,35],[3823,35],[3824,35],[3825,35],[3826,35],[3827,35],[3828,35],[3829,35],[3830,35],[3831,35],[3832,35],[3833,35],[3834,35],[3835,35],[3836,35],[3837,35],[3838,35],[3839,35],[3840,35],[3841,35],[3842,35],[3843,35],[3844,35],[3845,35],[3846,35],[3847,35],[3848,35],[3849,35],[3850,35],[3851,35],[3852,35],[3853,35],[3854,35],[3855,35],[3856,35],[3857,35],[3858,35],[3859,35],[3860,35],[3861,35],[3862,35],[3863,35],[3864,35],[3865,35],[3866,35],[3867,35],[3868,35],[3869,35],[3870,35],[3871,35],[3872,35],[3873,35],[3874,35],[3875,35],[3876,35],[3877,35],[3878,35],[3879,35],[3880,35],[3881,35],[3882,35],[3883,36],[3884,35],[3885,35],[3886,35],[3887,35],[3888,35],[3889,35],[3890,35],[3891,35],[3892,35],[3893,35],[3894,35],[3895,35],[3896,36],[3897,35],[3898,35],[3899,35],[3900,35],[3902,37],[3903,37],[3904,37],[3905,37],[3906,37],[3907,37],[3908,37],[3909,37],[3910,37],[3911,37],[3912,37],[3913,37],[3914,37],[3915,37],[3916,37],[3917,37],[3918,37],[3919,37],[3920,37],[3921,37],[3922,37],[3923,37],[3924,37],[3938,37],[3939,37],[3925,37],[3926,37],[3927,37],[3928,37],[3929,37],[3930,37],[3931,37],[3932,37],[3933,37],[3934,37],[3935,37],[3936,37],[3937,37],[3940,37],[3941,37],[3942,37],[3943,37],[3944,37],[3945,37],[3946,37],[3947,37],[3948,37],[3949,37],[3950,37],[3951,37],[3952,37],[3953,37],[3954,37],[3955,37],[3956,37],[3957,37],[3958,37],[3959,37],[3960,37],[3961,37],[3962,37],[3963,37],[3964,37],[3965,37],[3966,37],[3967,37],[3968,37],[3969,37],[3970,37],[3971,37],[3972,37],[3973,37],[3974,37],[3975,37],[3976,37],[3977,37],[3978,37],[3979,37],[3980,37],[3981,37],[3982,37],[3983,37],[3984,37],[3985,37],[3986,37],[3987,37],[3994,38],[3988,39],[3990,40],[4022,41],[4020,42],[3818,2],[4021,43],[3819,44],[3901,45],[3996,46],[3997,47],[3998,48],[3999,49],[4005,50],[4004,51],[4001,52],[4002,53],[4003,54],[4000,55],[4006,56],[4008,57],[4007,58],[4009,59],[4010,60],[4011,61],[4012,62],[4013,63],[4014,64],[3991,65],[3995,66],[4019,67],[4015,68],[4016,69],[4017,70],[4018,71],[2482,72],[2481,73],[2478,74],[2421,75],[2424,76],[2425,76],[2426,76],[2427,76],[2428,76],[2429,76],[2430,76],[2431,76],[2432,76],[2433,76],[2434,76],[2435,76],[2436,76],[2437,76],[2438,76],[2439,76],[2440,76],[2441,76],[2442,76],[2443,76],[2444,76],[2445,76],[2446,76],[2447,76],[2448,76],[2449,76],[2450,76],[2451,76],[2452,76],[2453,76],[2454,76],[2455,76],[2456,76],[2457,76],[2458,76],[2459,76],[2460,76],[2461,76],[2462,76],[2463,76],[2464,76],[2465,76],[2466,76],[2467,76],[2468,76],[2469,76],[2470,76],[2471,76],[2472,76],[2473,76],[2474,76],[2475,76],[2476,76],[2483,77],[2477,39],[2479,78],[2497,79],[2495,42],[2422,2],[2496,80],[2423,81],[2486,82],[2485,83],[2487,84],[2488,85],[2489,86],[2490,87],[2491,88],[2492,89],[2493,90],[2494,91],[2480,92],[2484,66],[2223,93],[2219,94],[2216,95],[2105,96],[2108,97],[2109,97],[2110,97],[2111,97],[2112,97],[2113,97],[2114,97],[2115,97],[2116,97],[2117,97],[2118,97],[2119,97],[2120,97],[2121,97],[2122,97],[2123,97],[2124,97],[2125,97],[2126,97],[2127,97],[2128,97],[2129,97],[2130,97],[2131,97],[2133,97],[2132,97],[2134,97],[2135,97],[2136,97],[2137,97],[2138,97],[2139,97],[2140,97],[2141,97],[2142,97],[2143,97],[2144,97],[2145,97],[2146,97],[2147,97],[2148,97],[2149,97],[2150,97],[2151,97],[2152,97],[2153,97],[2154,97],[2155,97],[2156,97],[2157,97],[2158,97],[2159,97],[2160,97],[2161,97],[2162,97],[2163,97],[2164,97],[2165,97],[2166,97],[2167,97],[2168,97],[2169,97],[2170,97],[2171,97],[2172,97],[2173,97],[2174,97],[2175,97],[2178,97],[2176,97],[2177,97],[2179,97],[2180,97],[2181,97],[2182,97],[2183,97],[2184,97],[2185,97],[2186,97],[2187,97],[2188,97],[2189,97],[2190,97],[2191,97],[2192,97],[2193,97],[2194,97],[2195,97],[2196,97],[2197,97],[2198,97],[2199,97],[2200,97],[2201,97],[2202,97],[2203,97],[2204,97],[2205,97],[2206,97],[2208,98],[2209,99],[2210,99],[2211,99],[2212,99],[2213,99],[2214,99],[2215,99],[2224,100],[2104,39],[2217,101],[2239,102],[2237,42],[2106,2],[2238,103],[2107,104],[2207,105],[2226,106],[2227,107],[2228,108],[2229,109],[2230,110],[2231,111],[2218,112],[2225,66],[2236,113],[2232,114],[2233,114],[2234,115],[2235,115],[2098,116],[2097,117],[2099,118],[2101,119],[2096,120],[2100,121],[2103,122],[2102,2],[1990,39],[1988,123],[1989,2],[2009,124],[1995,125],[1996,125],[1994,126],[1997,126],[1993,127],[1991,128],[1992,129],[1998,2],[1999,39],[2006,130],[2005,131],[2003,39],[2004,132],[2007,133],[2001,134],[2002,135],[2000,135],[2008,39],[1849,136],[1850,136],[1852,137],[1848,2],[1854,138],[1853,138],[1855,138],[1856,139],[1857,140],[1851,141],[1858,39],[1859,39],[2011,142],[2010,143],[1861,144],[1860,39],[1862,39],[1905,145],[1904,146],[1907,147],[1920,121],[1921,118],[1933,148],[1922,149],[1934,150],[1903,151],[1906,152],[1935,153],[1936,39],[1937,154],[2012,39],[2014,155],[2013,156],[1863,39],[1864,39],[1865,39],[1866,39],[1867,39],[1868,39],[1869,39],[1878,157],[1879,39],[1880,2],[1881,39],[1882,39],[1883,39],[1884,39],[1872,2],[1885,2],[1886,39],[1871,158],[1873,159],[1870,39],[1874,158],[1875,159],[1876,160],[1902,161],[1887,39],[1888,159],[1889,39],[1890,39],[1891,2],[1892,39],[1893,39],[1894,39],[1895,39],[1896,39],[1897,39],[1898,162],[1899,39],[1900,39],[1877,39],[1901,39],[3588,163],[3499,164],[3500,164],[3501,164],[3502,164],[3503,164],[3504,164],[3505,164],[3506,164],[3507,164],[3508,164],[3509,164],[3510,164],[3511,164],[3512,164],[3513,164],[3514,164],[3515,164],[3516,164],[3517,164],[3518,164],[3519,164],[3520,164],[3521,164],[3522,164],[3523,164],[3524,164],[3525,164],[3526,164],[3527,164],[3528,164],[3529,164],[3530,164],[3531,164],[3532,164],[3533,164],[3534,164],[3535,164],[3536,164],[3537,164],[3538,164],[3539,164],[3540,164],[3541,164],[3542,164],[3543,164],[3544,164],[3545,164],[3546,164],[3547,164],[3548,164],[3549,164],[3550,164],[3551,164],[3552,164],[3553,164],[3554,164],[3555,164],[3556,164],[3557,164],[3558,164],[3559,164],[3560,164],[3561,164],[3562,164],[3563,164],[3564,164],[3565,164],[3566,164],[3567,164],[3568,164],[3569,164],[3570,164],[3571,164],[3572,164],[3573,164],[3574,164],[3575,164],[3576,164],[3577,164],[3578,164],[3579,164],[3580,164],[3581,164],[3582,164],[3583,164],[3584,164],[3585,164],[3586,165],[3496,2],[3498,2],[3587,166],[3497,167],[3379,168],[3378,169],[3377,170],[3368,171],[3374,172],[3325,173],[3327,174],[3366,175],[3367,176],[3376,177],[3369,178],[3372,179],[3371,179],[3370,179],[3373,180],[3324,2],[3375,181],[2989,182],[3072,183],[2987,182],[3070,183],[2988,182],[3071,183],[2990,182],[3073,183],[2991,182],[3074,183],[2995,184],[3078,185],[2993,186],[3076,187],[2994,186],[3077,187],[2992,182],[3075,183],[2974,2],[3057,2],[2932,188],[3015,188],[2983,189],[3066,190],[2951,1],[3034,1],[2971,1],[3054,1],[2944,1],[3027,1],[2977,1],[3060,1],[2976,1],[3059,1],[2985,1],[3068,1],[2964,191],[3047,192],[2972,193],[3055,194],[2982,191],[3065,192],[2958,1],[3041,1],[2945,195],[3028,196],[2980,191],[3063,192],[2949,195],[3032,196],[2948,195],[3031,196],[2938,191],[3021,192],[2935,1],[3018,1],[2937,191],[3020,192],[2939,1],[3022,1],[2967,1],[3050,1],[2981,195],[3064,196],[2947,195],[3030,196],[2957,1],[3040,1],[2946,1],[3029,1],[2934,1],[3017,1],[2963,191],[3046,192],[2986,197],[3069,198],[2978,2],[3061,2],[2969,2],[3052,2],[2970,1],[3053,1],[2984,199],[3067,200],[2933,195],[3016,196],[2961,2],[3044,2],[2950,195],[3033,196],[2979,2],[3062,2],[2954,2],[3037,2],[2965,2],[3048,2],[2942,201],[3025,202],[2943,195],[3026,196],[2940,191],[3023,192],[2953,191],[3036,192],[2959,1],[3042,1],[2952,1],[3035,1],[2975,1],[3058,1],[2956,195],[3039,196],[2955,1],[3038,1],[2960,191],[3043,192],[2936,1],[3019,1],[2962,1],[3045,1],[2941,1],[3024,1],[2968,2],[3051,2],[2966,195],[3049,196],[2973,2],[3056,2],[2736,1],[3759,203],[3761,204],[3768,205],[3765,206],[3767,207],[3766,208],[3764,209],[3762,210],[3763,211],[3724,212],[3725,213],[3726,212],[3727,210],[3723,207],[3721,207],[3722,207],[3729,214],[3730,210],[3734,210],[3735,210],[3731,210],[3732,214],[3733,210],[3736,214],[3737,210],[3738,210],[3728,207],[3739,210],[3758,215],[3754,210],[3755,210],[3740,210],[3741,210],[3742,210],[3743,210],[3744,210],[3745,210],[3746,210],[3747,210],[3748,210],[3749,210],[3750,210],[3751,210],[3752,210],[3753,210],[3756,207],[3757,207],[3720,216],[3760,2],[3716,2],[3708,217],[3718,2],[3709,2],[3714,2],[3719,218],[3717,2],[3707,219],[3715,220],[3704,221],[3705,2],[3706,222],[3669,2],[3710,223],[3713,224],[3712,225],[3711,226],[3670,2],[3671,2],[3672,2],[3684,2],[3673,2],[3674,2],[3675,2],[3676,2],[3679,2],[3681,2],[3682,2],[3677,2],[3678,2],[3680,2],[3701,227],[3683,2],[3685,2],[3686,2],[3687,2],[3688,2],[3694,2],[3695,2],[3689,2],[3691,2],[3690,2],[3692,2],[3693,2],[3696,2],[3697,2],[3698,2],[3699,2],[3700,2],[3703,2],[3702,223],[4615,2],[4618,228],[2913,229],[2912,2],[3093,230],[3094,231],[3095,232],[3096,232],[3097,233],[3098,1],[3099,1],[3102,234],[3100,1],[3101,1],[3107,235],[3089,236],[3090,237],[3091,238],[3092,239],[3106,240],[3105,241],[3103,242],[3104,243],[3779,244],[3780,245],[3781,246],[3777,247],[3778,248],[3782,249],[1615,2],[1616,2],[1618,250],[1617,2],[1619,251],[1620,252],[1623,253],[1621,2],[1622,254],[1652,255],[1648,256],[1657,257],[1653,258],[1654,2],[1655,258],[1656,259],[1658,2],[1659,2],[1660,260],[1664,261],[1649,262],[1647,259],[1651,263],[1650,264],[1661,2],[1662,2],[1663,265],[1676,2],[1679,266],[1680,267],[1681,258],[1682,258],[1683,268],[1687,269],[1684,270],[1685,271],[1677,272],[1678,273],[1686,274],[1688,275],[442,2],[315,2],[53,2],[304,276],[305,276],[306,2],[307,258],[317,277],[308,2],[309,278],[310,2],[311,2],[312,276],[313,276],[314,276],[316,279],[324,280],[326,2],[323,2],[329,281],[327,2],[325,2],[321,282],[322,283],[328,2],[330,284],[318,2],[320,285],[319,286],[259,2],[262,287],[258,2],[489,2],[260,2],[261,2],[347,288],[332,288],[339,288],[336,288],[349,288],[340,288],[346,288],[331,289],[350,288],[353,290],[344,288],[334,288],[352,288],[337,288],[335,288],[345,288],[341,288],[351,288],[338,288],[348,288],[333,288],[343,288],[342,288],[360,291],[356,292],[355,2],[354,2],[359,293],[398,294],[54,2],[55,2],[56,2],[471,295],[58,296],[477,297],[476,298],[248,299],[249,296],[369,2],[278,2],[279,2],[370,300],[250,2],[371,2],[372,301],[57,2],[252,302],[253,2],[251,303],[254,302],[255,2],[257,304],[269,305],[270,2],[275,306],[271,2],[272,2],[273,2],[274,2],[276,2],[277,307],[283,308],[286,309],[284,2],[285,2],[303,310],[287,2],[288,2],[520,311],[268,312],[266,313],[264,314],[265,315],[267,2],[295,316],[289,2],[298,317],[291,318],[296,319],[294,320],[297,321],[292,322],[293,323],[281,324],[299,325],[282,326],[301,327],[302,328],[290,2],[256,2],[263,329],[300,330],[366,331],[361,2],[367,332],[362,333],[363,334],[364,335],[365,336],[368,337],[384,338],[383,339],[389,340],[381,2],[382,341],[385,338],[386,342],[388,343],[387,344],[390,345],[375,346],[376,347],[379,348],[378,348],[377,347],[380,347],[374,349],[392,350],[391,351],[394,352],[393,353],[395,354],[357,324],[358,355],[280,2],[396,356],[373,357],[397,358],[533,258],[645,359],[646,360],[650,361],[534,2],[540,362],[643,363],[644,364],[535,2],[536,2],[539,365],[537,2],[538,2],[648,2],[649,366],[647,367],[651,368],[440,369],[441,370],[462,371],[463,372],[464,2],[465,373],[466,374],[475,375],[468,376],[472,377],[480,378],[478,258],[479,379],[469,380],[481,2],[483,381],[484,382],[485,383],[474,384],[470,385],[494,386],[482,387],[509,388],[467,389],[510,390],[507,391],[508,258],[532,392],[457,393],[453,394],[455,395],[506,396],[448,397],[496,398],[495,2],[456,399],[503,400],[460,401],[504,2],[505,402],[458,403],[452,404],[459,405],[454,406],[447,2],[500,407],[513,408],[511,258],[443,258],[499,409],[444,283],[445,372],[446,410],[450,411],[449,412],[512,413],[451,414],[488,415],[486,381],[487,416],[497,283],[498,417],[501,418],[516,419],[517,420],[514,421],[515,422],[518,423],[519,424],[521,425],[493,426],[490,427],[491,276],[492,416],[523,428],[522,429],[529,430],[461,258],[525,431],[524,258],[527,432],[526,2],[528,433],[473,434],[502,435],[531,436],[530,258],[3590,437],[3591,438],[3593,439],[3589,440],[3592,441],[3594,442],[1666,2],[1672,443],[1671,444],[1673,2],[1674,445],[1675,446],[1667,447],[1669,2],[1670,448],[1668,447],[2627,2],[2805,449],[2806,450],[2807,2],[2808,2],[2809,451],[2810,2],[2825,452],[2811,450],[2812,2],[2813,453],[2814,454],[2815,2],[2816,2],[2817,454],[2818,451],[2819,455],[2820,2],[2821,456],[2822,2],[2823,457],[2824,458],[2880,459],[2881,460],[2884,461],[2886,462],[2887,463],[2885,453],[2797,1],[2883,464],[2878,465],[2888,1],[2882,2],[2889,2],[2879,466],[2890,467],[2911,468],[2628,2],[2803,1],[2732,469],[2903,470],[2733,1],[2734,1],[2731,1],[2735,258],[2799,471],[2800,2],[2804,472],[2801,1],[2860,2],[2802,473],[2798,2],[2826,474],[2893,475],[2891,1],[2892,1],[2851,476],[2840,477],[2838,478],[2841,479],[2850,480],[2845,481],[2853,482],[2847,483],[2854,484],[2843,480],[2855,482],[2844,485],[2852,486],[2856,482],[2848,487],[2858,488],[2859,2],[2897,489],[2832,490],[2827,2],[2833,2],[2834,2],[2836,491],[2842,492],[2846,493],[2828,494],[2831,495],[2829,496],[2835,497],[2896,2],[2830,453],[2839,1],[2837,498],[2895,499],[2849,500],[2894,501],[2857,502],[2861,503],[2862,504],[2863,505],[2874,506],[2877,507],[2875,508],[2876,509],[2898,2],[2899,510],[2901,511],[2900,512],[2902,2],[2908,513],[2904,514],[2905,514],[2906,514],[2907,514],[2909,2],[2910,515],[2737,188],[2788,516],[2757,1],[2778,1],[2750,1],[2782,1],[2781,1],[2795,2],[2792,2],[2790,1],[2771,517],[2767,518],[2787,517],[2794,2],[2764,1],[2751,519],[2785,517],[2755,519],[2754,519],[2744,517],[2741,520],[2743,517],[2745,1],[2774,1],[2740,1],[2786,519],[2753,519],[2763,1],[2752,1],[2739,1],[2770,517],[2796,521],[2776,2],[2777,1],[2789,522],[2738,519],[2793,522],[2768,522],[2756,519],[2784,2],[2760,2],[2791,518],[2772,2],[2748,523],[2749,519],[2746,524],[2759,517],[2765,1],[2758,1],[2780,1],[2762,519],[2761,1],[2766,517],[2742,1],[2769,1],[2747,1],[2775,2],[2773,519],[2779,2],[3163,525],[3159,526],[3158,527],[3160,2],[3161,528],[3162,529],[3164,530],[4604,531],[4599,532],[4597,258],[4600,532],[4601,532],[4602,532],[4603,258],[4598,2],[4605,533],[1691,2],[1695,534],[1700,535],[1692,258],[1694,536],[1693,2],[1696,537],[1698,538],[1699,539],[1701,540],[1744,541],[1745,542],[1759,543],[1747,544],[1746,545],[1741,546],[1742,2],[1743,2],[1758,547],[1749,548],[1750,548],[1751,548],[1752,548],[1754,549],[1753,548],[1755,550],[1756,551],[1748,2],[1757,552],[1581,553],[1584,554],[1582,2],[1583,2],[1562,2],[1563,555],[1588,556],[1585,258],[1586,557],[1587,553],[1589,558],[399,2],[400,2],[403,559],[404,2],[405,2],[407,2],[406,2],[421,2],[408,2],[409,560],[410,2],[411,2],[412,561],[413,559],[414,2],[416,562],[417,559],[418,563],[419,561],[420,2],[422,564],[427,565],[436,566],[426,567],[401,2],[415,563],[424,568],[425,2],[423,2],[428,569],[433,570],[429,258],[430,258],[431,258],[432,258],[402,2],[434,2],[435,571],[437,572],[1716,573],[1707,574],[1713,2],[1704,2],[1705,575],[1708,576],[1709,258],[1710,577],[1706,578],[1711,579],[1712,580],[1714,581],[1715,2],[1041,582],[1039,583],[1040,584],[1045,585],[1038,586],[1043,587],[1042,588],[1044,589],[1046,590],[2514,591],[2513,592],[2515,2],[2516,2],[2529,593],[2517,258],[2527,594],[2528,2],[2531,595],[2530,2],[2532,258],[2533,596],[2535,597],[2536,598],[2518,599],[2522,600],[2519,2],[2520,2],[2521,2],[2526,601],[2534,2],[2523,330],[2524,2],[2525,2],[3260,602],[3263,603],[3268,604],[3271,605],[3292,606],[3270,607],[3252,2],[3253,608],[3254,609],[3257,2],[3255,2],[3256,2],[3293,610],[3259,602],[3258,2],[3294,611],[3262,603],[3261,2],[3298,612],[3295,613],[3265,614],[3267,615],[3264,616],[3266,617],[3296,618],[3269,602],[3297,619],[3282,620],[3284,621],[3286,622],[3285,623],[3279,624],[3272,625],[3291,626],[3288,627],[3290,628],[3275,629],[3277,630],[3274,627],[3278,2],[3289,631],[3276,2],[3287,2],[3273,2],[3280,632],[3281,2],[3283,633],[4474,634],[4475,635],[4476,634],[4477,634],[4478,634],[4479,634],[4480,634],[4481,635],[4482,634],[4489,636],[4483,634],[4484,634],[4485,635],[4486,634],[4487,634],[4488,634],[4490,637],[4473,638],[4357,639],[4359,640],[4358,2],[4450,641],[4448,2],[4446,2],[4451,642],[4449,643],[4447,2],[4457,644],[4453,645],[4463,646],[4461,647],[4464,648],[4456,649],[4455,650],[4354,651],[4462,652],[4452,653],[4445,654],[4472,655],[4458,656],[4460,657],[4459,650],[4367,634],[4368,634],[4369,634],[4370,634],[4371,634],[4372,658],[4373,635],[4376,634],[4361,634],[4377,659],[4378,635],[4362,634],[4379,634],[4363,634],[4364,634],[4380,634],[4381,634],[4365,634],[4382,634],[4383,660],[4384,634],[4385,634],[4387,661],[4390,661],[4389,661],[4386,634],[4391,661],[4388,661],[4392,634],[4393,634],[4394,635],[4395,635],[4400,634],[4401,634],[4396,634],[4398,634],[4399,634],[4402,634],[4403,634],[4404,634],[4405,662],[4374,635],[4406,634],[4407,634],[4408,634],[4409,663],[4410,634],[4411,634],[4412,634],[4397,634],[4366,634],[4413,635],[4414,634],[4441,634],[4442,634],[4443,634],[4416,664],[4417,635],[4419,634],[4418,635],[4420,634],[4421,634],[4422,634],[4423,634],[4424,634],[4425,634],[4426,635],[4427,635],[4428,665],[4429,634],[4430,664],[4432,635],[4431,635],[4433,635],[4434,635],[4375,635],[4435,634],[4436,634],[4437,634],[4438,634],[4439,635],[4440,635],[4415,2],[4353,634],[4444,666],[4355,2],[4356,667],[4454,668],[4470,669],[4471,670],[4469,671],[4468,2],[4467,672],[4466,673],[4465,674],[4360,668],[4491,635],[4492,634],[4493,635],[4494,634],[4495,634],[4496,634],[4497,634],[4498,635],[4499,635],[4500,634],[4501,634],[4502,635],[4503,634],[4504,634],[4505,634],[4506,675],[4507,676],[4510,677],[4511,678],[4508,635],[4512,679],[4513,634],[4514,680],[4515,634],[4516,681],[4517,682],[4518,683],[4509,684],[4519,685],[4520,634],[4521,634],[4522,634],[4523,635],[4524,686],[4525,687],[4526,688],[4529,689],[4528,690],[4530,634],[4532,634],[4531,690],[4533,691],[4534,692],[4535,688],[4536,693],[4537,694],[4544,695],[4539,695],[4540,695],[4541,696],[4543,695],[4542,697],[4538,691],[4527,698],[4545,699],[4546,700],[4617,2],[2015,118],[2016,118],[2019,701],[2018,702],[2017,39],[2029,703],[2020,118],[2022,704],[2021,39],[2024,705],[2023,2],[2025,706],[2026,706],[2027,707],[2028,708],[1985,709],[1986,710],[1939,2],[1938,2],[1987,711],[1940,712],[1971,66],[1972,713],[1975,714],[1974,715],[1976,716],[1973,39],[1953,717],[1954,2],[1984,718],[1977,148],[1978,2],[1979,719],[1980,719],[1982,720],[1981,719],[1983,66],[1969,721],[1955,39],[1970,722],[1957,723],[1956,39],[1964,724],[1959,725],[1960,725],[1965,39],[1961,725],[1958,39],[1966,725],[1963,725],[1962,39],[1967,39],[1968,39],[2082,39],[2083,2],[2086,726],[2094,727],[2087,2],[2088,2],[2089,2],[2090,2],[2091,2],[2092,2],[2093,2],[2030,39],[2031,728],[2034,729],[2036,730],[2035,39],[2037,729],[2038,729],[2040,731],[2032,39],[2039,39],[2033,2],[2055,120],[2059,732],[2056,39],[2058,39],[2052,733],[2051,734],[2053,2],[2054,39],[2050,39],[2057,735],[1846,736],[1831,39],[1844,737],[1845,39],[1847,738],[1928,39],[1929,739],[1926,740],[1927,741],[1925,742],[1923,39],[1924,39],[1932,743],[1930,2],[1931,39],[1836,2],[1840,2],[1832,2],[1833,2],[1834,2],[1835,2],[1843,744],[1837,745],[1838,39],[1839,746],[1842,2],[1841,39],[1909,747],[1908,39],[1910,2],[1916,39],[1911,39],[1912,39],[1913,39],[1917,39],[1919,748],[1914,39],[1915,39],[1918,39],[2077,39],[2060,39],[2061,749],[2062,750],[2063,2],[2064,751],[2065,2],[2066,2],[2067,2],[2068,39],[2069,749],[2070,39],[2072,752],[2073,753],[2071,39],[2074,2],[2075,2],[2095,754],[2076,2],[2078,2],[2079,749],[2080,2],[2081,2],[1761,755],[1762,756],[1764,2],[1777,757],[1778,758],[1775,759],[1776,760],[1763,2],[1779,761],[1782,762],[1784,763],[1785,764],[1767,765],[1786,2],[1790,766],[1788,767],[1789,2],[1783,2],[1792,768],[1768,769],[1794,770],[1795,771],[1797,772],[1796,773],[1798,774],[1793,775],[1791,776],[1799,777],[1800,778],[1804,779],[1805,780],[1803,781],[1781,782],[1769,2],[1772,783],[1806,784],[1807,785],[1808,785],[1765,2],[1810,786],[1809,785],[1830,787],[1770,2],[1774,788],[1811,789],[1812,2],[1766,2],[1802,790],[1818,791],[1817,792],[1814,2],[1815,793],[1816,2],[1813,794],[1801,795],[1819,796],[1820,797],[1821,762],[1822,762],[1823,798],[1787,2],[1825,799],[1826,800],[1780,2],[1827,2],[1828,801],[1824,2],[1771,802],[1773,776],[1829,755],[2043,803],[2044,804],[2045,805],[2041,39],[2046,2],[2047,2],[2049,806],[2048,2],[2042,2],[1941,2],[1943,39],[1942,807],[1944,808],[1945,809],[1946,807],[1947,807],[1948,810],[1952,811],[1949,807],[1950,810],[1951,2],[2221,812],[2222,813],[2220,39],[2085,814],[2084,2],[2555,2],[1599,815],[1598,816],[2546,816],[1595,817],[4352,818],[1601,819],[1600,817],[2270,820],[1596,2],[4623,821],[3157,822],[1573,823],[1566,824],[1570,825],[1568,826],[1571,827],[1569,828],[1572,829],[1567,2],[1565,830],[1564,831],[3156,2],[4549,832],[590,833],[591,833],[592,834],[546,835],[593,836],[594,837],[595,838],[541,2],[544,839],[542,2],[543,2],[596,840],[597,841],[598,842],[599,843],[600,844],[601,845],[602,845],[603,846],[604,847],[605,848],[606,849],[547,2],[545,2],[607,850],[608,851],[609,852],[641,853],[610,854],[611,855],[612,856],[613,857],[614,858],[615,859],[616,860],[617,861],[618,862],[619,863],[620,863],[621,864],[622,2],[623,865],[625,866],[624,867],[626,868],[627,869],[628,870],[629,871],[630,872],[631,873],[632,874],[633,875],[634,876],[635,877],[636,878],[637,879],[638,880],[548,2],[549,2],[550,2],[589,881],[639,882],[640,883],[3186,884],[3173,885],[3180,886],[3176,887],[3174,888],[3177,889],[3181,890],[3182,886],[3179,891],[3178,892],[3183,893],[3184,894],[3185,895],[3175,896],[3196,897],[3195,898],[1697,899],[1593,2],[1594,2],[1592,900],[1597,901],[2321,902],[2312,2],[2313,2],[2314,2],[2315,2],[2316,2],[2317,2],[2318,2],[2319,2],[2320,2],[2543,903],[3493,904],[3494,905],[3495,906],[3445,907],[3460,908],[3446,908],[3442,909],[3448,908],[3449,908],[3451,910],[3457,908],[3452,908],[3463,908],[3453,908],[3450,908],[3456,908],[3461,908],[3459,908],[3462,911],[3464,908],[3447,908],[3465,908],[3454,908],[3455,908],[3466,911],[3458,908],[3470,912],[3438,913],[3474,914],[3399,915],[3478,915],[3387,2],[3389,916],[3400,915],[3390,915],[3388,2],[3401,2],[3402,917],[3403,915],[3393,918],[3397,919],[3395,2],[3405,920],[3394,2],[3391,915],[3426,921],[3406,915],[3407,915],[3409,922],[3396,915],[3410,2],[3411,923],[3398,915],[3412,915],[3413,915],[3414,915],[3415,915],[3416,915],[3436,924],[3437,2],[3417,915],[3408,2],[3392,925],[3418,915],[3427,926],[3428,2],[3429,927],[3430,928],[3431,928],[3432,929],[3434,930],[3419,915],[3433,915],[3435,931],[3404,2],[3420,920],[3421,920],[3425,932],[3422,2],[3423,933],[3424,915],[3467,934],[3482,935],[3475,936],[3443,937],[3477,938],[3444,939],[3487,940],[3483,941],[3479,942],[3480,943],[3481,944],[3484,945],[3486,946],[3485,947],[3471,948],[3440,949],[3441,950],[3439,951],[3472,952],[3468,953],[3490,2],[3488,954],[3476,955],[3489,2],[3491,2],[3469,956],[3473,957],[3492,958],[1717,2],[3117,959],[3116,2],[551,2],[1646,960],[1561,961],[1412,962],[1413,2],[1414,962],[1415,2],[1416,963],[1417,964],[1418,962],[1419,962],[1420,2],[1421,2],[1422,2],[1423,2],[1424,2],[1425,2],[1426,2],[1427,964],[1428,962],[1429,962],[1430,2],[1431,962],[1432,962],[1433,2],[1439,965],[1434,965],[1435,964],[1436,2],[1437,2],[1438,966],[1463,967],[1440,964],[1454,968],[1441,968],[1442,968],[1443,968],[1444,964],[1445,968],[1446,968],[1447,968],[1448,968],[1449,964],[1450,964],[1451,964],[1452,968],[1453,969],[1455,964],[1456,964],[1457,2],[1458,2],[1460,2],[1459,2],[1461,964],[1462,2],[1464,970],[1411,971],[1401,972],[1398,973],[1406,974],[1404,975],[1400,976],[1399,977],[1408,978],[1407,979],[1410,980],[1409,981],[1049,2],[1052,964],[1053,964],[1054,964],[1055,964],[1056,964],[1057,964],[1058,964],[1060,964],[1059,964],[1061,964],[1062,964],[1063,964],[1064,964],[1176,964],[1065,964],[1066,964],[1067,964],[1068,964],[1177,964],[1178,2],[1179,982],[1180,964],[1181,963],[1182,963],[1184,983],[1185,964],[1186,984],[1187,964],[1189,985],[1190,963],[1191,986],[1069,976],[1070,964],[1071,964],[1072,2],[1074,2],[1073,964],[1075,987],[1076,976],[1077,976],[1078,976],[1079,964],[1080,976],[1081,964],[1082,976],[1083,964],[1085,963],[1086,2],[1087,2],[1088,2],[1089,964],[1090,963],[1091,2],[1092,2],[1093,2],[1094,2],[1095,2],[1096,2],[1097,2],[1098,2],[1099,2],[1100,25],[1101,2],[1102,988],[1103,2],[1104,2],[1105,2],[1106,2],[1107,2],[1108,964],[1114,963],[1109,964],[1110,964],[1111,964],[1112,963],[1113,964],[1115,962],[1116,2],[1117,2],[1118,964],[1192,963],[1119,2],[1193,964],[1194,964],[1195,964],[1120,964],[1196,964],[1121,964],[1198,962],[1199,962],[1200,962],[1197,962],[1201,964],[1202,963],[1203,963],[1204,964],[1122,2],[1206,962],[1205,962],[1123,2],[1124,989],[1125,964],[1126,964],[1127,964],[1128,964],[1130,963],[1129,963],[1131,964],[1132,964],[1133,964],[1207,963],[1208,963],[1209,964],[1210,964],[1213,963],[1217,963],[1219,990],[1215,963],[1216,991],[1218,990],[1220,992],[1211,963],[1212,993],[1214,992],[1221,963],[1223,994],[1222,994],[1224,964],[1225,963],[1226,964],[1227,964],[1228,964],[1229,964],[1230,964],[1134,995],[1231,963],[1232,964],[1233,996],[1234,964],[1235,964],[1236,963],[1237,964],[1238,964],[1239,964],[1240,964],[1241,964],[1242,964],[1243,996],[1244,996],[1245,964],[1246,964],[1247,964],[1248,997],[1249,998],[1250,963],[1251,999],[1252,964],[1253,963],[1254,964],[1255,964],[1256,964],[1257,964],[1258,964],[1259,964],[1135,2],[1136,964],[1137,2],[1138,2],[1139,964],[1140,2],[1141,964],[1260,976],[1262,1000],[1261,1000],[1263,1001],[1264,964],[1265,964],[1266,964],[1267,963],[1183,963],[1142,964],[1269,964],[1268,964],[1270,964],[1271,1002],[1272,964],[1273,964],[1274,964],[1275,964],[1276,964],[1277,964],[1143,2],[1144,2],[1145,2],[1146,2],[1147,2],[1278,964],[1279,995],[1148,2],[1149,2],[1150,2],[1151,962],[1280,964],[1281,1003],[1282,964],[1283,964],[1284,964],[1285,964],[1286,963],[1287,963],[1288,963],[1289,964],[1290,963],[1291,964],[1292,964],[1152,964],[1293,964],[1294,964],[1295,964],[1153,2],[1154,2],[1155,964],[1156,964],[1157,964],[1158,964],[1159,2],[1160,2],[1296,964],[1297,963],[1161,2],[1162,2],[1298,964],[1163,2],[1300,964],[1301,964],[1299,964],[1302,964],[1303,964],[1304,964],[1164,964],[1165,963],[1305,2],[1166,2],[1167,963],[1168,2],[1169,2],[1170,2],[1306,964],[1307,964],[1311,964],[1312,963],[1313,964],[1314,963],[1315,964],[1171,2],[1308,964],[1309,964],[1310,964],[1316,963],[1317,964],[1318,963],[1319,963],[1322,963],[1320,963],[1321,963],[1323,964],[1324,964],[1325,964],[1326,1004],[1327,964],[1328,963],[1329,964],[1330,964],[1331,964],[1172,2],[1173,2],[1332,964],[1333,964],[1334,964],[1335,964],[1174,2],[1175,2],[1336,964],[1337,964],[1338,964],[1339,963],[1340,1005],[1341,963],[1342,1006],[1343,964],[1344,964],[1345,963],[1346,964],[1347,963],[1348,964],[1349,964],[1350,964],[1351,963],[1352,964],[1354,964],[1353,964],[1355,963],[1356,963],[1357,963],[1358,963],[1359,964],[1360,964],[1361,963],[1362,964],[1363,964],[1364,964],[1366,964],[1365,1007],[1367,963],[1368,964],[1370,964],[1371,964],[1369,1008],[1372,964],[1188,963],[1373,963],[1374,963],[1375,1009],[1376,963],[1377,1010],[1378,964],[1379,1011],[1380,1012],[1381,964],[1383,964],[1384,964],[1385,1013],[1386,964],[1382,1014],[1387,964],[1388,964],[1389,964],[1390,964],[1391,964],[1392,964],[1393,963],[1394,963],[1395,964],[1397,964],[1396,1015],[1084,964],[1051,1016],[1402,964],[1050,964],[1403,1017],[1465,2],[1466,2],[1467,2],[1468,2],[1469,2],[1470,2],[1471,1018],[1472,1019],[1473,2],[1474,1020],[1475,1021],[1476,1022],[1477,1023],[1478,1023],[1479,1023],[1480,2],[1481,1023],[1482,2],[1483,2],[1484,2],[1485,2],[1486,1024],[1487,1023],[1488,1023],[1489,1024],[1490,1023],[1491,1023],[1492,2],[1493,2],[1494,2],[1495,1023],[1496,2],[1497,2],[1498,2],[1499,1025],[1500,1023],[1503,1026],[1504,1027],[1505,2],[1506,2],[1507,2],[1502,1028],[1508,2],[1509,2],[1510,1028],[1511,964],[1512,1029],[1513,964],[1514,964],[1515,2],[1516,2],[1517,1028],[1518,2],[1519,964],[1522,1030],[1521,1031],[1520,1026],[1523,1032],[1524,2],[1525,2],[1526,962],[1527,2],[1528,1033],[1529,1033],[1530,1034],[1531,2],[1532,2],[1533,964],[1534,2],[1501,2],[1535,1035],[1536,1036],[1537,1037],[1538,1038],[1539,1038],[1540,1037],[1541,1039],[1542,1039],[1543,2],[1544,1039],[1545,1039],[1546,1037],[1547,1040],[1548,1037],[1549,1039],[1550,1041],[1554,1039],[1555,1039],[1556,1039],[1557,1039],[1551,1039],[1552,1039],[1553,1039],[1558,1042],[1559,1037],[1560,1043],[1048,1044],[2611,1045],[2612,1045],[2613,1045],[2619,1046],[2614,1045],[2615,1045],[2616,1045],[2617,1045],[2618,1045],[2602,1047],[2601,2],[2620,1048],[2608,2],[2604,1049],[2595,2],[2594,2],[2596,2],[2597,1045],[2598,1050],[2610,1051],[2599,1045],[2600,1045],[2605,1052],[2606,1053],[2607,1045],[2603,2],[2609,2],[2282,2],[2284,1054],[2401,1055],[2405,1055],[2404,1055],[2402,1055],[2403,1055],[2406,1055],[2285,1055],[2297,1055],[2286,1055],[2299,1055],[2301,1055],[2294,1055],[2295,1055],[2296,1055],[2300,1055],[2302,1055],[2287,1055],[2298,1055],[2288,1055],[2290,1056],[2291,1055],[2292,1055],[2293,1055],[2309,1055],[2308,1055],[2409,1057],[2303,1055],[2305,1055],[2304,1055],[2306,1055],[2307,1055],[2408,1055],[2407,1055],[2310,1055],[2322,1058],[2323,1058],[2325,1055],[2370,1055],[2369,1055],[2390,1055],[2326,1058],[2367,1055],[2371,1055],[2327,1055],[2328,1055],[2329,1058],[2372,1055],[2366,1058],[2324,1058],[2373,1055],[2330,1058],[2374,1055],[2331,1058],[2354,1055],[2332,1055],[2375,1055],[2333,1055],[2364,1058],[2335,1055],[2336,1055],[2376,1055],[2338,1055],[2340,1055],[2341,1055],[2347,1055],[2348,1055],[2342,1058],[2378,1055],[2365,1058],[2377,1058],[2343,1055],[2344,1055],[2379,1055],[2345,1055],[2337,1058],[2380,1055],[2363,1055],[2381,1055],[2346,1058],[2349,1055],[2350,1055],[2368,1058],[2382,1055],[2383,1055],[2362,1059],[2339,1055],[2384,1058],[2385,1055],[2386,1055],[2387,1055],[2388,1058],[2351,1055],[2389,1055],[2353,1058],[2355,1055],[2352,1058],[2334,1055],[2356,1055],[2359,1055],[2357,1055],[2358,1055],[2311,1055],[2392,1055],[2391,1055],[2399,1055],[2393,1055],[2394,1055],[2396,1055],[2397,1055],[2395,1055],[2400,1055],[2398,1055],[2417,1060],[2415,1061],[2416,1062],[2414,1063],[2413,1055],[2412,1064],[2281,2],[2283,2],[2279,2],[2410,2],[2411,1065],[2289,1054],[2280,2],[4548,1066],[1575,2],[1574,2],[1580,1067],[1576,1068],[1579,1069],[1578,1070],[1577,2],[2783,2],[3218,2],[1640,2],[642,25],[2538,2],[2540,1071],[2539,1071],[2541,1072],[2547,2],[2554,1073],[2542,2],[2548,1074],[2545,1075],[2544,1076],[2552,1077],[2549,1078],[2550,1079],[2551,1080],[2553,1081],[1665,2],[4622,1082],[3383,1083],[3382,1084],[3380,2],[3381,2],[3386,1085],[3384,1086],[3385,2],[1405,1087],[3112,1088],[3108,1089],[3109,1090],[3110,1091],[3111,2],[2865,1092],[2864,1],[2867,1093],[2866,1092],[2639,1094],[2706,1095],[2705,1096],[2704,1097],[2644,1098],[2660,1099],[2658,1100],[2659,1101],[2645,1102],[2730,1103],[2630,2],[2632,2],[2633,1104],[2634,2],[2637,1105],[2640,2],[2657,1106],[2635,2],[2652,1107],[2638,1108],[2653,1109],[2656,1110],[2651,1111],[2654,1110],[2631,2],[2636,2],[2655,1112],[2661,1113],[2649,2],[2643,1114],[2641,1115],[2650,1116],[2647,1117],[2646,1117],[2642,1118],[2648,1119],[2662,1120],[2725,1121],[2719,1122],[2712,1123],[2711,1124],[2720,1125],[2721,1110],[2713,1126],[2726,1127],[2707,1128],[2708,1129],[2709,1130],[2729,1131],[2710,1124],[2714,1127],[2715,1132],[2728,1133],[2722,1134],[2723,1108],[2724,1132],[2716,1130],[2727,1110],[2717,1135],[2718,1136],[2663,1137],[2703,1138],[2667,1139],[2668,1139],[2669,1139],[2670,1139],[2671,1139],[2672,1139],[2673,1139],[2674,1139],[2693,1139],[2665,1139],[2675,1139],[2676,1139],[2677,1139],[2678,1139],[2679,1139],[2680,1139],[2700,1139],[2681,1139],[2682,1139],[2683,1139],[2698,1139],[2684,1139],[2699,1139],[2685,1139],[2696,1139],[2697,1139],[2686,1139],[2687,1139],[2688,1139],[2694,1139],[2695,1139],[2689,1139],[2690,1139],[2691,1139],[2692,1139],[2701,1139],[2702,1139],[2666,1140],[2664,1141],[2629,2],[3187,2],[4555,546],[3326,1142],[1625,1143],[1642,1144],[1644,1145],[1643,1146],[1626,885],[1641,1147],[1638,1148],[1639,1149],[1637,1150],[1630,1151],[1631,1152],[1633,1153],[1634,1154],[1632,1155],[1635,1156],[1645,1157],[1636,1158],[1628,1159],[1624,1160],[1629,1161],[1627,1143],[4620,1162],[4621,1163],[4616,2],[1689,2],[1047,2],[2361,1164],[2360,2],[2916,2],[4619,1165],[1590,2],[4547,1166],[52,2],[247,1167],[220,2],[198,1168],[196,1168],[111,1169],[62,1170],[61,1171],[197,1172],[182,1173],[104,1174],[60,1175],[59,1176],[246,1171],[211,1177],[210,1177],[122,1178],[218,1169],[219,1169],[221,1179],[222,1169],[223,1176],[224,1169],[195,1169],[225,1169],[226,1180],[227,1169],[228,1177],[229,1181],[230,1169],[231,1169],[232,1169],[233,1169],[234,1177],[235,1169],[236,1169],[237,1169],[238,1169],[239,1182],[240,1169],[241,1169],[242,1169],[243,1169],[244,1169],[64,1176],[65,1176],[66,1176],[67,1176],[68,1176],[69,1176],[70,1176],[71,1169],[73,1183],[74,1176],[72,1176],[75,1176],[76,1176],[77,1176],[78,1176],[79,1176],[80,1176],[81,1169],[82,1176],[83,1176],[84,1176],[85,1176],[86,1176],[87,1169],[88,1176],[89,1176],[90,1176],[91,1176],[92,1176],[93,1176],[94,1169],[96,1184],[95,1176],[97,1176],[98,1176],[99,1176],[100,1176],[101,1182],[102,1169],[103,1169],[117,1185],[105,1186],[106,1176],[107,1176],[108,1169],[109,1176],[110,1176],[112,1187],[113,1176],[114,1176],[115,1176],[116,1176],[118,1176],[119,1176],[120,1176],[121,1176],[123,1188],[124,1176],[125,1176],[126,1176],[127,1169],[128,1176],[129,1189],[130,1189],[131,1189],[132,1169],[133,1176],[134,1176],[135,1176],[140,1176],[136,1176],[137,1169],[138,1176],[139,1169],[141,1176],[142,1176],[143,1176],[144,1176],[145,1176],[146,1176],[147,1169],[148,1176],[149,1176],[150,1176],[151,1176],[152,1176],[153,1176],[154,1176],[155,1176],[156,1176],[157,1176],[158,1176],[159,1176],[160,1176],[161,1176],[162,1176],[163,1176],[164,1190],[165,1176],[166,1176],[167,1176],[168,1176],[169,1176],[170,1176],[171,1169],[172,1169],[173,1169],[174,1169],[175,1169],[176,1176],[177,1176],[178,1176],[179,1176],[245,1169],[181,1191],[204,1192],[199,1192],[190,1193],[188,1194],[202,1195],[191,1196],[205,1197],[200,1198],[201,1195],[203,1199],[189,2],[194,2],[186,1200],[187,1201],[184,2],[185,1202],[183,1176],[192,1203],[63,1204],[212,2],[213,2],[214,2],[215,2],[216,2],[217,2],[206,2],[209,1177],[208,2],[207,1205],[180,1206],[193,1207],[2243,885],[2560,1208],[2559,1209],[2561,1210],[2556,1211],[2563,1212],[2558,1213],[2566,1214],[2565,1215],[2562,1216],[2564,1217],[2557,1209],[4169,1218],[4042,1219],[4170,1220],[4043,1221],[4171,1222],[4044,1223],[4172,1224],[4045,1225],[4173,1226],[4046,1227],[4174,1228],[4175,1229],[4047,1230],[4176,1231],[4048,1232],[4177,1233],[4178,1234],[4049,1235],[4179,1236],[4180,1237],[4181,1238],[4050,1239],[4182,1240],[4051,1241],[4183,1242],[4052,1243],[4184,1244],[4053,1245],[4185,1246],[4054,1247],[4186,1248],[4187,1249],[4055,1250],[4188,1251],[4056,1252],[4189,1253],[4057,1254],[4190,1255],[4058,1256],[4191,1257],[4192,1258],[4193,1259],[4194,1260],[4059,1261],[4195,1262],[4060,1263],[4196,1264],[4061,1265],[4197,1266],[4062,1267],[4198,1268],[4063,1269],[4199,1270],[4064,1271],[4200,1272],[4201,1273],[4065,1274],[4202,1275],[4066,1276],[4203,1277],[4204,1278],[4067,1279],[4205,1280],[4206,1281],[4207,1282],[4068,1283],[4208,1284],[4209,1285],[4069,1286],[4210,1287],[4211,1288],[4070,1289],[4212,1290],[4213,1291],[4071,1292],[4214,1293],[4072,1294],[4215,1295],[4073,1296],[4036,1297],[4039,1298],[4216,1299],[4074,1300],[4217,1301],[4075,1302],[4218,1303],[4219,1304],[4220,1305],[4076,1306],[4221,1307],[4077,1308],[4222,1309],[4223,1310],[4224,1311],[4078,1312],[4225,1313],[4079,1314],[4226,1315],[4080,1316],[4227,1317],[4081,1318],[4228,1319],[4229,1320],[4082,1321],[4230,1322],[4083,1323],[4231,1324],[4084,1325],[4232,1326],[4233,1327],[4085,1328],[4234,1329],[4086,1330],[4235,1331],[4087,1332],[4236,1333],[4088,1334],[4237,1335],[4089,1336],[4238,1337],[4090,1338],[4239,1339],[4091,1340],[4240,1341],[4092,1342],[4241,1343],[4093,1344],[4242,1345],[4094,1346],[4243,1347],[4095,1348],[4244,1349],[4245,1350],[4246,1351],[4096,1352],[4037,1353],[4247,1354],[4097,1355],[4248,1356],[4098,1357],[4249,1358],[4099,1359],[4250,1360],[4100,1361],[4251,1362],[4101,1363],[4252,1364],[4102,1365],[4253,1366],[4254,1367],[4103,1368],[4255,1369],[4104,1370],[4256,1371],[4257,1372],[4105,1373],[4258,1374],[4106,1375],[4259,1376],[4107,1377],[4260,1378],[4108,1379],[4261,1380],[4109,1381],[4262,1382],[4110,1383],[4263,1384],[4111,1385],[4264,1386],[4112,1387],[4265,1388],[4113,1389],[4266,1390],[4267,1391],[4114,1392],[4268,1393],[4115,1394],[4269,1395],[4116,1396],[4270,1397],[4117,1398],[4271,1399],[4118,1400],[4272,1401],[4273,1402],[4274,1403],[4119,1404],[4275,1405],[4120,1406],[4276,1407],[4121,1408],[4277,1409],[4122,1410],[4278,1411],[4279,1412],[4123,1413],[4280,1414],[4124,1415],[4281,1416],[4125,1417],[4282,1418],[4283,1419],[4126,1420],[4284,1421],[4127,1422],[4285,1423],[4286,1424],[4128,1425],[4287,1426],[4129,1427],[4288,1428],[4130,1429],[4289,1430],[4131,1431],[4290,1432],[4132,1433],[4291,1434],[4133,1435],[4134,1436],[4135,1437],[4136,1438],[4137,1439],[4138,1440],[4139,1441],[4140,1442],[4141,1443],[4292,1444],[4142,1445],[4143,1446],[4144,1447],[4145,1448],[4146,1449],[4147,1450],[4041,1451],[4293,1452],[4148,1453],[4294,1454],[4149,1455],[4295,1456],[4296,1457],[4150,1458],[4297,1459],[4151,1460],[4298,1461],[4152,1462],[4299,1463],[4300,1464],[4153,1465],[4301,1466],[4154,1467],[4302,1468],[4155,1469],[4303,1470],[4156,1471],[4304,1472],[4157,1473],[4305,1474],[4158,1475],[4306,1476],[4159,1477],[4307,1478],[4160,1479],[4040,1480],[4308,1481],[4161,1482],[4162,1483],[4309,1484],[4163,1485],[4310,1486],[4164,1487],[4165,1488],[4167,1489],[4311,1490],[4166,1491],[4312,1492],[4313,1493],[4168,1494],[4038,1495],[4033,1496],[4314,1497],[4032,1498],[4034,1499],[4035,1500],[2869,1501],[2873,1502],[2871,2],[2872,2],[2870,1503],[2868,2],[720,1504],[719,2],[741,2],[659,1505],[721,2],[668,2],[658,2],[785,2],[872,2],[822,1506],[1028,1507],[869,1508],[1027,1509],[1026,1509],[871,2],[722,1510],[829,1511],[825,1512],[1023,1508],[993,2],[996,1513],[994,2],[990,1514],[995,2],[989,1515],[992,1516],[943,1517],[944,1518],[945,1518],[957,1518],[950,1519],[949,1520],[951,1518],[952,1518],[956,1521],[954,1522],[984,1523],[981,2],[980,1524],[982,1518],[958,2],[959,2],[962,2],[960,2],[961,2],[963,2],[964,2],[967,2],[965,2],[966,2],[968,2],[969,2],[664,1525],[937,2],[938,2],[939,2],[940,2],[665,1526],[941,2],[942,2],[971,1527],[696,1528],[970,2],[699,2],[700,1529],[701,1529],[948,1530],[946,1530],[947,2],[656,1528],[695,1531],[991,1532],[663,2],[955,1525],[983,586],[953,1533],[972,1529],[973,1534],[974,1535],[975,1535],[976,1535],[977,1535],[978,1536],[979,1536],[988,1537],[987,2],[985,2],[986,1538],[727,1539],[689,2],[690,1540],[815,2],[816,1541],[819,1506],[820,1506],[821,1506],[790,1542],[791,1543],[810,1506],[814,1506],[809,1544],[769,1545],[732,2],[733,1546],[792,2],[793,1547],[813,1506],[807,2],[808,1548],[794,1542],[795,1549],[812,1506],[817,2],[818,1550],[823,2],[824,1551],[796,1506],[811,1506],[1025,2],[805,1552],[806,1553],[798,2],[799,2],[800,2],[801,2],[802,2],[803,2],[797,2],[804,2],[662,2],[687,2],[692,2],[718,2],[694,2],[780,2],[688,1530],[723,2],[726,2],[786,1554],[775,1555],[826,1556],[715,1557],[706,2],[697,1558],[698,1559],[1032,1513],[707,2],[710,1558],[693,2],[708,1518],[714,1560],[709,1536],[702,1561],[705,1532],[875,1562],[898,1562],[879,1562],[882,1563],[884,1562],[933,1562],[910,1562],[874,1562],[902,1562],[930,1562],[881,1562],[911,1562],[896,1562],[899,1562],[887,1562],[920,1564],[916,1562],[909,1562],[891,1565],[890,1565],[907,1563],[917,1562],[935,1566],[921,1567],[913,1562],[894,1562],[880,1562],[883,1562],[915,1562],[900,1563],[908,1562],[905,1568],[922,1568],[906,1563],[892,1562],[901,1562],[934,1562],[924,1562],[912,1562],[932,1562],[914,1562],[893,1562],[928,1562],[918,1562],[895,1562],[923,1562],[931,1562],[897,1562],[919,1565],[903,1562],[927,1569],[878,1569],[889,1562],[888,1562],[886,1570],[873,2],[885,1562],[929,1568],[925,1568],[904,1568],[926,1568],[936,1571],[734,1572],[740,1573],[739,1574],[730,1575],[729,2],[738,1576],[737,1576],[736,1576],[1016,1577],[735,1578],[777,2],[728,2],[703,2],[745,1579],[744,1580],[997,1572],[999,1572],[1000,1572],[1001,1572],[1002,1572],[1003,1572],[1004,1581],[1009,1572],[1005,1572],[1006,1572],[1015,1572],[1007,1572],[1008,1572],[1010,1572],[1011,1572],[1012,1572],[1013,1572],[998,1572],[1014,1582],[870,1583],[1037,1584],[1017,1585],[1018,1586],[1021,1587],[1019,1586],[716,1588],[717,1589],[1020,1586],[762,2],[667,1590],[862,2],[676,2],[681,1591],[863,1592],[860,2],[766,2],[867,1593],[866,2],[832,2],[861,1518],[858,2],[859,1594],[868,1595],[857,2],[856,1536],[677,1536],[661,1596],[830,1597],[864,2],[865,2],[713,1537],[683,1532],[666,2],[763,1598],[686,1599],[685,1600],[682,1601],[831,1602],[767,1603],[674,1604],[833,1605],[679,1606],[678,1607],[675,1608],[684,1609],[712,1610],[653,2],[680,2],[654,2],[655,2],[657,2],[660,1592],[652,2],[704,2],[711,2],[789,1611],[1029,1612],[788,1588],[1030,1613],[1031,1614],[673,1615],[877,1616],[876,1617],[731,1618],[840,1619],[782,1620],[849,1621],[783,1622],[851,1623],[841,1624],[853,1625],[854,1626],[839,2],[847,1627],[770,1628],[828,1629],[827,1629],[852,1630],[844,2],[855,1631],[845,2],[850,1632],[848,1633],[846,2],[781,1634],[843,1635],[842,1635],[774,1636],[772,1637],[773,1637],[779,1638],[771,2],[838,1639],[1022,1640],[1024,1641],[1035,2],[776,1642],[743,2],[787,1643],[742,2],[778,1644],[784,1645],[1033,1646],[691,1647],[761,2],[669,2],[765,2],[724,2],[834,2],[836,1648],[746,2],[671,586],[837,1649],[764,1650],[670,1651],[768,1652],[725,1653],[835,1654],[747,1655],[672,1656],[760,1657],[748,2],[759,1658],[754,1659],[755,1660],[758,1556],[757,1661],[753,1660],[756,1661],[749,1556],[750,1556],[751,1556],[752,1662],[1034,1663],[1036,1664],[50,2],[51,2],[9,2],[11,2],[10,2],[2,2],[12,2],[13,2],[14,2],[15,2],[16,2],[17,2],[18,2],[19,2],[3,2],[20,2],[21,2],[4,2],[22,2],[26,2],[23,2],[24,2],[25,2],[27,2],[28,2],[29,2],[5,2],[30,2],[31,2],[32,2],[33,2],[6,2],[37,2],[34,2],[35,2],[36,2],[38,2],[7,2],[39,2],[44,2],[45,2],[40,2],[41,2],[42,2],[43,2],[8,2],[49,2],[46,2],[47,2],[48,2],[1,2],[567,1665],[577,1666],[566,1665],[587,1667],[558,1668],[557,1669],[586,25],[580,1670],[585,1671],[560,1672],[574,1673],[559,1674],[583,1675],[555,1676],[554,25],[584,1677],[556,1678],[561,1679],[562,2],[565,1679],[552,2],[588,1680],[578,1681],[569,1682],[570,1683],[572,1684],[568,1685],[571,1686],[581,25],[563,1687],[564,1688],[573,1689],[553,810],[576,1681],[575,1679],[579,2],[582,1690],[3365,1691],[3343,1692],[3353,1693],[3342,1692],[3363,1694],[3334,1695],[3333,1669],[3362,25],[3356,1696],[3361,1697],[3336,1698],[3350,1699],[3335,1700],[3359,1701],[3331,1702],[3330,25],[3360,1703],[3332,1704],[3337,1705],[3338,2],[3341,1705],[3328,2],[3364,1706],[3354,1707],[3345,1708],[3346,1709],[3348,1710],[3344,1711],[3347,1712],[3357,25],[3339,1713],[3340,1714],[3349,1715],[3329,810],[3352,1716],[3351,1705],[3355,2],[3358,1717],[2265,1718],[2249,2],[2250,2],[2252,1719],[2253,2],[2251,2],[2254,1719],[2255,1719],[2257,1720],[2256,1719],[2258,1719],[2259,1720],[2260,1719],[2261,2],[2262,1719],[2263,2],[2264,2],[3247,1721],[3248,1722],[3242,1723],[3244,1724],[3245,1725],[3240,1726],[3239,1727],[3241,1728],[3238,1729],[3243,1730],[3246,1731],[439,1732],[4351,1733],[438,258],[3153,1734],[3155,1735],[3152,1736],[4558,1737],[4559,1738],[3148,1739],[3149,1740],[3147,1741],[3146,1742],[3144,2],[3145,2],[3150,258],[3154,1743],[3151,1744],[3201,1745],[3191,1746],[3204,1747],[3192,1748],[4560,1749],[3190,2],[3202,1750],[3203,1751],[4561,1752],[3194,1753],[3205,1754],[3193,1755],[3127,258],[2575,1756],[3165,1757],[1703,1758],[2576,1759],[3169,1760],[4562,1761],[3197,1762],[2505,1763],[2507,1764],[2420,1765],[2498,1766],[2418,1767],[2500,1768],[2503,1737],[2504,1737],[2276,1769],[2278,1770],[2273,2],[2274,2],[2277,2],[2275,2],[2499,1771],[2419,1772],[2502,1773],[2506,1774],[2501,1775],[3314,1776],[3315,1777],[3124,2],[3316,1778],[3125,1779],[4563,258],[4564,1780],[3135,1781],[3134,1782],[3313,1783],[3812,1784],[3816,1785],[4024,1786],[3814,1787],[3815,1737],[1760,586],[3813,1788],[3810,1789],[4023,1790],[3811,1791],[2577,1792],[2578,1793],[2512,1794],[2537,2],[2508,1795],[2569,1796],[2511,258],[2510,1795],[2509,1795],[4565,1797],[3119,2],[1702,2],[1734,2],[1733,2],[4346,1798],[4347,1799],[4344,1800],[1614,1801],[4566,1802],[4567,1802],[4568,1802],[4328,588],[1611,588],[1612,1802],[1613,1803],[2589,258],[4569,1804],[3120,1805],[4349,1806],[4570,1807],[3128,1808],[2590,1809],[1726,1810],[2567,1811],[4571,2],[1731,1812],[4572,1813],[4551,1814],[4553,1815],[1609,1816],[4552,1817],[1730,1818],[4575,1819],[4574,1820],[4573,1821],[1728,258],[4345,1822],[1732,1823],[4577,1824],[4576,258],[1719,258],[1729,1788],[4348,1825],[4350,1826],[2240,2],[4332,1827],[1725,1828],[3784,1829],[3121,1830],[2624,2],[3122,2],[3118,1756],[2568,1831],[4578,1832],[3113,1833],[4554,1834],[4556,599],[3600,1835],[1690,1836],[1727,2],[4579,1837],[3142,1838],[3143,1839],[3136,1840],[3133,1841],[3131,1737],[3139,1737],[3137,1737],[3132,1842],[3141,1843],[2571,1844],[2573,1845],[2572,1846],[2570,1847],[4580,1848],[3140,1849],[3138,1850],[2584,1851],[2585,1852],[2579,1795],[2582,1795],[2583,1795],[2580,1795],[2581,1795],[3645,1853],[3623,1854],[3644,1855],[3624,1856],[3641,1857],[3637,1858],[3642,1737],[3622,1859],[3635,1860],[3625,1737],[3614,1861],[3617,1737],[4581,1862],[3627,1737],[3636,1863],[3626,1864],[3615,1865],[3618,1866],[3639,1867],[3649,1868],[3628,1869],[3603,1870],[3605,1871],[3632,1872],[3630,1873],[3634,1874],[3607,1875],[3609,1876],[3621,1877],[3647,1878],[3602,1878],[4582,1879],[3612,1880],[3613,1881],[3604,2],[3631,2],[3608,2],[3620,2],[4583,1882],[3606,2],[3611,2],[3610,2],[3629,2],[3633,2],[3648,1883],[3643,1884],[3616,1885],[3638,1886],[3619,1887],[3640,1888],[3646,1889],[3318,1890],[3322,1891],[3320,1892],[3323,1893],[3317,2],[3319,1890],[3321,1890],[3657,1894],[3659,1895],[3652,586],[3654,586],[3650,1896],[3653,1897],[3655,1898],[3651,1896],[3661,1899],[3662,1900],[3660,1901],[3658,1902],[3656,1903],[4584,1904],[4585,1904],[3221,1905],[3212,1906],[3210,1906],[4586,1907],[3209,1908],[4587,1909],[3217,1910],[3216,1911],[4588,1912],[3213,1913],[3211,1914],[3214,1915],[3215,1916],[3219,1917],[3220,473],[3208,1918],[3206,1919],[4589,1920],[3207,1921],[1720,1922],[1721,1923],[1718,1924],[4550,2],[3667,1925],[3668,1926],[3666,1927],[3665,258],[3664,258],[3663,258],[4334,1737],[4335,1737],[4336,1737],[4337,1859],[4338,1859],[4339,1737],[4330,586],[4331,1928],[4333,1929],[4340,2],[4342,1930],[4343,1931],[4341,1932],[4557,1933],[2268,1934],[2272,1935],[2266,1936],[2269,1937],[2247,1938],[2242,1939],[2271,1940],[2241,1941],[2244,2],[2245,1942],[2246,1943],[2248,1944],[2267,1945],[4594,1946],[4593,1947],[4592,1948],[4595,1949],[4591,1950],[4590,1951],[3223,1952],[3222,586],[3233,1953],[3236,1952],[3235,1954],[3237,1955],[3231,1956],[3234,1957],[3232,1958],[3225,1952],[3226,1952],[3227,1952],[3228,1952],[3229,1952],[3230,1952],[4596,1952],[3224,1952],[3774,1959],[3773,586],[3769,1960],[3771,1961],[3770,586],[3776,1962],[3775,1963],[3772,258],[1602,258],[1610,1964],[1591,1965],[1604,1966],[1608,1967],[1607,1968],[1606,258],[1605,1969],[1603,1970],[3168,1971],[3199,1972],[3188,1973],[3167,1896],[3166,1896],[3171,1974],[3198,1975],[3170,1976],[3200,1977],[3189,1978],[3172,1979],[3301,1980],[3249,2],[3251,1981],[3250,1982],[3300,1981],[3303,1983],[3304,1984],[3302,1985],[3299,1986],[3783,258],[3788,1987],[3787,1988],[3789,1989],[3785,1990],[3786,258],[4027,1991],[4031,1992],[4030,1993],[4606,1994],[4029,1995],[4025,1846],[4028,1996],[4026,1896],[4317,2],[4319,1997],[4329,1998],[4318,1999],[4607,2],[4316,2000],[4315,1788],[4327,2001],[4326,258],[4610,2002],[4320,586],[4608,2],[4609,2003],[4324,2004],[4321,2005],[4325,2006],[4323,2007],[4322,2008],[2621,2009],[2586,2],[2587,2010],[2593,2011],[2591,2012],[2625,2013],[2622,2014],[2626,2015],[2588,2016],[2623,2017],[2592,2018],[3309,1737],[3310,2019],[4613,2],[4612,2020],[3311,2021],[3307,2022],[3305,1988],[4611,2023],[3306,258],[3312,2024],[3308,2025],[3595,2026],[3596,2026],[3599,2026],[3598,2027],[3601,2028],[3597,2029],[3794,258],[3793,258],[3791,2030],[3795,2031],[3790,2032],[3792,258],[1722,2],[1724,2033],[1723,2034],[1737,2035],[1735,258],[1736,2036],[1738,2037],[1740,2038],[1739,2039],[3804,2040],[3801,2041],[3802,2042],[4614,258],[3807,258],[3800,2043],[3798,2044],[3797,2044],[3799,2044],[3796,586],[3808,2045],[3806,2046],[3805,2047],[3809,2048],[3803,2049],[3114,1757],[3123,2050],[3115,2051],[2574,2052],[3129,2053],[3130,2054],[3126,2055]],"semanticDiagnosticsPerFile":[[1605,[{"start":1312,"length":6,"code":2339,"category":1,"messageText":"Property 'values' does not exist on type 'unknown'."}]],[1606,[{"start":315,"length":7,"code":2339,"category":1,"messageText":"Property 'cpuLoad' does not exist on type 'unknown'."},{"start":341,"length":7,"code":2339,"category":1,"messageText":"Property 'cpuLoad' does not exist on type 'unknown'."},{"start":519,"length":11,"code":2339,"category":1,"messageText":"Property 'memoryUsage' does not exist on type 'unknown'."},{"start":549,"length":11,"code":2339,"category":1,"messageText":"Property 'memoryUsage' does not exist on type 'unknown'."},{"start":717,"length":11,"code":2339,"category":1,"messageText":"Property 'slowQueries' does not exist on type 'unknown'."},{"start":747,"length":11,"code":2339,"category":1,"messageText":"Property 'slowQueries' does not exist on type 'unknown'."},{"start":833,"length":11,"code":2339,"category":1,"messageText":"Property 'slowQueries' does not exist on type 'unknown'."},{"start":961,"length":11,"code":2339,"category":1,"messageText":"Property 'slowQueries' does not exist on type 'unknown'."},{"start":1076,"length":5,"code":2339,"category":1,"messageText":"Property 'table' does not exist on type 'unknown'."},{"start":1107,"length":8,"code":2339,"category":1,"messageText":"Property 'duration' does not exist on type 'unknown'."}]],[1607,[{"start":887,"length":7,"code":2339,"category":1,"messageText":"Property 'cpuLoad' does not exist on type 'unknown'."},{"start":985,"length":7,"code":2339,"category":1,"messageText":"Property 'cpuLoad' does not exist on type 'unknown'."},{"start":1051,"length":11,"code":2339,"category":1,"messageText":"Property 'memoryUsage' does not exist on type 'unknown'."},{"start":1160,"length":11,"code":2339,"category":1,"messageText":"Property 'memoryUsage' does not exist on type 'unknown'."}]],[1718,[{"start":9624,"length":4,"code":2339,"category":1,"messageText":"Property 'ping' does not exist on type 'unknown'."}]],[1735,[{"start":1984,"length":13,"messageText":"Spread types may only be created from object types.","category":1,"code":2698}]],[1736,[{"start":1873,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":1888,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":1942,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":1959,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":2046,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2079,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2137,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2155,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2235,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2257,"length":7,"code":2339,"category":1,"messageText":"Property 'version' does not exist on type 'unknown'."},{"start":2937,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":3039,"length":16,"messageText":"Spread types may only be created from object types.","category":1,"code":2698}]],[1737,[{"start":1562,"length":5,"code":2339,"category":1,"messageText":"Property 'store' does not exist on type 'unknown'."},{"start":1603,"length":4,"code":2339,"category":1,"messageText":"Property 'keys' does not exist on type 'unknown'."},{"start":1663,"length":4,"code":2339,"category":1,"messageText":"Property 'keys' does not exist on type 'unknown'."},{"start":2637,"length":5,"code":2339,"category":1,"messageText":"Property 'reset' does not exist on type 'unknown'."},{"start":2710,"length":5,"code":2339,"category":1,"messageText":"Property 'reset' does not exist on type 'unknown'."}]],[2268,[{"start":2567,"length":21,"code":2339,"category":1,"messageText":"Property 'uploadValidationError' does not exist on type 'unknown'."},{"start":2675,"length":21,"code":2339,"category":1,"messageText":"Property 'uploadValidationError' does not exist on type 'unknown'."},{"start":2920,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."},{"start":5035,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[2269,[{"start":1144,"length":4,"code":2339,"category":1,"messageText":"Property 'text' does not exist on type 'unknown'."},{"start":1649,"length":10,"code":2339,"category":1,"messageText":"Property 'statusCode' does not exist on type 'unknown'."},{"start":1735,"length":10,"code":2339,"category":1,"messageText":"Property 'statusCode' does not exist on type 'unknown'."},{"start":1806,"length":2,"code":2339,"category":1,"messageText":"Property 'on' does not exist on type 'unknown'."},{"start":1871,"length":2,"code":2339,"category":1,"messageText":"Property 'on' does not exist on type 'unknown'."}]],[2271,[{"start":2832,"length":6,"code":2740,"category":1,"messageText":"Type '{}' is missing the following properties from type 'ReadableStream': readable, read, setEncoding, pause, and 22 more.","relatedInformation":[{"file":"./src/common/types/file.types.ts","start":353,"length":6,"messageText":"The expected type comes from property 'stream' which is declared here on type 'UploadedFile'","category":3,"code":6500}],"canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type 'ReadableStream'."}},{"start":2954,"length":19,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'UploadedFile'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'UploadedFile': fieldname, originalname, encoding, mimetype, and 2 more.","category":1,"code":2740}]}},{"start":3335,"length":11,"code":2339,"category":1,"messageText":"Property 'hlsManifest' does not exist on type 'unknown'."},{"start":3652,"length":11,"code":2322,"category":1,"messageText":"Type 'unknown' is not assignable to type 'ContentStatus'."},{"start":4581,"length":10,"code":2339,"category":1,"messageText":"Property 'statusCode' does not exist on type 'unknown'."},{"start":4667,"length":10,"code":2339,"category":1,"messageText":"Property 'statusCode' does not exist on type 'unknown'."},{"start":4699,"length":4,"code":2339,"category":1,"messageText":"Property 'pipe' does not exist on type 'unknown'."}]],[2420,[{"start":8098,"length":6,"code":2339,"category":1,"messageText":"Property 'region' does not exist on type 'unknown'."}]],[2501,[{"start":5010,"length":20,"code":2339,"category":1,"messageText":"Property 'connectionSuccessful' does not exist on type 'unknown'."}]],[2506,[{"start":5499,"length":3,"code":2339,"category":1,"messageText":"Property 'add' does not exist on type 'unknown'."}]],[2510,[{"start":6148,"length":4,"code":2339,"category":1,"messageText":"Property 'type' does not exist on type 'unknown'."},{"start":6182,"length":4,"code":2339,"category":1,"messageText":"Property 'type' does not exist on type 'unknown'."},{"start":6217,"length":9,"code":2339,"category":1,"messageText":"Property 'operation' does not exist on type 'unknown'."},{"start":6256,"length":9,"code":2339,"category":1,"messageText":"Property 'operation' does not exist on type 'unknown'."},{"start":6891,"length":8,"code":2339,"category":1,"messageText":"Property 'newValue' does not exist on type 'unknown'."},{"start":6943,"length":8,"code":2339,"category":1,"messageText":"Property 'newValue' does not exist on type 'unknown'."},{"start":6982,"length":4,"code":2339,"category":1,"messageText":"Property 'data' does not exist on type 'unknown'."},{"start":7030,"length":4,"code":2339,"category":1,"messageText":"Property 'data' does not exist on type 'unknown'."}]],[2512,[{"start":1945,"length":9,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId, position","category":1,"code":2739}]}},{"start":2099,"length":9,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId","category":1,"code":2739}]}}]],[2567,[{"start":471,"length":13,"code":2339,"category":1,"messageText":"Property 'remoteAddress' does not exist on type 'unknown'."},{"start":501,"length":10,"code":2339,"category":1,"messageText":"Property 'connection' does not exist on type 'unknown'."},{"start":562,"length":3,"code":2339,"category":1,"messageText":"Property 'sub' does not exist on type 'unknown'."},{"start":575,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."}]],[2568,[{"start":550,"length":7,"code":2339,"category":1,"messageText":"Property 'sockets' does not exist on type 'unknown'."}]],[2569,[{"start":4024,"length":21,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type '\"document\" | \"whiteboard\"'."},{"start":4349,"length":23,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type '\"document\" | \"whiteboard\"'."},{"start":6064,"length":6,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId, position","category":1,"code":2739}]}},{"start":6229,"length":6,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId","category":1,"code":2739}]}},{"start":9115,"length":10,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown[]' is not assignable to parameter of type 'DocumentOperation[]'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'DocumentOperation': id, type, position, timestamp, userId","category":1,"code":2739,"canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type 'DocumentOperation'."}}]}},{"start":9278,"length":10,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown[]' is not assignable to parameter of type 'WhiteboardOperation[]'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'WhiteboardOperation': id, type, userId, timestamp","category":1,"code":2739,"canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type 'WhiteboardOperation'."}}]}}]],[2577,[{"start":3597,"length":9,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId, position","category":1,"code":2739}]}},{"start":4368,"length":9,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': type, userId","category":1,"code":2739}]}}]],[2579,[{"start":5842,"length":7,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":6317,"length":7,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":6590,"length":7,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":10647,"length":7,"code":2339,"category":1,"messageText":"Property 'mapping' does not exist on type 'unknown'."},{"start":10665,"length":7,"code":2339,"category":1,"messageText":"Property 'mapping' does not exist on type 'unknown'."},{"start":10723,"length":7,"code":2339,"category":1,"messageText":"Property 'mapping' does not exist on type 'unknown'."},{"start":10864,"length":8,"code":2339,"category":1,"messageText":"Property 'operator' does not exist on type 'unknown'."},{"start":10925,"length":5,"code":2339,"category":1,"messageText":"Property 'value' does not exist on type 'unknown'."},{"start":10966,"length":8,"code":2339,"category":1,"messageText":"Property 'operator' does not exist on type 'unknown'."},{"start":11030,"length":5,"code":2339,"category":1,"messageText":"Property 'value' does not exist on type 'unknown'."},{"start":11071,"length":8,"code":2339,"category":1,"messageText":"Property 'operator' does not exist on type 'unknown'."},{"start":11132,"length":5,"code":2339,"category":1,"messageText":"Property 'value' does not exist on type 'unknown'."},{"start":11267,"length":9,"code":2339,"category":1,"messageText":"Property 'operation' does not exist on type 'unknown'."},{"start":11322,"length":6,"code":2339,"category":1,"messageText":"Property 'field1' does not exist on type 'unknown'."},{"start":11351,"length":6,"code":2339,"category":1,"messageText":"Property 'field2' does not exist on type 'unknown'."},{"start":11400,"length":9,"code":2339,"category":1,"messageText":"Property 'operation' does not exist on type 'unknown'."},{"start":11460,"length":6,"code":2339,"category":1,"messageText":"Property 'field1' does not exist on type 'unknown'."},{"start":11489,"length":6,"code":2339,"category":1,"messageText":"Property 'field2' does not exist on type 'unknown'."},{"start":11541,"length":6,"code":2339,"category":1,"messageText":"Property 'field1' does not exist on type 'unknown'."},{"start":11647,"length":6,"code":2339,"category":1,"messageText":"Property 'format' does not exist on type 'unknown'."},{"start":11696,"length":5,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 4, '(value: string | number | Date): Date', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string | number | Date'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 4, '(value: string | number): Date', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string | number'.","category":1,"code":2345}]}]},"relatedInformation":[]},{"start":11752,"length":6,"code":2339,"category":1,"messageText":"Property 'format' does not exist on type 'unknown'."},{"start":11860,"length":6,"code":2339,"category":1,"messageText":"Property 'format' does not exist on type 'unknown'."}]],[2581,[{"start":15792,"length":5,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 4, '(value: string | number | Date): Date', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string | number | Date'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 4, '(value: string | number): Date', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string | number'.","category":1,"code":2345}]}]},"relatedInformation":[]}]],[2584,[{"start":1212,"length":6,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'ETLConfig'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'ETLConfig': sourceConnection, targetConnection, transformations","category":1,"code":2739}]}},{"start":1998,"length":14,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, name, measures, foreignKeys, granularity","category":1,"code":2739}]}},{"start":2407,"length":14,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, name, measures, foreignKeys, granularity","category":1,"code":2739}]}},{"start":3084,"length":11,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, metrics, name, query, and 4 more.","category":1,"code":2740}]}},{"start":3392,"length":15,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type '{ [key: string]: unknown; }'.","category":1,"code":2345,"next":[{"messageText":"Index signature for type 'string' is missing in type '{}'.","category":1,"code":2329}]}},{"start":3674,"length":13,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, name, rules","category":1,"code":2739}]}},{"start":5218,"length":11,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, name","category":1,"code":2739}]}},{"start":5904,"length":10,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, type, name, metadata, system","category":1,"code":2739}]}},{"start":6185,"length":10,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': description, metadata, sourceId, targetId","category":1,"code":2739}]}},{"start":7274,"length":11,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Omit'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Omit': retryDelay, loadType, batchSize, maxRetries, and 2 more.","category":1,"code":2740}]}}]],[2622,[{"start":4732,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":6805,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":7275,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":7767,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":8543,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":8976,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":9644,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":10121,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":10437,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":10693,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":11038,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}},{"start":11558,"length":14,"code":1241,"category":1,"messageText":{"messageText":"Unable to resolve signature of method decorator when called as an expression.","category":1,"code":1241,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}}]],[3136,[{"start":1412,"length":18,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":1478,"length":12,"code":2339,"category":1,"messageText":"Property 'instructorId' does not exist on type 'unknown'."},{"start":3158,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'CursorPaginatedResponse>' is not assignable to type 'CursorPaginatedResponse'.","category":1,"code":2322,"next":[{"messageText":"Type 'Record' is missing the following properties from type 'Course': id, title, description, price, and 8 more.","category":1,"code":2740,"canonicalHead":{"code":2322,"messageText":"Type 'Record' is not assignable to type 'Course'."}}]}},{"start":4279,"length":5,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'SelectQueryBuilder' is not assignable to parameter of type 'SelectQueryBuilder>'.","category":1,"code":2345,"next":[{"messageText":"The types returned by 'getRawAndEntities()' are incompatible between these types.","category":1,"code":2201,"next":[{"messageText":"Type 'Promise<{ entities: Course[]; raw: any[]; }>' is not assignable to type 'Promise<{ entities: Record[]; raw: any[]; }>'.","category":1,"code":2322,"next":[{"messageText":"Type '{ entities: Course[]; raw: any[]; }' is not assignable to type '{ entities: Record[]; raw: any[]; }'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'entities' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Course[]' is not assignable to type 'Record[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Course' is not assignable to type 'Record'.","category":1,"code":2322,"next":[{"messageText":"Index signature for type 'string' is missing in type 'Course'.","category":1,"code":2329}]}],"canonicalHead":{"code":2322,"messageText":"Type '{ entities: Course[]; raw: any[]; }' is not assignable to type '{ entities: Record[]; raw: any[]; }'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type '() => Promise<{ entities: Course[]; raw: T[]; }>' is not assignable to type '() => Promise<{ entities: Record[]; raw: T[]; }>'."}}]}]}}]],[3151,[{"start":609,"length":6,"code":2339,"category":1,"messageText":"Property 'passed' does not exist on type 'unknown'."}]],[3152,[{"start":3629,"length":10,"code":2339,"category":1,"messageText":"Property 'questionId' does not exist on type 'unknown'."},{"start":3658,"length":8,"code":2339,"category":1,"messageText":"Property 'response' does not exist on type 'unknown'."}]],[3153,[{"start":512,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[3169,[{"start":872,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."},{"start":2127,"length":3,"code":2339,"category":1,"messageText":"Property 'kid' does not exist on type 'unknown'."}]],[3170,[{"start":2125,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[3171,[{"start":690,"length":10,"code":2339,"category":1,"messageText":"Property 'courseName' does not exist on type 'unknown'."},{"start":906,"length":6,"code":2339,"category":1,"messageText":"Property 'amount' does not exist on type 'unknown'."},{"start":921,"length":8,"code":2339,"category":1,"messageText":"Property 'currency' does not exist on type 'unknown'."},{"start":953,"length":10,"code":2339,"category":1,"messageText":"Property 'courseName' does not exist on type 'unknown'."},{"start":1172,"length":10,"code":2339,"category":1,"messageText":"Property 'senderName' does not exist on type 'unknown'."},{"start":1340,"length":10,"code":2339,"category":1,"messageText":"Property 'senderName' does not exist on type 'unknown'."},{"start":1386,"length":10,"code":2339,"category":1,"messageText":"Property 'courseName' does not exist on type 'unknown'."}]],[3193,[{"start":12829,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'JwtTokenPayload' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type 'JwtTokenPayload' is not assignable to parameter of type 'Buffer'.","category":1,"code":2345,"next":[{"messageText":"Type 'JwtTokenPayload' is missing the following properties from type 'Buffer': slice, subarray, write, toJSON, and 97 more.","category":1,"code":2740}]}]},{"messageText":"Overload 3 of 3, '(payload: JwtTokenPayload, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]}]],[3194,[{"start":2425,"length":6,"code":2339,"category":1,"messageText":"Property 'userId' does not exist on type 'unknown'."},{"start":2438,"length":9,"code":2339,"category":1,"messageText":"Property 'sessionId' does not exist on type 'unknown'."},{"start":3677,"length":6,"code":2339,"category":1,"messageText":"Property 'userId' does not exist on type 'unknown'."}]],[3197,[{"start":1165,"length":3,"code":2339,"category":1,"messageText":"Property 'kid' does not exist on type 'unknown'."},{"start":1262,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":1598,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":1940,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2292,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2789,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2975,"length":3,"code":2339,"category":1,"messageText":"Property 'res' does not exist on type 'unknown'."},{"start":3039,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type '{ sub: string; email: string; role: string; roles: string[]; sid: string; }' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'sub' does not exist in type 'Buffer'.","category":1,"code":2353}]},{"messageText":"Overload 3 of 3, '(payload: { sub: string; email: string; role: string; roles: string[]; sid: string; }, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]},{"start":3462,"length":3,"code":2339,"category":1,"messageText":"Property 'res' does not exist on type 'unknown'."},{"start":4219,"length":3,"code":2339,"category":1,"messageText":"Property 'kid' does not exist on type 'unknown'."}]],[3211,[{"start":1302,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."},{"start":1874,"length":3,"code":2339,"category":1,"messageText":"Property 'req' does not exist on type 'unknown'."},{"start":2279,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."},{"start":2855,"length":3,"code":2339,"category":1,"messageText":"Property 'req' does not exist on type 'unknown'."},{"start":3320,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."},{"start":3431,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'questions' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Question[]' is not assignable to type 'QuestionType[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Question' is not assignable to type 'QuestionType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'correctAnswer' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322,"canonicalHead":{"code":2322,"messageText":"Type 'Question' is not assignable to type 'QuestionType'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'."}}]}]}},{"start":3606,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Assessment[]' is not assignable to type 'AssessmentType[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'questions' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Question[]' is not assignable to type 'QuestionType[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Question' is not assignable to type 'QuestionType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'correctAnswer' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322,"canonicalHead":{"code":2322,"messageText":"Type 'Question' is not assignable to type 'QuestionType'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'."}}]}]}]}}]],[3213,[{"start":3617,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'questions' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Question[]' is not assignable to type 'QuestionType[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Question' is not assignable to type 'QuestionType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'correctAnswer' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322,"canonicalHead":{"code":2322,"messageText":"Type 'Question' is not assignable to type 'QuestionType'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'."}}]}]}},{"start":4064,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'questions' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Question[]' is not assignable to type 'QuestionType[]'.","category":1,"code":2322,"next":[{"messageText":"Type 'Question' is not assignable to type 'QuestionType'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'correctAnswer' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322,"canonicalHead":{"code":2322,"messageText":"Type 'Question' is not assignable to type 'QuestionType'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'Assessment' is not assignable to type 'AssessmentType'."}}]}]}}]],[3215,[{"start":672,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."}]],[3216,[{"start":748,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."}]],[3220,[{"start":7237,"length":8,"code":2322,"category":1,"messageText":"Type 'unknown' is not assignable to type 'number'."},{"start":8262,"length":11,"code":2339,"category":1,"messageText":"Property 'definitions' does not exist on type 'unknown'."}]],[3225,[{"start":687,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":880,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1079,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2328,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2428,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2531,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2831,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2902,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2968,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3226,[{"start":717,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1306,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1413,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1727,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3227,[{"start":754,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1074,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1406,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3228,[{"start":728,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1199,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1503,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3229,[{"start":796,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1352,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1464,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1578,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1689,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1813,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2145,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3230,[{"start":982,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1200,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":1972,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":2048,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[3244,[{"start":3084,"length":14,"code":2339,"category":1,"messageText":"Property 'conversionRate' does not exist on type 'unknown'."},{"start":3165,"length":14,"code":2339,"category":1,"messageText":"Property 'conversionRate' does not exist on type 'unknown'."},{"start":3277,"length":5,"code":2339,"category":1,"messageText":"Property 'value' does not exist on type 'unknown'."}]],[3245,[{"start":2728,"length":24,"code":2339,"category":1,"messageText":"Property 'statisticallySignificant' does not exist on type 'unknown'."},{"start":3682,"length":15,"code":2339,"category":1,"messageText":"Property 'confidenceLevel' does not exist on type 'unknown'."},{"start":4633,"length":8,"code":2339,"category":1,"messageText":"Property 'variants' does not exist on type 'unknown'."},{"start":4954,"length":10,"code":2339,"category":1,"messageText":"Property 'sampleSize' does not exist on type 'unknown'."},{"start":5202,"length":26,"code":2339,"category":1,"messageText":"Property 'isStatisticallySignificant' does not exist on type 'unknown'."},{"start":9848,"length":24,"code":2339,"category":1,"messageText":"Property 'statisticallySignificant' does not exist on type 'unknown'."},{"start":11712,"length":5,"code":2339,"category":1,"messageText":"Property 'score' does not exist on type 'unknown'."},{"start":11904,"length":14,"messageText":"Operator '>' cannot be applied to types 'unknown' and 'number'.","category":1,"code":2365},{"start":11927,"length":5,"code":2339,"category":1,"messageText":"Property 'score' does not exist on type 'unknown'."},{"start":11935,"length":10,"messageText":"The right-hand side of an arithmetic operation must be of type 'any', 'number', 'bigint' or an enum type.","category":1,"code":2363}]],[3246,[{"start":4300,"length":24,"code":2339,"category":1,"messageText":"Property 'statisticallySignificant' does not exist on type 'unknown'."},{"start":4482,"length":24,"code":2339,"category":1,"messageText":"Property 'statisticallySignificant' does not exist on type 'unknown'."},{"start":10854,"length":8,"code":2339,"category":1,"messageText":"Property 'variants' does not exist on type 'unknown'."}]],[3247,[{"start":5793,"length":8,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'WinnerSelectionCriteria'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'WinnerSelectionCriteria': confidenceLevel, minimumSampleSize, effectSizeThreshold, durationThreshold","category":1,"code":2739}]}}]],[3299,[{"start":885,"length":5,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'AttributeValue'."},{"start":2398,"length":5,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'AttributeValue'."},{"start":3515,"length":10,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'Record' is not assignable to parameter of type 'Attributes | TimeInput'.","category":1,"code":2345,"next":[{"messageText":"Type 'Record' is missing the following properties from type 'Date': toDateString, toTimeString, toLocaleDateString, toLocaleTimeString, and 37 more.","category":1,"code":2740,"canonicalHead":{"code":2322,"messageText":"Type 'Record' is not assignable to type 'Date'."}}]}}]],[3315,[{"start":785,"length":17,"code":1238,"category":1,"messageText":{"messageText":"Unable to resolve signature of class decorator when called as an expression.","category":1,"code":1238,"next":[{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}]}}]],[3595,[{"start":1122,"length":4,"code":2339,"category":1,"messageText":"Property 'text' does not exist on type 'unknown'."}]],[3596,[{"start":1827,"length":10,"code":2339,"category":1,"messageText":"Property 'categories' does not exist on type 'unknown'."},{"start":1880,"length":6,"code":2339,"category":1,"messageText":"Property 'levels' does not exist on type 'unknown'."},{"start":1932,"length":9,"code":2339,"category":1,"messageText":"Property 'languages' does not exist on type 'unknown'."},{"start":1988,"length":11,"code":2339,"category":1,"messageText":"Property 'price_stats' does not exist on type 'unknown'."},{"start":2038,"length":12,"code":2339,"category":1,"messageText":"Property 'price_ranges' does not exist on type 'unknown'."},{"start":2099,"length":13,"code":2339,"category":1,"messageText":"Property 'rating_ranges' does not exist on type 'unknown'."}]],[3597,[{"start":5347,"length":10,"code":2339,"category":1,"messageText":"Property 'categories' does not exist on type 'unknown'."},{"start":5408,"length":6,"code":2339,"category":1,"messageText":"Property 'levels' does not exist on type 'unknown'."},{"start":5470,"length":12,"code":2339,"category":1,"messageText":"Property 'price_ranges' does not exist on type 'unknown'."},{"start":7473,"length":14,"code":2339,"category":1,"messageText":"Property 'total_searches' does not exist on type 'unknown'."},{"start":7531,"length":11,"code":2339,"category":1,"messageText":"Property 'top_queries' does not exist on type 'unknown'."},{"start":7596,"length":19,"code":2339,"category":1,"messageText":"Property 'zero_result_queries' does not exist on type 'unknown'."},{"start":7677,"length":18,"code":2339,"category":1,"messageText":"Property 'searches_over_time' does not exist on type 'unknown'."},{"start":7835,"length":8,"code":2339,"category":1,"messageText":"Property 'category' does not exist on type 'unknown'."},{"start":7911,"length":8,"code":2339,"category":1,"messageText":"Property 'category' does not exist on type 'unknown'."},{"start":8169,"length":5,"code":2339,"category":1,"messageText":"Property 'level' does not exist on type 'unknown'."},{"start":8239,"length":5,"code":2339,"category":1,"messageText":"Property 'level' does not exist on type 'unknown'."},{"start":8482,"length":5,"code":2339,"category":1,"messageText":"Property 'price' does not exist on type 'unknown'."},{"start":8544,"length":5,"code":2339,"category":1,"messageText":"Property 'price' does not exist on type 'unknown'."},{"start":8586,"length":8,"code":2339,"category":1,"messageText":"Property 'language' does not exist on type 'unknown'."},{"start":8662,"length":8,"code":2339,"category":1,"messageText":"Property 'language' does not exist on type 'unknown'."},{"start":8920,"length":12,"code":2339,"category":1,"messageText":"Property 'instructorId' does not exist on type 'unknown'."},{"start":8998,"length":12,"code":2339,"category":1,"messageText":"Property 'instructorId' does not exist on type 'unknown'."},{"start":11239,"length":7,"code":2339,"category":1,"messageText":"Property '_source' does not exist on type 'unknown'."},{"start":11268,"length":3,"code":2339,"category":1,"messageText":"Property '_id' does not exist on type 'unknown'."},{"start":11296,"length":6,"code":2339,"category":1,"messageText":"Property '_score' does not exist on type 'unknown'."},{"start":11332,"length":9,"code":2339,"category":1,"messageText":"Property 'highlight' does not exist on type 'unknown'."},{"start":12731,"length":8,"code":2339,"category":1,"messageText":"Property 'category' does not exist on type 'object'."},{"start":12881,"length":5,"code":2339,"category":1,"messageText":"Property 'level' does not exist on type 'object'."},{"start":13022,"length":8,"code":2339,"category":1,"messageText":"Property 'language' does not exist on type 'object'."},{"start":13173,"length":12,"code":2339,"category":1,"messageText":"Property 'instructorId' does not exist on type 'object'."},{"start":13344,"length":5,"code":2339,"category":1,"messageText":"Property 'price' does not exist on type 'object'."}]],[3599,[{"start":1505,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'unknown'."},{"start":3021,"length":5,"code":2339,"category":1,"messageText":"Property 'index' does not exist on type 'unknown'."}]],[3624,[{"start":4350,"length":3,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string'."},{"start":4386,"length":3,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string'."},{"start":4481,"length":3,"code":2345,"category":1,"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'string'."}]],[3637,[{"start":10587,"length":26,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":10925,"length":5,"code":2322,"category":1,"messageText":"Type 'unknown' is not assignable to type 'number'.","relatedInformation":[{"file":"./node_modules/bull/index.d.ts","start":12504,"length":5,"messageText":"The expected type comes from property 'delay' which is declared here on type 'JobOptions'","category":3,"code":6500}]},{"start":11976,"length":31,"messageText":"Spread types may only be created from object types.","category":1,"code":2698}]],[3663,[{"start":169,"length":5,"code":2339,"category":1,"messageText":"Property 'level' does not exist on type 'unknown'."},{"start":215,"length":4,"code":2339,"category":1,"messageText":"Property 'goal' does not exist on type 'unknown'."}]],[3665,[{"start":160,"length":7,"messageText":"Spread types may only be created from object types.","category":1,"code":2698},{"start":198,"length":10,"code":2339,"category":1,"messageText":"Property 'milestones' does not exist on type 'unknown'."}]],[3785,[{"start":1052,"length":7,"code":2322,"category":1,"messageText":"Type 'Record' is not assignable to type 'AxiosHeaders | (Partial & Partial<...>)'.","relatedInformation":[{"file":"./node_modules/axios/index.d.cts","start":13348,"length":7,"messageText":"The expected type comes from property 'headers' which is declared here on type 'AxiosRequestConfig'","category":3,"code":6500}]}]],[3793,[{"start":553,"length":7,"messageText":"Spread types may only be created from object types.","category":1,"code":2698}]],[3804,[{"start":3815,"length":43,"messageText":"Type 'unknown' must have a '[Symbol.iterator]()' method that returns an iterator.","category":1,"code":2488}]],[3806,[{"start":2845,"length":8,"code":2339,"category":1,"messageText":"Property 'andWhere' does not exist on type 'unknown'."},{"start":3997,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'unknown' is not assignable to type 'Record'.","category":1,"code":2322,"next":[{"messageText":"Index signature for type 'string' is missing in type '{}'.","category":1,"code":2329}]}}]],[3810,[{"start":710,"length":5,"code":2339,"category":1,"messageText":"Property 'width' does not exist on type 'unknown'."},{"start":728,"length":6,"code":2339,"category":1,"messageText":"Property 'height' does not exist on type 'unknown'."},{"start":829,"length":5,"code":2339,"category":1,"messageText":"Property 'width' does not exist on type 'unknown'."},{"start":873,"length":6,"code":2339,"category":1,"messageText":"Property 'height' does not exist on type 'unknown'."},{"start":1020,"length":7,"code":2339,"category":1,"messageText":"Property 'quality' does not exist on type 'unknown'."},{"start":1102,"length":7,"code":2339,"category":1,"messageText":"Property 'quality' does not exist on type 'unknown'."},{"start":1153,"length":6,"code":2339,"category":1,"messageText":"Property 'format' does not exist on type 'unknown'."},{"start":1196,"length":6,"code":2339,"category":1,"messageText":"Property 'format' does not exist on type 'unknown'."},{"start":1318,"length":7,"code":2339,"category":1,"messageText":"Property 'quality' does not exist on type 'unknown'."},{"start":1477,"length":7,"code":2339,"category":1,"messageText":"Property 'quality' does not exist on type 'unknown'."},{"start":1667,"length":7,"code":2339,"category":1,"messageText":"Property 'quality' does not exist on type 'unknown'."}]],[3811,[{"start":5179,"length":10,"code":2339,"category":1,"messageText":"Property 'cacheLevel' does not exist on type 'unknown'."}]],[4315,[{"start":1105,"length":8,"code":2322,"category":1,"messageText":{"messageText":"Type 'unknown' is not assignable to type 'MetadataParam'.","category":1,"code":2322,"next":[{"messageText":"Index signature for type 'string' is missing in type '{}'.","category":1,"code":2329}]},"relatedInformation":[{"file":"./node_modules/stripe/types/PaymentIntentsResource.d.ts","start":5476,"length":8,"messageText":"The expected type comes from property 'metadata' which is declared here on type 'PaymentIntentCreateParams'","category":3,"code":6500}]},{"start":2248,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":2310,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."}]],[4318,[{"start":3006,"length":15,"code":2339,"category":1,"messageText":"Property 'paymentIntentId' does not exist on type 'unknown'."},{"start":3304,"length":12,"code":2339,"category":1,"messageText":"Property 'clientSecret' does not exist on type 'unknown'."},{"start":3360,"length":14,"code":2339,"category":1,"messageText":"Property 'requiresAction' does not exist on type 'unknown'."},{"start":6106,"length":8,"code":2339,"category":1,"messageText":"Property 'refundId' does not exist on type 'unknown'."}]],[4321,[{"start":3426,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":5361,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."}]],[4322,[{"start":1889,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":2640,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":4999,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."}]],[4323,[{"start":1539,"length":7,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Record'.","category":1,"code":2345,"next":[{"messageText":"Index signature for type 'string' is missing in type '{}'.","category":1,"code":2329}]}}]],[4325,[{"start":3510,"length":5,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Error'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Error': name, message","category":1,"code":2739}]}},{"start":6476,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":6562,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'unknown'."},{"start":6644,"length":4,"code":2339,"category":1,"messageText":"Property 'type' does not exist on type 'unknown'."},{"start":6675,"length":4,"code":2339,"category":1,"messageText":"Property 'type' does not exist on type 'unknown'."},{"start":6790,"length":4,"code":2339,"category":1,"messageText":"Property 'data' does not exist on type 'unknown'."},{"start":6959,"length":4,"code":2339,"category":1,"messageText":"Property 'data' does not exist on type 'unknown'."},{"start":7109,"length":4,"code":2339,"category":1,"messageText":"Property 'data' does not exist on type 'unknown'."},{"start":7524,"length":4,"code":2339,"category":1,"messageText":"Property 'type' does not exist on type 'unknown'."}]],[4345,[{"start":1173,"length":9,"code":2339,"category":1,"messageText":"Property 'csrfToken' does not exist on type 'unknown'."},{"start":1396,"length":9,"code":2339,"category":1,"messageText":"Property 'csrfToken' does not exist on type 'unknown'."},{"start":2006,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":2057,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."}]],[4346,[{"start":2095,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":2146,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."}]],[4552,[{"start":1389,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Observable<{ success: true; message: string; data: unknown; metadata: Record; }>' is not assignable to type 'Observable>'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'operator' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'Operator; }>' is not assignable to type 'Operator>'.","category":1,"code":2322,"next":[{"messageText":"Type '{ success: true; message: string; data: unknown; metadata: Record; }' is not assignable to type 'ApiResponse'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'data' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}],"canonicalHead":{"code":2322,"messageText":"Type '{ success: true; message: string; data: unknown; metadata: Record; }' is not assignable to type 'ApiResponse'."}}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'Observable<{ success: true; message: string; data: unknown; metadata: Record; }>' is not assignable to type 'Observable>'."}}]}]}},{"start":1916,"length":7,"code":2339,"category":1,"messageText":"Property 'message' does not exist on type 'object & Record<\"data\", unknown>'."},{"start":1961,"length":8,"code":2339,"category":1,"messageText":"Property 'metadata' does not exist on type 'object & Record<\"data\", unknown>'."}]],[4557,[{"start":3188,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":3218,"length":4,"code":2349,"category":1,"messageText":{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}},{"start":3266,"length":7,"code":2339,"category":1,"messageText":"Property 'headers' does not exist on type 'unknown'."},{"start":3319,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":3356,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":3418,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":3476,"length":7,"code":2339,"category":1,"messageText":"Property 'session' does not exist on type 'unknown'."},{"start":3644,"length":6,"code":2339,"category":1,"messageText":"Property 'status' does not exist on type 'unknown'."},{"start":3760,"length":4,"code":2349,"category":1,"messageText":{"messageText":"This expression is not callable.","category":1,"code":2349,"next":[{"messageText":"Type '{}' has no call signatures.","category":1,"code":2757}]}}]],[4561,[{"start":454,"length":13,"code":2740,"category":1,"messageText":"Type '{}' is missing the following properties from type 'ConfigService, false>': internalConfig, isCacheEnabled, skipProcessEnv, cache, and 16 more.","canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type 'ConfigService, false>'."}},{"start":581,"length":15,"code":2345,"category":1,"messageText":{"messageText":"Argument of type 'unknown' is not assignable to parameter of type 'Repository'.","category":1,"code":2345,"next":[{"messageText":"Type '{}' is missing the following properties from type 'Repository': target, manager, metadata, createQueryBuilder, and 42 more.","category":1,"code":2740}]}}]],[4562,[{"start":1295,"length":3,"code":2339,"category":1,"messageText":"Property 'kid' does not exist on type 'unknown'."},{"start":1392,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":1635,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":1977,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2225,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2669,"length":26,"code":2339,"category":1,"messageText":"Property 'jwtAccessSecretVersionUsed' does not exist on type 'unknown'."},{"start":2855,"length":3,"code":2339,"category":1,"messageText":"Property 'res' does not exist on type 'unknown'."},{"start":2919,"length":9,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 3, '(payload: string, options?: Omit): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Argument of type '{ sub: string; email: string; role: UserRole; sid: string; }' is not assignable to parameter of type 'string'.","category":1,"code":2345}]},{"messageText":"Overload 2 of 3, '(payload: Buffer, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'sub' does not exist in type 'Buffer'.","category":1,"code":2353}]},{"messageText":"Overload 3 of 3, '(payload: { sub: string; email: string; role: UserRole; sid: string; }, options?: JwtSignOptions): Promise', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Property 'alg' is missing in type '{ kid: string; }' but required in type 'JwtHeader'.","category":1,"code":2741,"canonicalHead":{"code":2322,"messageText":"Type '{ kid: string; }' is not assignable to type 'JwtHeader'."}}]}]},"relatedInformation":[{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":3392,"length":3,"messageText":"'alg' is declared here.","category":3,"code":2728},{"file":"./node_modules/@types/jsonwebtoken/index.d.ts","start":1682,"length":6,"messageText":"The expected type comes from property 'header' which is declared here on type 'JwtSignOptions'","category":3,"code":6500}]},{"start":3323,"length":3,"code":2339,"category":1,"messageText":"Property 'res' does not exist on type 'unknown'."},{"start":4027,"length":3,"code":2339,"category":1,"messageText":"Property 'kid' does not exist on type 'unknown'."}]],[4563,[{"start":1814,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}},{"start":3363,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}},{"start":4148,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}},{"start":4692,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}},{"start":5246,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}},{"start":5941,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type '(target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor) => TypedPropertyDescriptor' is not assignable to type 'MethodDecorator'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'descriptor' and 'descriptor' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'.","category":1,"code":2322,"next":[{"messageText":"Types of property 'set' are incompatible.","category":1,"code":2326,"next":[{"messageText":"Type '(value: T) => void' is not assignable to type '(value: unknown) => void'.","category":1,"code":2322,"next":[{"messageText":"Types of parameters 'value' and 'value' are incompatible.","category":1,"code":2328,"next":[{"messageText":"Type 'unknown' is not assignable to type 'T'.","category":1,"code":2322,"next":[{"messageText":"'T' could be instantiated with an arbitrary type which could be unrelated to 'unknown'.","category":1,"code":5082}]}]}],"canonicalHead":{"code":2322,"messageText":"Type 'TypedPropertyDescriptor' is not assignable to type 'TypedPropertyDescriptor'."}}]}]}]}]}}]],[4564,[{"start":5265,"length":6,"code":2339,"category":1,"messageText":"Property 'params' does not exist on type 'unknown'."},{"start":5281,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":5296,"length":4,"code":2339,"category":1,"messageText":"Property 'body' does not exist on type 'unknown'."},{"start":5387,"length":5,"code":2339,"category":1,"messageText":"Property 'route' does not exist on type 'unknown'."},{"start":5410,"length":3,"code":2339,"category":1,"messageText":"Property 'url' does not exist on type 'unknown'."},{"start":5461,"length":6,"code":2339,"category":1,"messageText":"Property 'params' does not exist on type 'unknown'."},{"start":5521,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":5566,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[4567,[{"start":4361,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":4604,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":4764,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[4568,[{"start":9150,"length":11,"messageText":"Property 'action_type' does not exist on type '{}'.","category":1,"code":2339},{"start":9163,"length":10,"messageText":"Property 'parameters' does not exist on type '{}'.","category":1,"code":2339},{"start":9282,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."},{"start":9536,"length":5,"code":2339,"category":1,"messageText":"Property 'query' does not exist on type 'unknown'."}]],[4570,[{"start":3455,"length":4,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 4, '(entities: DeepPartial[], options?: SaveOptions): Promise<(DeepPartial & Payment)[]>', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'userId' does not exist in type 'DeepPartial[]'.","category":1,"code":2353}]},{"messageText":"Overload 2 of 4, '(entity: DeepPartial, options?: SaveOptions): Promise & Payment>', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322}]}]},"relatedInformation":[{"start":662,"length":6,"messageText":"The expected type comes from property 'status' which is declared here on type 'DeepPartial'","category":3,"code":6500}]},{"start":3792,"length":4,"code":2769,"category":1,"messageText":{"messageText":"No overload matches this call.","category":1,"code":2769,"next":[{"messageText":"Overload 1 of 4, '(entities: DeepPartial[], options?: SaveOptions): Promise<(DeepPartial & Invoice)[]>', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Object literal may only specify known properties, and 'userId' does not exist in type 'DeepPartial[]'.","category":1,"code":2353}]},{"messageText":"Overload 2 of 4, '(entity: DeepPartial, options?: SaveOptions): Promise & Invoice>', gave the following error.","category":1,"code":2772,"next":[{"messageText":"Type 'unknown' is not assignable to type 'string'.","category":1,"code":2322}]}]},"relatedInformation":[{"start":853,"length":6,"messageText":"The expected type comes from property 'status' which is declared here on type 'DeepPartial'","category":3,"code":6500}]},{"start":3878,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4368,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4583,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4753,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4777,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4840,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":4879,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type 'DeepPartial[]'."},{"start":6306,"length":2,"code":2339,"category":1,"messageText":"Property 'id' does not exist on type '{ userId: string; data: unknown; createdAt: Date; }'."},{"start":8148,"length":6,"code":2740,"category":1,"messageText":"Type '{}' is missing the following properties from type 'Repository': target, manager, metadata, createQueryBuilder, and 42 more.","canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type 'Repository'."}}]],[4580,[{"start":745,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[4584,[{"start":2005,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[4585,[{"start":2040,"length":4,"code":2339,"category":1,"messageText":"Property 'user' does not exist on type 'unknown'."}]],[4587,[{"start":625,"length":7,"code":2339,"category":1,"messageText":"Property 'loaders' does not exist on type 'unknown'."}]],[4594,[{"start":4513,"length":6,"code":2322,"category":1,"messageText":{"messageText":"Type 'Record' is not assignable to type 'Record'.","category":1,"code":2322,"next":[{"messageText":"'string' index signatures are incompatible.","category":1,"code":2634,"next":[{"messageText":"Type '{}' is missing the following properties from type '{ state: CircuitState; failures: number; lastFailureTime: number; }': state, failures, lastFailureTime","category":1,"code":2739,"canonicalHead":{"code":2322,"messageText":"Type 'unknown' is not assignable to type '{ state: CircuitState; failures: number; lastFailureTime: number; }'."}}]}]}}]]],"affectedFilesPendingEmit":[3247,3248,3242,3244,3245,3240,3239,3241,3238,3243,3246,439,4351,438,3153,3155,3152,4558,4559,3148,3149,3147,3146,3144,3145,3150,3154,3151,3201,3191,3204,3192,4560,3190,3202,3203,4561,3194,3205,3193,3127,2575,3165,1703,2576,3169,4562,3197,2505,2507,2420,2498,2418,2500,2503,2504,2276,2278,2273,2274,2277,2275,2499,2419,2502,2506,2501,3314,3315,3124,3316,3125,4563,4564,3135,3134,3313,3812,3816,4024,3814,3815,1760,3813,3810,4023,3811,2577,2578,2512,2537,2508,2569,2511,2510,2509,4565,3119,1702,1734,1733,4346,4347,4344,1614,4566,4567,4568,4328,1611,1612,1613,2589,4569,3120,4349,4570,3128,2590,1726,2567,4571,1731,4572,4551,4553,1609,4552,1730,4574,4573,1728,4345,1732,4577,4576,1719,1729,4348,4350,2240,4332,1725,3784,3121,2624,3122,3118,2568,4578,3113,4554,4556,3600,1690,1727,4579,3142,3143,3136,3133,3131,3139,3137,3132,3141,2571,2573,2572,2570,4580,3140,3138,2584,2585,2579,2582,2583,2580,2581,3645,3623,3644,3624,3641,3637,3642,3622,3635,3625,3614,3617,4581,3627,3636,3626,3615,3618,3639,3649,3628,3603,3605,3632,3630,3634,3607,3609,3621,3647,3602,4582,3612,3613,3604,3631,3608,3620,3606,3611,3610,3629,3633,3648,3643,3616,3638,3619,3640,3646,3318,3322,3320,3323,3317,3319,3321,3657,3659,3652,3654,3650,3653,3655,3651,3661,3662,3660,3658,3656,4584,4585,3221,3212,3210,3209,4587,3217,3216,4588,3213,3211,3214,3215,3219,3220,3208,3206,3207,1720,1721,1718,4550,3667,3668,3666,3665,3664,3663,4334,4335,4336,4337,4338,4339,4330,4331,4333,4340,4342,4343,4341,4557,2268,2272,2266,2269,2247,2242,2271,2241,2244,2245,2246,2248,2267,4594,4593,4592,4595,4591,4590,3223,3222,3233,3236,3235,3237,3231,3234,3232,3225,3226,3227,3228,3229,3230,4596,3224,3774,3773,3769,3771,3770,3776,3775,3772,1602,1610,1591,1604,1608,1607,1606,1605,1603,3168,3199,3188,3167,3166,3171,3198,3170,3200,3189,3172,3301,3249,3251,3250,3300,3303,3304,3302,3299,3783,3788,3787,3789,3785,3786,4027,4031,4030,4606,4029,4025,4028,4026,4317,4319,4329,4318,4607,4316,4315,4327,4326,4610,4320,4608,4609,4324,4321,4325,4323,4322,2621,2586,2587,2593,2591,2625,2622,2626,2588,2623,2592,3309,3310,4613,4612,3311,3307,3305,4611,3306,3312,3308,3595,3596,3599,3598,3601,3597,3794,3793,3791,3795,3790,3792,1724,1723,1737,1735,1736,1738,1740,1739,3804,3801,3802,4614,3807,3800,3798,3797,3799,3796,3808,3806,3805,3809,3803,3114,3123,3115,2574,3129,3130,3126],"version":"6.0.3"} \ No newline at end of file